feat: manually mirror opencoze's code from bytedance
Change-Id: I09a73aadda978ad9511264a756b2ce51f5761adf
This commit is contained in:
@@ -0,0 +1,47 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { memo } from 'react';
|
||||
|
||||
import { isEqual, isFunction, noop, omitBy } from 'lodash-es';
|
||||
import { typeSafeJsonParse } from '@coze-common/chat-area-utils';
|
||||
|
||||
import { isWorkflowNodeData } from './utils';
|
||||
import { type RenderNodeEntryProps } from './type';
|
||||
import { QuestionNodeRender } from './question-node-render';
|
||||
import { InputNodeRender } from './input-node-render';
|
||||
|
||||
const BaseComponent: React.FC<RenderNodeEntryProps> = ({
|
||||
message,
|
||||
...restProps
|
||||
}) => {
|
||||
const data = typeSafeJsonParse(message.content, noop);
|
||||
if (!isWorkflowNodeData(data)) {
|
||||
return 'card content is not supported';
|
||||
}
|
||||
|
||||
if (data.content_type === 'option') {
|
||||
return <QuestionNodeRender data={data} message={message} {...restProps} />;
|
||||
}
|
||||
if (data.content_type === 'form_schema') {
|
||||
return <InputNodeRender data={data} message={message} {...restProps} />;
|
||||
}
|
||||
return 'content type is not supported';
|
||||
};
|
||||
|
||||
export const WorkflowRenderEntry = memo(BaseComponent, (prevProps, nextProps) =>
|
||||
isEqual(omitBy(prevProps, isFunction), omitBy(nextProps, isFunction)),
|
||||
);
|
||||
@@ -0,0 +1,103 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { useState } from 'react';
|
||||
|
||||
import { noop } from 'lodash-es';
|
||||
import { produce } from 'immer';
|
||||
import { typeSafeJsonParse } from '@coze-common/chat-area-utils';
|
||||
import { I18n } from '@coze-arch/i18n';
|
||||
import { Button, Input, Space, Typography } from '@coze-arch/coze-design';
|
||||
|
||||
import {
|
||||
isInputWorkflowNodeContent,
|
||||
isInputWorkflowNodeContentLikelyArray,
|
||||
} from './utils';
|
||||
import { type InputRenderNodeProps } from './type';
|
||||
import { NodeWrapperUI } from './node-wrapper-ui';
|
||||
|
||||
export const InputNodeRender: React.FC<InputRenderNodeProps> = ({
|
||||
data,
|
||||
onCardSendMsg,
|
||||
readonly,
|
||||
isDisable,
|
||||
message,
|
||||
}) => {
|
||||
const [inputData, setInputData] = useState<Record<string, string>>({});
|
||||
const [hasSend, setHasSend] = useState(false);
|
||||
const disabled = readonly || isDisable || hasSend;
|
||||
const parsedContent = typeSafeJsonParse(data.content, noop);
|
||||
|
||||
if (!isInputWorkflowNodeContentLikelyArray(parsedContent)) {
|
||||
return 'input node content is not supported';
|
||||
}
|
||||
|
||||
const validContent = parsedContent.filter(isInputWorkflowNodeContent);
|
||||
|
||||
return (
|
||||
<NodeWrapperUI>
|
||||
<Space spacing={12} vertical className="w-full">
|
||||
{validContent.map((item, index) => (
|
||||
<Space
|
||||
align="start"
|
||||
className="w-full"
|
||||
spacing={6}
|
||||
vertical
|
||||
key={item.name + index}
|
||||
>
|
||||
<Typography.Text ellipsis className="text-lg !font-medium">
|
||||
{item?.name}
|
||||
</Typography.Text>
|
||||
<Input
|
||||
disabled={disabled || hasSend}
|
||||
value={inputData[item.name]}
|
||||
onChange={value => {
|
||||
setInputData(
|
||||
produce(draft => {
|
||||
draft[item.name] = value;
|
||||
}),
|
||||
);
|
||||
}}
|
||||
/>
|
||||
</Space>
|
||||
))}
|
||||
|
||||
<Button
|
||||
className="w-full"
|
||||
disabled={disabled}
|
||||
onClick={() => {
|
||||
if (disabled) {
|
||||
return;
|
||||
}
|
||||
setHasSend(true);
|
||||
onCardSendMsg?.({
|
||||
message,
|
||||
extra: {
|
||||
msg:
|
||||
validContent
|
||||
.map(item => `${item.name}:${inputData[item.name] || ''}`)
|
||||
.join('\n') || '',
|
||||
mentionList: [],
|
||||
},
|
||||
});
|
||||
}}
|
||||
>
|
||||
{I18n.t('workflow_detail_title_testrun_submit')}
|
||||
</Button>
|
||||
</Space>
|
||||
</NodeWrapperUI>
|
||||
);
|
||||
};
|
||||
@@ -0,0 +1,23 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { type PropsWithChildren } from 'react';
|
||||
|
||||
export const NodeWrapperUI: React.FC<PropsWithChildren> = ({ children }) => (
|
||||
<div className="overflow-hidden w-full min-w-[282px] max-w-[546px]">
|
||||
{children}
|
||||
</div>
|
||||
);
|
||||
@@ -0,0 +1,57 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { Button, Space, Typography } from '@coze-arch/coze-design';
|
||||
|
||||
import { type QuestionRenderNodeProps } from './type';
|
||||
import { NodeWrapperUI } from './node-wrapper-ui';
|
||||
|
||||
export const QuestionNodeRender: React.FC<QuestionRenderNodeProps> = ({
|
||||
data,
|
||||
onCardSendMsg,
|
||||
readonly,
|
||||
isDisable,
|
||||
message,
|
||||
}) => {
|
||||
const disabled = readonly || isDisable;
|
||||
return (
|
||||
<NodeWrapperUI>
|
||||
<Space className="w-full" vertical spacing={12} align="start">
|
||||
<Typography.Text ellipsis className="text-18px">
|
||||
{data.content.question}
|
||||
</Typography.Text>
|
||||
<Space className="w-full" vertical spacing={16}>
|
||||
{data.content.options.map((option, index) => (
|
||||
<Button
|
||||
key={option.name + index}
|
||||
className="w-full"
|
||||
color="primary"
|
||||
disabled={disabled}
|
||||
onClick={() =>
|
||||
onCardSendMsg?.({
|
||||
message,
|
||||
extra: { msg: option.name, mentionList: [] },
|
||||
})
|
||||
}
|
||||
>
|
||||
{option.name}
|
||||
</Button>
|
||||
))}
|
||||
</Space>
|
||||
</Space>
|
||||
</NodeWrapperUI>
|
||||
);
|
||||
};
|
||||
@@ -0,0 +1,60 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import {
|
||||
type IEventCallbacks,
|
||||
type IMessage,
|
||||
} from '@coze-common/chat-uikit-shared';
|
||||
|
||||
export interface QuestionWorkflowNode {
|
||||
type: 'question';
|
||||
content_type: 'option';
|
||||
content: {
|
||||
question: string;
|
||||
options: { name: string }[];
|
||||
};
|
||||
}
|
||||
|
||||
type StringifyInputWorkflowNodeContent = string;
|
||||
|
||||
export interface InputWorkflowNode {
|
||||
content_type: 'form_schema';
|
||||
/** 嵌套的 stringify 数据, 需要二次 parse */
|
||||
content: StringifyInputWorkflowNodeContent;
|
||||
}
|
||||
|
||||
export interface InputWorkflowNodeContent {
|
||||
type: string;
|
||||
name: string;
|
||||
}
|
||||
|
||||
export type WorkflowNode = QuestionWorkflowNode | InputWorkflowNode;
|
||||
|
||||
interface RenderNodeBaseProps extends Pick<IEventCallbacks, 'onCardSendMsg'> {
|
||||
isDisable: boolean | undefined;
|
||||
readonly: boolean | undefined;
|
||||
}
|
||||
export interface RenderNodeEntryProps extends RenderNodeBaseProps {
|
||||
message: IMessage;
|
||||
}
|
||||
|
||||
export interface QuestionRenderNodeProps extends RenderNodeEntryProps {
|
||||
data: QuestionWorkflowNode;
|
||||
}
|
||||
|
||||
export interface InputRenderNodeProps extends RenderNodeEntryProps {
|
||||
data: InputWorkflowNode;
|
||||
}
|
||||
@@ -0,0 +1,31 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { isObject } from 'lodash-es';
|
||||
|
||||
import { type InputWorkflowNodeContent, type WorkflowNode } from './type';
|
||||
|
||||
export const isWorkflowNodeData = (value: unknown): value is WorkflowNode =>
|
||||
isObject(value) && 'content' in value && 'content_type' in value;
|
||||
|
||||
export const isInputWorkflowNodeContent = (
|
||||
value: unknown,
|
||||
): value is InputWorkflowNodeContent =>
|
||||
isObject(value) && 'type' in value && 'name' in value;
|
||||
|
||||
export const isInputWorkflowNodeContentLikelyArray = (
|
||||
value: unknown,
|
||||
): value is unknown[] => Array.isArray(value);
|
||||
@@ -0,0 +1,87 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { ContentBoxType } from '@coze-common/chat-uikit-shared';
|
||||
import {
|
||||
ContentBox,
|
||||
type EnhancedContentConfig,
|
||||
ContentType,
|
||||
} from '@coze-common/chat-uikit';
|
||||
import {
|
||||
PluginScopeContextProvider,
|
||||
usePluginCustomComponents,
|
||||
type ComponentTypesMap,
|
||||
} from '@coze-common/chat-area';
|
||||
|
||||
import { WorkflowRenderEntry } from './components';
|
||||
|
||||
const defaultEnable = (value?: boolean) => {
|
||||
if (typeof value === 'undefined') {
|
||||
return true;
|
||||
}
|
||||
return value;
|
||||
};
|
||||
|
||||
export const WorkflowRender: ComponentTypesMap['contentBox'] = props => {
|
||||
const customTextMessageInnerTopSlotList = usePluginCustomComponents(
|
||||
'TextMessageInnerTopSlot',
|
||||
);
|
||||
const enhancedContentConfigList: EnhancedContentConfig[] = [
|
||||
{
|
||||
rule: ({ contentType, contentConfigs }) => {
|
||||
const isCardEnable = defaultEnable(
|
||||
contentConfigs?.[ContentBoxType.CARD]?.enable,
|
||||
);
|
||||
return contentType === ContentType.Card && isCardEnable;
|
||||
},
|
||||
render: ({ message, eventCallbacks, contentConfigs, options }) => {
|
||||
const { isCardDisabled, readonly } = options;
|
||||
|
||||
const { onCardSendMsg } = eventCallbacks ?? {};
|
||||
|
||||
return (
|
||||
<WorkflowRenderEntry
|
||||
message={message}
|
||||
onCardSendMsg={onCardSendMsg}
|
||||
readonly={readonly}
|
||||
isDisable={isCardDisabled}
|
||||
/>
|
||||
);
|
||||
},
|
||||
},
|
||||
];
|
||||
return (
|
||||
<ContentBox
|
||||
enhancedContentConfigList={enhancedContentConfigList}
|
||||
multimodalTextContentAddonTop={
|
||||
<>
|
||||
{customTextMessageInnerTopSlotList.map(
|
||||
// eslint-disable-next-line @typescript-eslint/naming-convention -- 符合预期的命名
|
||||
({ pluginName, Component }, index) => (
|
||||
<PluginScopeContextProvider
|
||||
pluginName={pluginName}
|
||||
key={pluginName}
|
||||
>
|
||||
<Component key={index} message={props.message} />
|
||||
</PluginScopeContextProvider>
|
||||
),
|
||||
)}
|
||||
</>
|
||||
}
|
||||
{...props}
|
||||
/>
|
||||
);
|
||||
};
|
||||
@@ -0,0 +1,17 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
export { WorkflowRender } from './components/workflow-render';
|
||||
17
frontend/packages/common/chat-area/chat-workflow-render/src/typings.d.ts
vendored
Normal file
17
frontend/packages/common/chat-area/chat-workflow-render/src/typings.d.ts
vendored
Normal file
@@ -0,0 +1,17 @@
|
||||
/*
|
||||
* Copyright 2025 coze-dev Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
/// <reference types='@coze-arch/bot-typings' />
|
||||
Reference in New Issue
Block a user