Skip to content

Commit 81b940c

Browse files
corinagumaacebo
andauthored
[devtools] Fixed pasted JSON card render (#78)
* Fixed pasted JSON card render * Update ComposeBoxToolbar.tsx * Update package.json --------- Co-authored-by: Corina Gum <> Co-authored-by: Alex Acebo <[email protected]>
1 parent 4198f84 commit 81b940c

File tree

4 files changed

+40
-31
lines changed

4 files changed

+40
-31
lines changed

packages/devtools/src/components/AttachmentsContainer/AttachmentsContainer.tsx

+5-11
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { memo } from 'react';
1+
import { memo, useCallback } from 'react';
22
import { Button, Image } from '@fluentui/react-components';
33
import { Dismiss20Regular } from '@fluentui/react-icons/lib/fonts';
44

@@ -20,13 +20,7 @@ const AttachmentItem = memo(
2020
}) => {
2121
const classes = useClasses();
2222

23-
// Create a stable key based on content
24-
const contentKey =
25-
typeof attachment.content === 'object'
26-
? JSON.stringify(attachment.content).substring(0, 20)
27-
: String(attachment.content).substring(0, 20);
28-
29-
const renderAttachmentContent = () => {
23+
const renderAttachmentContent = useCallback(() => {
3024
switch (attachment.type) {
3125
case 'card':
3226
return attachment.content && <AdaptiveCard value={attachment.content} />;
@@ -45,10 +39,10 @@ const AttachmentItem = memo(
4539
default:
4640
return <div>{attachment.name || 'Attachment'}</div>;
4741
}
48-
};
42+
}, [attachment.content, attachment.type, attachment.name, classes]);
4943

5044
return (
51-
<div key={`attachment-${index}-${contentKey}`} className={classes.inlineAttachmentCard}>
45+
<div className={classes.inlineAttachmentCard}>
5246
{showRemoveButton && (
5347
<Button
5448
appearance="transparent"
@@ -85,7 +79,7 @@ const AttachmentsContainer = memo(
8579
<div className={classes.inlineAttachmentsContainer}>
8680
{attachments.map((attachment, index) => (
8781
<AttachmentItem
88-
key={index}
82+
key={`${attachment.type}-${index}`}
8983
attachment={attachment}
9084
index={index}
9185
onRemove={onRemoveAttachment}

packages/devtools/src/components/Card/AdaptiveCard.tsx

+6-4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { ComponentProps } from 'react';
1+
import { ComponentProps, memo } from 'react';
22
import { makeStyles, mergeClasses, tokens } from '@fluentui/react-components';
33
import * as cards from '@microsoft/spark.cards';
44

@@ -21,8 +21,7 @@ const useAdaptiveCardStyles = makeStyles({
2121
},
2222
});
2323

24-
export default function AdaptiveCard({ value }: AdaptiveCardProps) {
25-
console.log('AC value', value);
24+
const AdaptiveCard = memo(({ value }: AdaptiveCardProps) => {
2625
const classes = useAdaptiveCardStyles();
2726
return (
2827
<div className={mergeClasses(classes.root)}>
@@ -43,4 +42,7 @@ export default function AdaptiveCard({ value }: AdaptiveCardProps) {
4342
)}
4443
</div>
4544
);
46-
}
45+
});
46+
47+
export default AdaptiveCard;
48+
AdaptiveCard.displayName = 'AdaptiveCard';

packages/devtools/src/components/ComposeBox/ComposeBoxToolbar/ComposeBoxToolbar.tsx

+29-14
Original file line numberDiff line numberDiff line change
@@ -21,16 +21,19 @@ import {
2121
MenuTrigger,
2222
Textarea,
2323
Toast,
24-
ToastBody,
2524
ToastTitle,
25+
useToastController,
2626
Toolbar,
2727
ToolbarButton,
2828
ToolbarProps,
2929
ToolbarDivider,
3030
useId,
31-
useToastController,
3231
} from '@fluentui/react-components';
3332
import { useNavigate } from 'react-router';
33+
import { Card } from '@microsoft/spark.cards';
34+
35+
import { useCardStore } from '../../../stores/CardStore';
36+
import Logger from '../../Logger/Logger';
3437

3538
import { useClasses } from './ComposeBoxToolbar.styles';
3639

@@ -50,13 +53,15 @@ const ComposeBoxToolbar: FC<ComposeBoxToolbarProps> = ({
5053
}) => {
5154
const classes = useClasses();
5255
const navigate = useNavigate();
53-
const { dispatchToast } = useToastController();
5456
const [isDialogOpen, setIsDialogOpen] = useState(false);
5557
const [jsonInput, setJsonInput] = useState('');
5658
const [menuOpen, setMenuOpen] = useState(false);
5759
const dialogTitleId = useId('dialog-title');
5860
const textareaId = useId('json-input');
5961
const jsonInputRef = useRef<HTMLTextAreaElement>(null);
62+
const { currentCard } = useCardStore();
63+
const { dispatchToast } = useToastController();
64+
const childLog = Logger.child('ComposeBoxToolbar');
6065

6166
const handleNavigateToCards = () => {
6267
navigate('/cards');
@@ -65,41 +70,51 @@ const ComposeBoxToolbar: FC<ComposeBoxToolbarProps> = ({
6570

6671
const handleSaveJson = useCallback(() => {
6772
try {
68-
const card = JSON.parse(jsonInput);
73+
const card = JSON.parse(jsonInput) as Card;
6974

7075
if (onAttachment) {
7176
onAttachment({
7277
type: 'card',
7378
content: card,
79+
contentType: 'application/vnd.microsoft.card.adaptive',
7480
});
7581
} else if (onSend) {
7682
onSend([
7783
{
78-
contentType: 'application/vnd.microsoft.card.adaptive',
84+
type: 'card',
7985
content: card,
86+
contentType: 'application/vnd.microsoft.card.adaptive',
8087
},
8188
]);
8289
}
8390

8491
setIsDialogOpen(false);
92+
setJsonInput('');
8593
} catch (error) {
94+
childLog.debug('Failed to parse JSON:', error);
8695
dispatchToast(
8796
<Toast>
88-
<ToastTitle>Error</ToastTitle>
89-
<ToastBody>
90-
Failed to parse JSON: {error instanceof Error ? error.message : String(error)}
91-
</ToastBody>
97+
<ToastTitle>Failed to parse JSON. Please check your input and try again.</ToastTitle>
9298
</Toast>,
9399
{ intent: 'error' }
94100
);
95101
}
96-
}, [jsonInput, onAttachment, onSend, dispatchToast, setIsDialogOpen]);
102+
}, [childLog, jsonInput, onAttachment, onSend, dispatchToast]);
97103

98104
const handleSend = useCallback(() => {
99-
if (onSend && hasContent) {
100-
onSend();
105+
if (onSend) {
106+
if (currentCard) {
107+
onSend([
108+
{
109+
contentType: 'application/vnd.microsoft.card.adaptive',
110+
content: currentCard,
111+
},
112+
]);
113+
} else if (hasContent) {
114+
onSend();
115+
}
101116
}
102-
}, [onSend, hasContent]);
117+
}, [onSend, hasContent, currentCard]);
103118

104119
return (
105120
<Toolbar aria-label="New message actions" {...props} className={classes.toolbar}>
@@ -144,7 +159,7 @@ const ComposeBoxToolbar: FC<ComposeBoxToolbarProps> = ({
144159
Cancel
145160
</Button>
146161
<Button appearance="primary" onClick={handleSaveJson}>
147-
Attach Card
162+
Attach card
148163
</Button>
149164
</DialogActions>
150165
</DialogBody>

packages/graph/package.json

-2
Original file line numberDiff line numberDiff line change
@@ -25,8 +25,6 @@
2525
],
2626
"scripts": {
2727
"clean": "npx rimraf ./dist",
28-
"lint": "npx eslint",
29-
"lint:fix": "npx eslint --fix",
3028
"build": "npx cross-env NODE_OPTIONS='--max-old-space-size=16384' npx tsup && npx rimraf ./dist/index.d.mts",
3129
"test": "npx jest",
3230
"gen": "npm run gen:types && npm run gen:endpoints",

0 commit comments

Comments
 (0)