Skip to content

Commit

Permalink
Dev missing item tips (#17)
Browse files Browse the repository at this point in the history
  • Loading branch information
ikun97 authored Sep 20, 2024
1 parent 028c7ce commit 3444d7c
Show file tree
Hide file tree
Showing 9 changed files with 107 additions and 32 deletions.
23 changes: 4 additions & 19 deletions web/apps/web/src/components/workflow/import-modal/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,6 @@ const ImportModal: React.FC<{
onOpenChange: (open: boolean) => void;
}> = ({ open, onOpenChange }) => {
const [openTips, setOpenTips] = useBoolean(false);
const [isExpand, isExpandAction] = useBoolean(false);
const [fileUrl, setFileUrl] = useState<string | undefined>();
const [workflow, setWorkflow] = useState<Workflow | undefined>();
const { existedInfo, importWorkflow, setExistedInfo } = useWorkflowStore(
Expand Down Expand Up @@ -204,31 +203,17 @@ const ImportModal: React.FC<{
</AlertDialogTitle>
</AlertDialogHeader>
<AlertDialogDescription className="flex flex-col gap-1.5">
<Heading size="h2">Warning</Heading>
<Heading size="h2">Missing Items</Heading>
<Text size="sm" color="subtle">
The following items are missing.
<br />
Replace it with installed models/widgets, or contact us to add
it to the standard environment.
If this workflow would run in MyShell, please replace the
missing models/widgets with similar installed ones, or contact
us to add them to the standard environment.
</Text>
<div
className="flex items-center cursor-pointer"
onClick={() => isExpandAction.toggle()}>
<Text size="sm" color="subtler">
{isExpand ? 'Show Less' : 'View Detail'}
</Text>
<ChevronDownIcon
className={cn('w-4 h-4 ml-1.5 text-subtler', {
'rotate-180': isExpand,
})}
/>
</div>
<div
className={cn(
'flex-1 rounded-lg border border-default p-3 w-[340px] max-h-60 overflow-x-hidden',
{
hidden: !isExpand,
},
)}>
{!isEmpty(existedInfo.non_existed_models) ? (
<div>
Expand Down
19 changes: 16 additions & 3 deletions web/apps/web/src/components/workflow/node-card/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,9 +14,19 @@ const NodeCard: React.FC<
NodeData & {
children: React.ReactNode;
selected?: boolean;
has_error?: boolean;
mode?: string;
}
> = ({ id, children, runtime_data, selected, name, display_name, mode }) => {
> = ({
id,
children,
runtime_data,
selected,
name,
display_name,
mode,
has_error,
}) => {
const selectRef = useRef(null);
const active = useHover(selectRef);
const isUndefined = mode === 'undefined';
Expand All @@ -35,7 +45,9 @@ const NodeCard: React.FC<
},
{
'!border-critical':
runtime_data?.node_status === NodeStatusEnum.failed || isUndefined,
runtime_data?.node_status === NodeStatusEnum.failed ||
isUndefined ||
has_error,
},
{
'!border-brand': runtime_data?.node_status === NodeStatusEnum.start,
Expand All @@ -48,13 +60,14 @@ const NodeCard: React.FC<
<SchemaProvider
id={id}
name={name}
mode={mode}
display_name={display_name}
output={runtime_data?.output}>
<ContextMenu>
{isUndefined ? (
<div className="pb-2">
<Heading size="h1" color="critical">
Undefined Widget
Missing Widget
</Heading>
<Text color="critical">
Replace it with installed widgets, or contact us to add it to
Expand Down
15 changes: 14 additions & 1 deletion web/apps/web/src/components/workflow/node-form/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -44,10 +44,22 @@ interface NodeFormProps {
loading?: boolean;
modeMap?: Record<string, TFieldMode>;
onModeChange?: (name: string, mode: TFieldMode) => void;
onStatusChange?: (status: { [key: string]: string }) => void;
}

const NodeForm = forwardRef<FormRef, NodeFormProps>(
({ values, onChange, schema, loading, onModeChange, modeMap }, ref) => {
(
{
values,
onChange,
schema,
loading,
onModeChange,
onStatusChange,
modeMap,
},
ref,
) => {
const { schema: formSchema, formKey } = useSchemaContext(state => ({
schema: state.schema,
formKey: state.formKey,
Expand Down Expand Up @@ -78,6 +90,7 @@ const NodeForm = forwardRef<FormRef, NodeFormProps>(
schema={currentSchema}
modeMap={modeMap}
onModeChange={onModeChange}
onStatusChange={onStatusChange}
components={{
Input,
Select,
Expand Down
24 changes: 22 additions & 2 deletions web/apps/web/src/components/workflow/nodes/widget-node/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,13 @@ import {
import { TValues, TFieldMode } from '@shellagent/form-engine';
import { FormRef } from '@shellagent/ui';
import { useKeyPress } from 'ahooks';
import React, { useCallback, useRef, useEffect } from 'react';
import React, {
useCallback,
useRef,
useEffect,
useState,
useMemo,
} from 'react';
import { useShallow } from 'zustand/react/shallow';

import { useWorkflowState } from '@/stores/workflow/use-workflow-state';
Expand All @@ -21,6 +27,7 @@ import { useDuplicateState } from './hook/use-duplicate-state';
import { EventType, useEventEmitter } from '../../emitter';
import NodeCard from '../../node-card';
import NodeForm from '../../node-form';
import { some } from 'lodash-es';

const WidgetNode: React.FC<NodeProps<WidgetNodeType>> = ({
id,
Expand All @@ -29,6 +36,7 @@ const WidgetNode: React.FC<NodeProps<WidgetNodeType>> = ({
}) => {
const formRef = useRef<FormRef>(null);
const nodeRef = useRef<HTMLElement | null>(null);
const statusRef = useRef({});
const { duplicateState } = useDuplicateState(id);
const { onDelNode } = useReactFlowStore(state => ({
onDelNode: state.onDelNode,
Expand All @@ -37,6 +45,8 @@ const WidgetNode: React.FC<NodeProps<WidgetNodeType>> = ({
setCurrentCopyId: state.setCurrentCopyId,
}));

const [hasError, setHasError] = useState(false);

const {
setNodeData,
nodeData,
Expand Down Expand Up @@ -144,14 +154,24 @@ const WidgetNode: React.FC<NodeProps<WidgetNodeType>> = ({
}
});

const onStatusChange = (obj: { [key: string]: string }) => {
statusRef.current = { ...statusRef.current, ...obj };
setHasError(some(statusRef.current, value => value === 'missOption'));
};

return (
<NodeCard selected={selected} mode={nodeData[data.id]?.mode} {...data}>
<NodeCard
selected={selected}
mode={nodeData[data.id]?.mode}
has_error={some(statusRef.current, value => value === 'missOption')}
{...data}>
<NodeForm
ref={formRef}
loading={loading.getReactFlow}
values={nodeData[data.id]}
onChange={onChange}
onModeChange={onModeChange}
onStatusChange={onStatusChange}
modeMap={fieldsModeMap?.[data.id] || {}}
/>
</NodeCard>
Expand Down
10 changes: 6 additions & 4 deletions web/apps/web/src/stores/workflow/schema-provider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -52,14 +52,16 @@ export const useSchemaContext = <T,>(

export interface SchemaProviderProps {
id: string;
name: string | undefined;
display_name: string | undefined;
name?: string;
mode?: string;
display_name?: string;
children: React.ReactNode | React.ReactNode[];
output?: Record<string, any>;
}

export const SchemaProvider: React.FC<SchemaProviderProps> = ({
name = '',
mode,
display_name = '',
id,
children,
Expand Down Expand Up @@ -122,7 +124,7 @@ export const SchemaProvider: React.FC<SchemaProviderProps> = ({
memoized.schema = endSchema;
} else if (!isEmpty(currentWidgetSchema)) {
const schema = getSchemaByWidget({
name: display_name,
name: mode === 'undefined' ? name : display_name,
inputSchema: currentWidgetSchema?.input_schema,
outputSchema: currentWidgetSchema?.output_schema,
fieldsModeMap: currentFieldsModeMap,
Expand All @@ -142,7 +144,7 @@ export const SchemaProvider: React.FC<SchemaProviderProps> = ({
outputs: memoized.outputRefTypes,
});
return memoized;
}, [id, name, display_name, currentWidgetSchema, currentFieldsModeMap]);
}, [id, mode, name, display_name, currentWidgetSchema, currentFieldsModeMap]);

const formKey = useMemo(() => {
return `${JSON.stringify({
Expand Down
20 changes: 19 additions & 1 deletion web/apps/web/src/stores/workflow/workflow-store.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -395,7 +395,6 @@ export const createWorkflowStore = () => {
);
}
if (type === EventStatusEnum.workflow_end) {
get().flowInstance?.fitView();
set(
produce(state => {
state.loading.workflowRunning = false;
Expand Down Expand Up @@ -453,11 +452,30 @@ export const createWorkflowStore = () => {
return nodes;
});
if (data?.node_status === NodeStatusEnum.failed) {
get().flowInstance?.fitView({
minZoom: 0.8,
maxZoom: 1.5,
nodes: [{ id: data?.node_id as string }],
padding: 400,
});
// 设置为选中状态,放到最上层
get().flowInstance?.setNodes(nodes =>
nodes.map(node =>
node.id === data?.node_id
? {
...node,
selected: true,
}
: node,
),
);
set(
produce(state => {
state.loading.workflowRunning = false;
}),
);
} else {
get().flowInstance?.fitView();
}
}
},
Expand Down
19 changes: 18 additions & 1 deletion web/packages/form-engine/src/components/control/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import {
TooltipContent,
IconButton,
Switch,
Text,
} from '@shellagent/ui';
import { isEmpty, omit } from 'lodash-es';
import React, { useMemo, useRef, useState } from 'react';
Expand All @@ -37,7 +38,7 @@ type Mode = 'raw' | 'ui' | 'ref';
*/
const Control: React.FC<IControlProps> = props => {
const { name } = props;
const { fields, components, remove, modeMap, onModeChange } =
const { fields, components, remove, modeMap, onModeChange, onStatusChange } =
useFormEngineContext();
const { setValue, getValues } = useFormContext();
const { schema, state, parent } = fields[name] || {};
Expand Down Expand Up @@ -284,11 +285,27 @@ const Control: React.FC<IControlProps> = props => {
if (xOnChangePropsName) {
newField[xOnChangePropsName] = field.onChange;
}
const missOption =
xComponentProps?.options?.length &&
newField[valuePropsName] &&
!xComponentProps?.options?.find(
(item: { value: string }) =>
item.value === (newField[valuePropsName] as unknown as string),
);
if (missOption !== undefined) {
onStatusChange?.({ [name]: missOption ? 'missOption' : '' });
}
const FormItemWithDesc =
(!checked && xSwithable) || xHiddenControl ? null : (
<div className="grow">
<FormControl>{renderFormItem()}</FormControl>
{fieldState.error ? <FormMessage /> : null}
{missOption ? (
<Text color="critical">
{newField[valuePropsName] as unknown as string} is
undefined.
</Text>
) : null}
</div>
);

Expand Down
6 changes: 5 additions & 1 deletion web/packages/form-engine/src/components/provider/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ const FormEngineContext = createContext<{
reorder: (path: TPath, startIndex: number, endIndex: number) => void;
modeMap?: Record<string, TFieldMode>;
onModeChange?: (name: string, mode: TFieldMode) => void;
onStatusChange?: (obj: { [key: string]: string }) => void;
}>({
components: {},
fields: {},
Expand All @@ -42,6 +43,7 @@ export interface IFormEngineProviderProps {
layout?: 'Horizontal' | 'Vertical';
modeMap?: Record<string, TFieldMode>;
onModeChange?: (name: string, mode: TFieldMode) => void;
onStatusChange?: (obj: { [key: string]: string }) => void;
children: React.ReactNode | React.ReactNode[];
}

Expand All @@ -52,7 +54,8 @@ export const useFormEngineContext = () => {
const Counter: { [path: string]: number } = {};

export const FormEngineProvider: React.FC<IFormEngineProviderProps> = props => {
const { children, fields, components, parent, layout } = props;
const { children, fields, components, parent, layout, onStatusChange } =
props;
const { getValues, setValue } = useFormContext();

const [modeMap, setModeMap] = useImmer(props.modeMap || {});
Expand Down Expand Up @@ -178,6 +181,7 @@ export const FormEngineProvider: React.FC<IFormEngineProviderProps> = props => {
reorder,
modeMap,
onModeChange,
onStatusChange,
}}>
{children}
</FormEngineContext.Provider>
Expand Down
3 changes: 3 additions & 0 deletions web/packages/form-engine/src/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ export interface IFormEngineProps {
children?: React.ReactNode;
modeMap?: Record<string, TFieldMode>;
onModeChange?: (name: string, mode: TFieldMode) => void;
onStatusChange?: (obj: { [key: string]: string }) => void;
}

const FormEngine = forwardRef<FormRef, IFormEngineProps>((props, ref) => {
Expand All @@ -52,6 +53,7 @@ const FormEngine = forwardRef<FormRef, IFormEngineProps>((props, ref) => {
onModeChange,
onChange,
onSubmit,
onStatusChange,
} = props;

const [fields, setFields] = useState(createFields(schema, values));
Expand Down Expand Up @@ -107,6 +109,7 @@ const FormEngine = forwardRef<FormRef, IFormEngineProps>((props, ref) => {
<FormEngineProvider
modeMap={modeMap}
onModeChange={onModeChange}
onStatusChange={onStatusChange}
fields={fields}
parent={parent}
layout={layout}
Expand Down

0 comments on commit 3444d7c

Please sign in to comment.