diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 000000000..d70a5c3b4 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,6 @@ +{ + "name": "MaxKB", + "lockfileVersion": 3, + "requires": true, + "packages": {} +} diff --git a/ui/package.json b/ui/package.json index 4bc39ccc8..b70010026 100644 --- a/ui/package.json +++ b/ui/package.json @@ -14,6 +14,7 @@ }, "dependencies": { "@ctrl/tinycolor": "^4.1.0", + "@logicflow/core": "^1.2.27", "@vueuse/core": "^10.9.0", "axios": "^0.28.0", "cropperjs": "^1.6.2", diff --git a/ui/src/components/work-flow/common/app-node/index.ts b/ui/src/components/work-flow/common/app-node/index.ts new file mode 100644 index 000000000..8e2ae4af2 --- /dev/null +++ b/ui/src/components/work-flow/common/app-node/index.ts @@ -0,0 +1,165 @@ +import Components from '@/components' +import ElementPlus from 'element-plus' +import { HtmlNode, HtmlNodeModel } from '@logicflow/core' +import { createApp, h } from 'vue' +import directives from '@/directives' + +class AppNode extends HtmlNode { + isMounted + r + app + + constructor(props: any, VueNode: any) { + super(props) + this.isMounted = false + + this.r = h(VueNode, { + properties: props.model.properties, + nodeModel: props.model + }) + + this.app = createApp({ + render: () => this.r + }) + this.app.use(ElementPlus) + this.app.use(Components) + this.app.use(directives) + } + + setHtml(rootEl: HTMLElement) { + if (!this.isMounted) { + this.isMounted = true + const node = document.createElement('div') + rootEl.appendChild(node) + this.app.mount(node) + } else { + if (this.r && this.r.component) { + this.r.component.props.properties = this.props.model.getProperties() + } + } + } +} + +class AppNodeModel extends HtmlNodeModel { + /** + * 给model自定义添加字段方法 + */ + addField(item: any) { + this.properties.fields.unshift(item) + this.setAttributes() + // 为了保持节点顶部位置不变,在节点变化后,对节点进行一个位移,位移距离为添加高度的一半。 + this.move(0, 24 / 2) + // 更新节点连接边的path + this.incoming.edges.forEach((egde: any) => { + // 调用自定义的更新方案 + egde.updatePathByAnchor() + }) + this.outgoing.edges.forEach((edge: any) => { + // 调用自定义的更新方案 + edge.updatePathByAnchor() + }) + } + getOutlineStyle() { + const style = super.getOutlineStyle() + style.stroke = 'none' + if (style.hover) { + style.hover.stroke = 'none' + } + return style + } + // 如果不用修改锚地形状,可以重写颜色相关样式 + getAnchorStyle(anchorInfo: any) { + const style = super.getAnchorStyle(anchorInfo) + if (anchorInfo.type === 'left') { + style.fill = 'red' + style.hover.fill = 'transparent' + style.hover.stroke = 'transpanrent' + style.className = 'lf-hide-default' + } else { + style.fill = 'green' + } + return style + } + setHeight(height: number, inputContainerHeight: number, outputContainerHeight: number) { + this.height = height + inputContainerHeight + outputContainerHeight + 100 + this.baseHeight = height + this.inputContainerHeight = inputContainerHeight + this.outputContainerHeight = outputContainerHeight + + this.outgoing.edges.forEach((edge: any) => { + // 调用自定义的更新方案 + edge.updatePathByAnchor() + }) + this.incoming.edges.forEach((edge: any) => { + // 调用自定义的更新方案 + edge.updatePathByAnchor() + }) + } + setAttributes() { + this.width = 500 + + const circleOnlyAsTarget = { + message: '只允许从右边的锚点连出', + validate: (sourceNode: any, targetNode: any, sourceAnchor: any) => { + return sourceAnchor.type === 'right' + } + } + this.sourceRules.push(circleOnlyAsTarget) + this.targetRules.push({ + message: '只允许连接左边的锚点', + validate: (sourceNode: any, targetNode: any, sourceAnchor: any, targetAnchor: any) => { + return targetAnchor.type === 'left' + } + }) + } + getDefaultAnchor() { + const { + id, + x, + y, + width, + height, + properties: { input, output } + } = this + + if (this.baseHeight === undefined) { + this.baseHeight = 0 + } + if (this.inputContainerHeight === undefined) { + this.inputContainerHeight = 0 + } + if (this.height === undefined) { + this.height = 200 + } + if (this.outputContainerHeight === undefined) { + this.outputContainerHeight = 0 + } + + const anchors: any = [] + if (input) { + input.forEach((feild: any, index: any) => { + anchors.push({ + x: x - width / 2 + 10, + y: y - height / 2 + this.baseHeight + 35 + index * 24, + id: `${id}_${feild.key}_left`, + edgeAddable: false, + type: 'left' + }) + }) + } + if (output) { + output.forEach((feild: any, index: any) => { + anchors.push({ + x: x + width / 2 - 10, + y: y - height / 2 + this.baseHeight + this.inputContainerHeight + 30 + index * 24, + id: `${id}_${feild.key}_right`, + type: 'right' + }) + }) + } + + return anchors + } +} + +export { AppNodeModel, AppNode } diff --git a/ui/src/components/work-flow/common/edge/index.ts b/ui/src/components/work-flow/common/edge/index.ts new file mode 100644 index 000000000..df9c55089 --- /dev/null +++ b/ui/src/components/work-flow/common/edge/index.ts @@ -0,0 +1,62 @@ +import { BezierEdge, BezierEdgeModel } from '@logicflow/core' + +class CustomEdge2 extends BezierEdge {} + +class CustomEdgeModel2 extends BezierEdgeModel { + getEdgeStyle() { + const style = super.getEdgeStyle() + + // svg属性 + style.strokeWidth = 1 + style.stroke = '#ababac' + return style + } + /** + * 重写此方法,使保存数据是能带上锚点数据。 + */ + getData() { + const data: any = super.getData() + if (data) { + data.sourceAnchorId = this.sourceAnchorId + data.targetAnchorId = this.targetAnchorId + } + return data + } + /** + * 给边自定义方案,使其支持基于锚点的位置更新边的路径 + */ + updatePathByAnchor() { + // TODO + const sourceNodeModel = this.graphModel.getNodeModelById(this.sourceNodeId) + const sourceAnchor = sourceNodeModel + .getDefaultAnchor() + .find((anchor: any) => anchor.id === this.sourceAnchorId) + + const targetNodeModel = this.graphModel.getNodeModelById(this.targetNodeId) + const targetAnchor = targetNodeModel + .getDefaultAnchor() + .find((anchor: any) => anchor.id === this.targetAnchorId) + if (sourceAnchor && targetAnchor) { + const startPoint = { + x: sourceAnchor.x, + y: sourceAnchor.y + } + this.updateStartPoint(startPoint) + const endPoint = { + x: targetAnchor.x, + y: targetAnchor.y + } + this.updateEndPoint(endPoint) + } + + // 这里需要将原有的pointsList设置为空,才能触发bezier的自动计算control点。 + this.pointsList = [] + this.initPoints() + } +} + +export default { + type: 'app-edge', + view: CustomEdge2, + model: CustomEdgeModel2 +} diff --git a/ui/src/components/work-flow/common/menu/data.ts b/ui/src/components/work-flow/common/menu/data.ts new file mode 100644 index 000000000..bf19a69d9 --- /dev/null +++ b/ui/src/components/work-flow/common/menu/data.ts @@ -0,0 +1,24 @@ +const shapeList = [ + { + type: 'ai-chat-node', + text: '', + label: 'AI对话', + icon: '', + properties: { + height: 200, + stepName: 'AI对话', + input: [ + { + key: '输入' + } + ], + output: [ + { + key: '输出' + } + ] + } + } +] + +export { shapeList } diff --git a/ui/src/components/work-flow/common/menu/index.ts b/ui/src/components/work-flow/common/menu/index.ts new file mode 100644 index 000000000..a9076d115 --- /dev/null +++ b/ui/src/components/work-flow/common/menu/index.ts @@ -0,0 +1,115 @@ +import LogicFlow from '@logicflow/core' +import { shapeList } from './data' +type ShapeItem = { + type?: string + text?: string + icon?: string + label?: string + className?: string + disabled?: boolean + properties?: Record + callback?: (lf: LogicFlow, container: HTMLElement) => void +} + +class AppMenu { + lf: LogicFlow + shapeList: ShapeItem[] + panelEl?: HTMLDivElement + static pluginName = 'AppMenu' + domContainer?: HTMLElement + constructor({ lf }: { lf: LogicFlow }) { + this.lf = lf + this.lf.setPatternItems = (shapeList: Array) => { + this.setPatternItems(shapeList) + } + this.shapeList = shapeList + this.panelEl = undefined + this.domContainer = undefined + } + render(lf: LogicFlow, domContainer: HTMLElement) { + this.destroy() + if (!this.shapeList || this.shapeList.length === 0) { + // 首次render后失败后,后续调用setPatternItems支持渲染 + this.domContainer = domContainer + return + } + this.panelEl = document.createElement('div') + this.panelEl.className = 'lf-dndpanel' + this.shapeList.forEach((shapeItem) => { + this.panelEl?.appendChild(this.createDndItem(shapeItem)) + }) + domContainer.appendChild(this.panelEl) + this.domContainer = domContainer + } + destroy() { + if (this.domContainer && this.panelEl && this.domContainer.contains(this.panelEl)) { + this.domContainer.removeChild(this.panelEl) + } + } + setPatternItems(shapeList: Array) { + this.shapeList = shapeList + // 支持渲染后重新设置拖拽面板 + if (this.domContainer) { + this.render(this.lf, this.domContainer) + } + } + private createDndItem(shapeItem: ShapeItem): HTMLElement { + const el = document.createElement('div') + el.className = shapeItem.className ? `lf-dnd-item ${shapeItem.className}` : 'lf-dnd-item' + const shape = document.createElement('div') + shape.className = 'lf-dnd-shape' + + if (shapeItem.icon) { + shape.style.backgroundImage = `url(${shapeItem.icon})` + } + el.appendChild(shape) + if (shapeItem.label) { + const text = document.createElement('div') + text.innerText = shapeItem.label + text.className = 'lf-dnd-text' + el.appendChild(text) + } + if (shapeItem.disabled) { + el.classList.add('disabled') + // 保留callback的执行,可用于界面提示当前shapeItem的禁用状态 + el.onmousedown = () => { + if (shapeItem.callback && this.domContainer) { + shapeItem.callback(this.lf, this.domContainer) + } + } + return el + } + el.onmousedown = () => { + if (shapeItem.type) { + this.lf.dnd.startDrag({ + type: shapeItem.type, + properties: shapeItem.properties + }) + } + if (shapeItem.callback && this.domContainer) { + shapeItem.callback(this.lf, this.domContainer) + } + } + el.ondblclick = (e) => { + this.lf.graphModel.eventCenter.emit('dnd:panel-dbclick', { + e, + data: shapeItem + }) + } + el.onclick = (e) => { + this.lf.graphModel.eventCenter.emit('dnd:panel-click', { + e, + data: shapeItem + }) + } + el.oncontextmenu = (e) => { + this.lf.graphModel.eventCenter.emit('dnd:panel-contextmenu', { + e, + data: shapeItem + }) + } + return el + } +} + +export { AppMenu } diff --git a/ui/src/components/work-flow/common/node-container/index.vue b/ui/src/components/work-flow/common/node-container/index.vue new file mode 100644 index 000000000..102ef4c60 --- /dev/null +++ b/ui/src/components/work-flow/common/node-container/index.vue @@ -0,0 +1,170 @@ + + + diff --git a/ui/src/components/work-flow/index.vue b/ui/src/components/work-flow/index.vue new file mode 100644 index 000000000..a1082f564 --- /dev/null +++ b/ui/src/components/work-flow/index.vue @@ -0,0 +1,153 @@ + + + diff --git a/ui/src/components/work-flow/nodes/ai-chat-node/index.ts b/ui/src/components/work-flow/nodes/ai-chat-node/index.ts new file mode 100644 index 000000000..04581c63e --- /dev/null +++ b/ui/src/components/work-flow/nodes/ai-chat-node/index.ts @@ -0,0 +1,12 @@ +import ChatNodeVue from '@/flow/nodes/ai-chat-node/index.vue' +import { AppNode, AppNodeModel } from '@/flow/common/app-node/index' +class ChatNode extends AppNode { + constructor(props: any) { + super(props, ChatNodeVue) + } +} +export default { + type: 'ai-chat-node', + model: AppNodeModel, + view: ChatNode +} diff --git a/ui/src/components/work-flow/nodes/ai-chat-node/index.vue b/ui/src/components/work-flow/nodes/ai-chat-node/index.vue new file mode 100644 index 000000000..2698b3093 --- /dev/null +++ b/ui/src/components/work-flow/nodes/ai-chat-node/index.vue @@ -0,0 +1,72 @@ + + + diff --git a/ui/src/components/work-flow/nodes/search-dataset-node/index.ts b/ui/src/components/work-flow/nodes/search-dataset-node/index.ts new file mode 100644 index 000000000..4a8539034 --- /dev/null +++ b/ui/src/components/work-flow/nodes/search-dataset-node/index.ts @@ -0,0 +1,12 @@ +import SearchDatasetVue from './index.vue' +import { AppNode, AppNodeModel } from '@/flow/common/app-node/index' +class SearchDatasetNode extends AppNode { + constructor(props: any) { + super(props, SearchDatasetVue) + } +} +export default { + type: 'search-dataset-node', + model: AppNodeModel, + view: SearchDatasetNode +} diff --git a/ui/src/components/work-flow/nodes/search-dataset-node/index.vue b/ui/src/components/work-flow/nodes/search-dataset-node/index.vue new file mode 100644 index 000000000..786cfe539 --- /dev/null +++ b/ui/src/components/work-flow/nodes/search-dataset-node/index.vue @@ -0,0 +1,51 @@ + + +