@agentscope-ai/chat 1.1.45-beta.1766559488425 → 1.1.45-beta.1766631141555
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/components/ChatAnywhere/Input/UploadPopover.tsx +45 -24
- package/components/ChatAnywhere/Input/index.tsx +2 -2
- package/components/ChatAnywhere/Input/style.ts +10 -0
- package/lib/ChatAnywhere/Input/UploadPopover.js +57 -30
- package/lib/ChatAnywhere/Input/index.js +4 -2
- package/lib/ChatAnywhere/Input/style.js +3 -1
- package/package.json +2 -1
- package/bin/starter_webui/README.md +0 -75
- package/bin/starter_webui/eslint.config.js +0 -28
- package/bin/starter_webui/index.html +0 -12
- package/bin/starter_webui/package.json +0 -34
- package/bin/starter_webui/src/App.tsx +0 -20
- package/bin/starter_webui/src/components/Chat/OptionsPanel/FormItem.tsx +0 -37
- package/bin/starter_webui/src/components/Chat/OptionsPanel/OptionsEditor.tsx +0 -160
- package/bin/starter_webui/src/components/Chat/OptionsPanel/defaultConfig.ts +0 -41
- package/bin/starter_webui/src/components/Chat/OptionsPanel/index.tsx +0 -27
- package/bin/starter_webui/src/components/Chat/index.tsx +0 -45
- package/bin/starter_webui/src/components/Chat/sessionApi/index.ts +0 -53
- package/bin/starter_webui/src/main.tsx +0 -9
- package/bin/starter_webui/src/vite-env.d.ts +0 -4
- package/bin/starter_webui/tsconfig.app.json +0 -24
- package/bin/starter_webui/tsconfig.json +0 -7
- package/bin/starter_webui/tsconfig.node.json +0 -22
- package/bin/starter_webui/vite.config.ts +0 -11
|
@@ -1,44 +1,65 @@
|
|
|
1
|
+
import React, { useMemo, useRef, useState } from "react";
|
|
1
2
|
import { IconButton, Popover } from "@agentscope-ai/design";
|
|
2
3
|
import { PlusOutlined } from "@ant-design/icons";
|
|
3
|
-
import {
|
|
4
|
+
import { useProviderContext } from "@agentscope-ai/chat";
|
|
4
5
|
import { Flex, Upload } from "antd";
|
|
5
|
-
import React, { useMemo } from "react";
|
|
6
|
-
import { useEffect, useState } from "react";
|
|
7
6
|
|
|
8
7
|
export default function UploadPopover({
|
|
9
8
|
uploadPropsList
|
|
10
9
|
}: {
|
|
11
10
|
uploadPropsList: any[];
|
|
12
11
|
}) {
|
|
13
|
-
const
|
|
12
|
+
const { getPrefixCls } = useProviderContext();
|
|
13
|
+
const prefixCls = getPrefixCls('chat-anywhere-sender');
|
|
14
|
+
const [visible, setVisible] = useState(false);
|
|
15
|
+
const uploadRefs = useRef<Array<HTMLDivElement | null>>([]);
|
|
14
16
|
|
|
15
|
-
const
|
|
17
|
+
const popoverNodes = useMemo(() => {
|
|
16
18
|
return uploadPropsList.map((item, index) => {
|
|
17
19
|
return (
|
|
18
|
-
<
|
|
20
|
+
<div
|
|
19
21
|
key={index}
|
|
20
|
-
{
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
item.customRequest(v);
|
|
24
|
-
}
|
|
22
|
+
onClick={() => {
|
|
23
|
+
// 触发对应Upload组件的children的click事件
|
|
24
|
+
uploadRefs.current[index]?.click();
|
|
25
25
|
setVisible(false);
|
|
26
26
|
}}
|
|
27
|
-
|
|
27
|
+
>
|
|
28
|
+
{item.trigger}
|
|
29
|
+
</div>
|
|
30
|
+
);
|
|
31
|
+
})
|
|
32
|
+
}, [uploadPropsList]);
|
|
33
|
+
|
|
34
|
+
const nodes = useMemo(() => {
|
|
35
|
+
return uploadPropsList.map((item, index) => {
|
|
36
|
+
const { trigger, ...rest } = item;
|
|
37
|
+
return (
|
|
38
|
+
<Upload
|
|
39
|
+
key={index}
|
|
40
|
+
{...rest}
|
|
41
|
+
>
|
|
42
|
+
<div ref={(el) => (uploadRefs.current[index] = el)} />
|
|
43
|
+
</Upload>
|
|
28
44
|
)
|
|
29
45
|
});
|
|
30
46
|
}, [uploadPropsList]);
|
|
31
47
|
|
|
32
|
-
return
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
48
|
+
return (
|
|
49
|
+
<>
|
|
50
|
+
<Popover
|
|
51
|
+
placement='bottomLeft'
|
|
52
|
+
open={visible}
|
|
53
|
+
onOpenChange={setVisible}
|
|
54
|
+
content={<Flex vertical>
|
|
55
|
+
{popoverNodes}
|
|
56
|
+
</Flex>} trigger="click" styles={{ body: { padding: 4 } }}>
|
|
57
|
+
<IconButton
|
|
58
|
+
icon={<PlusOutlined />}
|
|
59
|
+
bordered={false}
|
|
60
|
+
/>
|
|
61
|
+
</Popover>
|
|
62
|
+
<div className={`${prefixCls}-upload-hidden-nodes`}>{nodes}</div>
|
|
63
|
+
</>
|
|
64
|
+
)
|
|
44
65
|
}
|
|
@@ -122,12 +122,12 @@ export default forwardRef(function (_, ref) {
|
|
|
122
122
|
}
|
|
123
123
|
},
|
|
124
124
|
showUploadList: false,
|
|
125
|
-
|
|
125
|
+
trigger,
|
|
126
126
|
}
|
|
127
127
|
});
|
|
128
128
|
|
|
129
129
|
if (uploadPropsList.length === 1) return (
|
|
130
|
-
<Upload {...uploadPropsList[0]}
|
|
130
|
+
<Upload {...uploadPropsList[0]}>{uploadPropsList[0].trigger}</Upload>
|
|
131
131
|
);
|
|
132
132
|
return <UploadPopover uploadPropsList={uploadPropsList} />
|
|
133
133
|
|
|
@@ -11,4 +11,14 @@ export default createGlobalStyle`
|
|
|
11
11
|
}
|
|
12
12
|
}
|
|
13
13
|
|
|
14
|
+
.${(p) => p.theme.prefixCls}-chat-anywhere-sender-upload-hidden-nodes {
|
|
15
|
+
position: absolute;
|
|
16
|
+
z-index: -999;
|
|
17
|
+
top: -100vh;
|
|
18
|
+
left: -100vw;
|
|
19
|
+
width: 0;
|
|
20
|
+
height: 0;
|
|
21
|
+
overflow: hidden;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
14
24
|
`;
|
|
@@ -1,57 +1,84 @@
|
|
|
1
1
|
function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
|
|
2
|
+
var _excluded = ["trigger"];
|
|
2
3
|
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
|
3
4
|
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
|
|
4
5
|
function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
5
6
|
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : String(i); }
|
|
6
7
|
function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
|
|
8
|
+
function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
|
|
9
|
+
function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }
|
|
7
10
|
function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
|
|
8
11
|
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
|
|
9
12
|
function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
|
|
10
13
|
function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i]; return arr2; }
|
|
11
14
|
function _iterableToArrayLimit(r, l) { var t = null == r ? null : "undefined" != typeof Symbol && r[Symbol.iterator] || r["@@iterator"]; if (null != t) { var e, n, i, u, a = [], f = !0, o = !1; try { if (i = (t = t.call(r)).next, 0 === l) { if (Object(t) !== t) return; f = !1; } else for (; !(f = (e = i.call(t)).done) && (a.push(e.value), a.length !== l); f = !0); } catch (r) { o = !0, n = r; } finally { try { if (!f && null != t.return && (u = t.return(), Object(u) !== u)) return; } finally { if (o) throw n; } } return a; } }
|
|
12
15
|
function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
|
|
16
|
+
import React, { useMemo, useRef, useState } from "react";
|
|
13
17
|
import { IconButton, Popover } from "@agentscope-ai/design";
|
|
14
18
|
import { PlusOutlined } from "@agentscope-ai/icons-override-antd";
|
|
15
|
-
import {
|
|
19
|
+
import { useProviderContext } from "../..";
|
|
16
20
|
import { Flex, Upload } from "antd";
|
|
17
|
-
import React, { useMemo } from "react";
|
|
18
21
|
import { jsx as _jsx } from "react/jsx-runtime";
|
|
22
|
+
import { Fragment as _Fragment } from "react/jsx-runtime";
|
|
23
|
+
import { jsxs as _jsxs } from "react/jsx-runtime";
|
|
19
24
|
export default function UploadPopover(_ref) {
|
|
20
25
|
var uploadPropsList = _ref.uploadPropsList;
|
|
21
|
-
var
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
+
var _useProviderContext = useProviderContext(),
|
|
27
|
+
getPrefixCls = _useProviderContext.getPrefixCls;
|
|
28
|
+
var prefixCls = getPrefixCls('chat-anywhere-sender');
|
|
29
|
+
var _useState = useState(false),
|
|
30
|
+
_useState2 = _slicedToArray(_useState, 2),
|
|
31
|
+
visible = _useState2[0],
|
|
32
|
+
setVisible = _useState2[1];
|
|
33
|
+
var uploadRefs = useRef([]);
|
|
34
|
+
var popoverNodes = useMemo(function () {
|
|
35
|
+
return uploadPropsList.map(function (item, index) {
|
|
36
|
+
return /*#__PURE__*/_jsx("div", {
|
|
37
|
+
onClick: function onClick() {
|
|
38
|
+
var _uploadRefs$current$i;
|
|
39
|
+
// 触发对应Upload组件的children的click事件
|
|
40
|
+
(_uploadRefs$current$i = uploadRefs.current[index]) === null || _uploadRefs$current$i === void 0 || _uploadRefs$current$i.click();
|
|
41
|
+
setVisible(false);
|
|
42
|
+
},
|
|
43
|
+
children: item.trigger
|
|
44
|
+
}, index);
|
|
45
|
+
});
|
|
46
|
+
}, [uploadPropsList]);
|
|
26
47
|
var nodes = useMemo(function () {
|
|
27
48
|
return uploadPropsList.map(function (item, index) {
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
49
|
+
var trigger = item.trigger,
|
|
50
|
+
rest = _objectWithoutProperties(item, _excluded);
|
|
51
|
+
return /*#__PURE__*/_jsx(Upload, _objectSpread(_objectSpread({}, rest), {}, {
|
|
52
|
+
children: /*#__PURE__*/_jsx("div", {
|
|
53
|
+
ref: function ref(el) {
|
|
54
|
+
return uploadRefs.current[index] = el;
|
|
32
55
|
}
|
|
33
|
-
|
|
34
|
-
}
|
|
56
|
+
})
|
|
35
57
|
}), index);
|
|
36
58
|
});
|
|
37
59
|
}, [uploadPropsList]);
|
|
38
|
-
return /*#__PURE__*/
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
60
|
+
return /*#__PURE__*/_jsxs(_Fragment, {
|
|
61
|
+
children: [/*#__PURE__*/_jsx(Popover, {
|
|
62
|
+
placement: "bottomLeft",
|
|
63
|
+
open: visible,
|
|
64
|
+
onOpenChange: setVisible,
|
|
65
|
+
content: /*#__PURE__*/_jsx(Flex, {
|
|
66
|
+
vertical: true,
|
|
67
|
+
children: popoverNodes
|
|
68
|
+
}),
|
|
69
|
+
trigger: "click",
|
|
70
|
+
styles: {
|
|
71
|
+
body: {
|
|
72
|
+
padding: 4
|
|
73
|
+
}
|
|
74
|
+
},
|
|
75
|
+
children: /*#__PURE__*/_jsx(IconButton, {
|
|
76
|
+
icon: /*#__PURE__*/_jsx(PlusOutlined, {}),
|
|
77
|
+
bordered: false
|
|
78
|
+
})
|
|
79
|
+
}), /*#__PURE__*/_jsx("div", {
|
|
80
|
+
className: "".concat(prefixCls, "-upload-hidden-nodes"),
|
|
44
81
|
children: nodes
|
|
45
|
-
})
|
|
46
|
-
trigger: "click",
|
|
47
|
-
styles: {
|
|
48
|
-
body: {
|
|
49
|
-
padding: 4
|
|
50
|
-
}
|
|
51
|
-
},
|
|
52
|
-
children: /*#__PURE__*/_jsx(IconButton, {
|
|
53
|
-
icon: /*#__PURE__*/_jsx(PlusOutlined, {}),
|
|
54
|
-
bordered: false
|
|
55
|
-
})
|
|
82
|
+
})]
|
|
56
83
|
});
|
|
57
84
|
}
|
|
@@ -163,10 +163,12 @@ export default /*#__PURE__*/forwardRef(function (_, ref) {
|
|
|
163
163
|
}
|
|
164
164
|
},
|
|
165
165
|
showUploadList: false,
|
|
166
|
-
|
|
166
|
+
trigger: trigger
|
|
167
167
|
});
|
|
168
168
|
});
|
|
169
|
-
if (uploadPropsList.length === 1) return /*#__PURE__*/_jsx(Upload, _objectSpread({}, uploadPropsList[0])
|
|
169
|
+
if (uploadPropsList.length === 1) return /*#__PURE__*/_jsx(Upload, _objectSpread(_objectSpread({}, uploadPropsList[0]), {}, {
|
|
170
|
+
children: uploadPropsList[0].trigger
|
|
171
|
+
}));
|
|
170
172
|
return /*#__PURE__*/_jsx(UploadPopover, {
|
|
171
173
|
uploadPropsList: uploadPropsList
|
|
172
174
|
});
|
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
var _templateObject;
|
|
2
2
|
function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
|
|
3
3
|
import { createGlobalStyle } from 'antd-style';
|
|
4
|
-
export default createGlobalStyle(_templateObject || (_templateObject = _taggedTemplateLiteral(["\n.", "-chat-anywhere-sender-wrapper {\n position: relative;\n\n &-header {\n display: flex;\n gap: 8px;\n margin-bottom: 12px;\n }\n}\n\n"])), function (p) {
|
|
4
|
+
export default createGlobalStyle(_templateObject || (_templateObject = _taggedTemplateLiteral(["\n.", "-chat-anywhere-sender-wrapper {\n position: relative;\n\n &-header {\n display: flex;\n gap: 8px;\n margin-bottom: 12px;\n }\n}\n\n.", "-chat-anywhere-sender-upload-hidden-nodes {\n position: absolute;\n z-index: -999;\n top: -100vh;\n left: -100vw;\n width: 0;\n height: 0;\n overflow: hidden;\n }\n}\n"])), function (p) {
|
|
5
|
+
return p.theme.prefixCls;
|
|
6
|
+
}, function (p) {
|
|
5
7
|
return p.theme.prefixCls;
|
|
6
8
|
});
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@agentscope-ai/chat",
|
|
3
|
-
"version": "1.1.45-beta.
|
|
3
|
+
"version": "1.1.45-beta.1766631141555",
|
|
4
4
|
"description": "a free and open-source chat framework for building excellent LLM-powered chat experiences",
|
|
5
5
|
"license": "Apache-2.0",
|
|
6
6
|
"sideEffects": [
|
|
@@ -58,6 +58,7 @@
|
|
|
58
58
|
]
|
|
59
59
|
},
|
|
60
60
|
"dependencies": {
|
|
61
|
+
"dayjs": "^1.11.7",
|
|
61
62
|
"@agentscope-ai/design": "^1.0.0",
|
|
62
63
|
"@agentscope-ai/icons": "^1.0.32",
|
|
63
64
|
"@agentscope-ai/icons-override-antd": "^6.0.0",
|
|
@@ -1,75 +0,0 @@
|
|
|
1
|
-
# agentscope-runtime-starter-webui
|
|
2
|
-
|
|
3
|
-
## node version
|
|
4
|
-
|
|
5
|
-
> =22
|
|
6
|
-
|
|
7
|
-
## install
|
|
8
|
-
|
|
9
|
-
```
|
|
10
|
-
$ npm run install
|
|
11
|
-
```
|
|
12
|
-
|
|
13
|
-
## dev
|
|
14
|
-
|
|
15
|
-
```
|
|
16
|
-
$ npm run dev
|
|
17
|
-
```
|
|
18
|
-
|
|
19
|
-
## build
|
|
20
|
-
|
|
21
|
-
```
|
|
22
|
-
$ npm run build
|
|
23
|
-
```
|
|
24
|
-
|
|
25
|
-
## Core Code
|
|
26
|
-
```tsx
|
|
27
|
-
import { AgentScopeRuntimeWebUI } from '@agentscope-ai/chat';
|
|
28
|
-
|
|
29
|
-
const options = {
|
|
30
|
-
theme: {
|
|
31
|
-
colorPrimary: '#615CED',
|
|
32
|
-
darkMode: true,
|
|
33
|
-
prefix: 'agentscope-runtime-webui',
|
|
34
|
-
leftHeader: {
|
|
35
|
-
logo: 'https://img.alicdn.com/imgextra/i2/O1CN01lmoGYn1kjoXATy4PX_!!6000000004720-2-tps-200-200.png',
|
|
36
|
-
title: 'Runtime WebUI',
|
|
37
|
-
},
|
|
38
|
-
},
|
|
39
|
-
sender: {
|
|
40
|
-
maxLength: 10000,
|
|
41
|
-
disclaimer:
|
|
42
|
-
'AI can also make mistakes, so please check carefully and use it with caution',
|
|
43
|
-
},
|
|
44
|
-
|
|
45
|
-
welcome: {
|
|
46
|
-
greeting: 'Hello, how can I help you today?',
|
|
47
|
-
description:
|
|
48
|
-
'I am a helpful assistant that can help you with your questions.',
|
|
49
|
-
avatar:
|
|
50
|
-
'https://img.alicdn.com/imgextra/i2/O1CN01lmoGYn1kjoXATy4PX_!!6000000004720-2-tps-200-200.png',
|
|
51
|
-
prompts: [
|
|
52
|
-
{
|
|
53
|
-
value: 'Hello',
|
|
54
|
-
},
|
|
55
|
-
{
|
|
56
|
-
value: 'How are you?',
|
|
57
|
-
},
|
|
58
|
-
{
|
|
59
|
-
value: 'What can you do?',
|
|
60
|
-
},
|
|
61
|
-
],
|
|
62
|
-
},
|
|
63
|
-
api: {
|
|
64
|
-
baseURL: 'YOUR_API_URL',
|
|
65
|
-
token: 'YOUR_API_TOKEN', // is not required
|
|
66
|
-
},
|
|
67
|
-
};
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
<AgentScopeRuntimeWebUI
|
|
71
|
-
options={options}
|
|
72
|
-
/>
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
```
|
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
import js from '@eslint/js'
|
|
2
|
-
import globals from 'globals'
|
|
3
|
-
import reactHooks from 'eslint-plugin-react-hooks'
|
|
4
|
-
import reactRefresh from 'eslint-plugin-react-refresh'
|
|
5
|
-
import tseslint from 'typescript-eslint'
|
|
6
|
-
|
|
7
|
-
export default tseslint.config(
|
|
8
|
-
{ ignores: ['dist'] },
|
|
9
|
-
{
|
|
10
|
-
extends: [js.configs.recommended, ...tseslint.configs.recommended],
|
|
11
|
-
files: ['**/*.{ts,tsx}'],
|
|
12
|
-
languageOptions: {
|
|
13
|
-
ecmaVersion: 2020,
|
|
14
|
-
globals: globals.browser,
|
|
15
|
-
},
|
|
16
|
-
plugins: {
|
|
17
|
-
'react-hooks': reactHooks,
|
|
18
|
-
'react-refresh': reactRefresh,
|
|
19
|
-
},
|
|
20
|
-
rules: {
|
|
21
|
-
...reactHooks.configs.recommended.rules,
|
|
22
|
-
'react-refresh/only-export-components': [
|
|
23
|
-
'warn',
|
|
24
|
-
{ allowConstantExport: true },
|
|
25
|
-
],
|
|
26
|
-
},
|
|
27
|
-
},
|
|
28
|
-
)
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
<!doctype html>
|
|
2
|
-
<html lang="en">
|
|
3
|
-
<head>
|
|
4
|
-
<meta charset="UTF-8" />
|
|
5
|
-
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
6
|
-
<title>AgentScope Runtime Starter WebUI</title>
|
|
7
|
-
</head>
|
|
8
|
-
<body>
|
|
9
|
-
<div id="root"></div>
|
|
10
|
-
<script type="module" src="/src/main.tsx"></script>
|
|
11
|
-
</body>
|
|
12
|
-
</html>
|
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"name": "agentscope-runtime-starter-webui",
|
|
3
|
-
"private": true,
|
|
4
|
-
"version": "0.0.0",
|
|
5
|
-
"type": "module",
|
|
6
|
-
"scripts": {
|
|
7
|
-
"dev": "vite --host",
|
|
8
|
-
"build": "tsc -b && vite build",
|
|
9
|
-
"lint": "eslint .",
|
|
10
|
-
"preview": "vite preview"
|
|
11
|
-
},
|
|
12
|
-
"dependencies": {
|
|
13
|
-
"@agentscope-ai/icons": "^1.0.46",
|
|
14
|
-
"@agentscope-ai/chat": "^1.1.44",
|
|
15
|
-
"@agentscope-ai/design": "^1.0.19",
|
|
16
|
-
"antd": "^5.29.1",
|
|
17
|
-
"antd-style": "^3.7.1",
|
|
18
|
-
"react": "^18",
|
|
19
|
-
"react-dom": "^18"
|
|
20
|
-
},
|
|
21
|
-
"devDependencies": {
|
|
22
|
-
"@eslint/js": "^9.25.0",
|
|
23
|
-
"@types/react": "^18",
|
|
24
|
-
"@types/react-dom": "^18",
|
|
25
|
-
"@vitejs/plugin-react": "^4.4.1",
|
|
26
|
-
"eslint": "^9.25.0",
|
|
27
|
-
"eslint-plugin-react-hooks": "^5.2.0",
|
|
28
|
-
"eslint-plugin-react-refresh": "^0.4.19",
|
|
29
|
-
"globals": "^16.0.0",
|
|
30
|
-
"typescript": "~5.8.3",
|
|
31
|
-
"typescript-eslint": "^8.30.1",
|
|
32
|
-
"vite": "^6.3.5"
|
|
33
|
-
}
|
|
34
|
-
}
|
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
import Chat from './components/Chat';
|
|
2
|
-
|
|
3
|
-
import { createGlobalStyle } from 'antd-style';
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
const GlobalStyle = createGlobalStyle`
|
|
7
|
-
* {
|
|
8
|
-
margin: 0;
|
|
9
|
-
box-sizing: border-box;
|
|
10
|
-
}
|
|
11
|
-
`;
|
|
12
|
-
|
|
13
|
-
function App() {
|
|
14
|
-
return <>
|
|
15
|
-
<GlobalStyle />
|
|
16
|
-
<Chat />
|
|
17
|
-
</>
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
export default App
|
|
@@ -1,37 +0,0 @@
|
|
|
1
|
-
import { Form } from 'antd';
|
|
2
|
-
import { createStyles } from 'antd-style';
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
interface FormItemProps {
|
|
6
|
-
name: string | string[];
|
|
7
|
-
label: string;
|
|
8
|
-
isList?: boolean;
|
|
9
|
-
children: any;
|
|
10
|
-
normalize?: (value: any) => any;
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
const useStyles = createStyles(({ token }) => ({
|
|
15
|
-
label: {
|
|
16
|
-
marginBottom: 6,
|
|
17
|
-
fontSize: 12,
|
|
18
|
-
color: token.colorTextSecondary,
|
|
19
|
-
},
|
|
20
|
-
|
|
21
|
-
}));
|
|
22
|
-
|
|
23
|
-
export default function FormItem(props: FormItemProps) {
|
|
24
|
-
const { styles } = useStyles();
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
const node = props.isList ?
|
|
28
|
-
<Form.List name={props.name}>{props.children}</Form.List> :
|
|
29
|
-
<Form.Item name={props.name} normalize={props.normalize}>{props.children}</Form.Item>;
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
return <div>
|
|
33
|
-
{props.label && <div className={styles.label}>{props.label}</div>}
|
|
34
|
-
{node}
|
|
35
|
-
</div>
|
|
36
|
-
|
|
37
|
-
}
|
|
@@ -1,160 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
import { Form, Input, ColorPicker, Flex, Divider, InputNumber } from 'antd';
|
|
3
|
-
import { createStyles } from 'antd-style';
|
|
4
|
-
import { Button, IconButton, Switch } from '@agentscope-ai/design'
|
|
5
|
-
import { SparkDeleteLine, SparkPlusLine } from '@agentscope-ai/icons';
|
|
6
|
-
import FormItem from './FormItem';
|
|
7
|
-
import defaultConfig from './defaultConfig';
|
|
8
|
-
|
|
9
|
-
const useStyles = createStyles(({ token }) => ({
|
|
10
|
-
container: {
|
|
11
|
-
height: '100%',
|
|
12
|
-
display: 'flex',
|
|
13
|
-
flexDirection: 'column',
|
|
14
|
-
},
|
|
15
|
-
|
|
16
|
-
form: {
|
|
17
|
-
height: 0,
|
|
18
|
-
flex: 1,
|
|
19
|
-
padding: '8px 16px 16px 16px',
|
|
20
|
-
overflow: 'auto',
|
|
21
|
-
},
|
|
22
|
-
actions: {
|
|
23
|
-
padding: 16,
|
|
24
|
-
display: 'flex',
|
|
25
|
-
borderTop: `1px solid ${token.colorBorderSecondary}`,
|
|
26
|
-
justifyContent: 'flex-end',
|
|
27
|
-
gap: 16,
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
}));
|
|
31
|
-
|
|
32
|
-
interface OptionsEditorProps {
|
|
33
|
-
value?: any;
|
|
34
|
-
onChange?: any;
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
const OptionsEditor: React.FC<OptionsEditorProps> = ({
|
|
38
|
-
value,
|
|
39
|
-
onChange,
|
|
40
|
-
}) => {
|
|
41
|
-
const { styles } = useStyles();
|
|
42
|
-
const [form] = Form.useForm();
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
const handleSave = () => {
|
|
46
|
-
form.validateFields().then((values) => {
|
|
47
|
-
onChange(values);
|
|
48
|
-
});
|
|
49
|
-
};
|
|
50
|
-
|
|
51
|
-
const handleReset = () => {
|
|
52
|
-
form.setFieldsValue(defaultConfig);
|
|
53
|
-
};
|
|
54
|
-
|
|
55
|
-
return (
|
|
56
|
-
<div className={styles.container}>
|
|
57
|
-
<Form
|
|
58
|
-
className={styles.form}
|
|
59
|
-
form={form}
|
|
60
|
-
layout="vertical"
|
|
61
|
-
initialValues={value}
|
|
62
|
-
>
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
<Divider orientation="left">Theme</Divider>
|
|
66
|
-
|
|
67
|
-
<FormItem name={['theme', 'colorPrimary']} label="colorPrimary" normalize={value => value.toHexString()}>
|
|
68
|
-
<ColorPicker />
|
|
69
|
-
</FormItem>
|
|
70
|
-
|
|
71
|
-
<FormItem name={['theme', 'colorBgBase']} label="colorBgBase" normalize={value => value.toHexString()}>
|
|
72
|
-
<ColorPicker />
|
|
73
|
-
</FormItem>
|
|
74
|
-
|
|
75
|
-
<FormItem name={['theme', 'colorTextBase']} label="colorTextBase" normalize={value => value.toHexString()}>
|
|
76
|
-
<ColorPicker />
|
|
77
|
-
</FormItem>
|
|
78
|
-
|
|
79
|
-
<FormItem name={['theme', 'darkMode']} label="darkMode" >
|
|
80
|
-
<Switch />
|
|
81
|
-
</FormItem>
|
|
82
|
-
|
|
83
|
-
<FormItem name={['theme', 'leftHeader', 'logo']} label="leftHeader.logo" >
|
|
84
|
-
<Input />
|
|
85
|
-
</FormItem>
|
|
86
|
-
|
|
87
|
-
<FormItem name={['theme', 'leftHeader', 'title']} label="leftHeader.title" >
|
|
88
|
-
<Input />
|
|
89
|
-
</FormItem>
|
|
90
|
-
|
|
91
|
-
<Divider orientation="left">Sender</Divider>
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
<FormItem name={['sender', 'disclaimer']} label="disclaimer" >
|
|
95
|
-
<Input />
|
|
96
|
-
</FormItem>
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
<FormItem name={['sender', 'maxLength']} label="maxLength" >
|
|
102
|
-
<InputNumber min={1000} />
|
|
103
|
-
</FormItem>
|
|
104
|
-
|
|
105
|
-
<Divider orientation="left">Welcome</Divider>
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
<FormItem name={['welcome', 'greeting']} label="greeting" >
|
|
109
|
-
<Input />
|
|
110
|
-
</FormItem>
|
|
111
|
-
|
|
112
|
-
<FormItem name={['welcome', 'description']} label="description" >
|
|
113
|
-
<Input />
|
|
114
|
-
</FormItem>
|
|
115
|
-
|
|
116
|
-
<FormItem name={['welcome', 'avatar']} label="avatar" >
|
|
117
|
-
<Input />
|
|
118
|
-
</FormItem>
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
<FormItem name={['welcome', 'prompts']} isList label="prompts" >
|
|
122
|
-
{(fields: { key: string, name: string }[], { add, remove }: { add: (item: any) => void, remove: (name: string) => void }) => {
|
|
123
|
-
return <div>
|
|
124
|
-
{fields.map(field => {
|
|
125
|
-
return <Flex key={field.key} gap={6}>
|
|
126
|
-
<Form.Item style={{ flex: 1 }} key={field.key} name={[field.name, 'value']}>
|
|
127
|
-
<Input />
|
|
128
|
-
</Form.Item>
|
|
129
|
-
<IconButton icon={<SparkPlusLine />} onClick={() => add({})}></IconButton>
|
|
130
|
-
<IconButton icon={<SparkDeleteLine />} onClick={() => remove(field.name)}></IconButton>
|
|
131
|
-
</Flex>
|
|
132
|
-
})}
|
|
133
|
-
</div>
|
|
134
|
-
}}
|
|
135
|
-
</FormItem>
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
<Divider orientation="left">API</Divider>
|
|
139
|
-
|
|
140
|
-
<FormItem name={['api', 'baseURL']} label="baseURL" >
|
|
141
|
-
<Input />
|
|
142
|
-
</FormItem>
|
|
143
|
-
|
|
144
|
-
<FormItem name={['api', 'token']} label="token" >
|
|
145
|
-
<Input />
|
|
146
|
-
</FormItem>
|
|
147
|
-
</Form>
|
|
148
|
-
|
|
149
|
-
<div className={styles.actions}>
|
|
150
|
-
<Button onClick={handleReset}>Reset</Button>
|
|
151
|
-
<Button type="primary" onClick={handleSave}>
|
|
152
|
-
Save & Copy
|
|
153
|
-
</Button>
|
|
154
|
-
</div>
|
|
155
|
-
</div>
|
|
156
|
-
);
|
|
157
|
-
};
|
|
158
|
-
|
|
159
|
-
export default OptionsEditor;
|
|
160
|
-
|
|
@@ -1,41 +0,0 @@
|
|
|
1
|
-
export default {
|
|
2
|
-
theme: {
|
|
3
|
-
colorPrimary: '#615CED',
|
|
4
|
-
darkMode: true,
|
|
5
|
-
prefix: 'agentscope-runtime-webui',
|
|
6
|
-
leftHeader: {
|
|
7
|
-
logo: 'https://img.alicdn.com/imgextra/i2/O1CN01lmoGYn1kjoXATy4PX_!!6000000004720-2-tps-200-200.png',
|
|
8
|
-
title: 'Runtime WebUI',
|
|
9
|
-
},
|
|
10
|
-
},
|
|
11
|
-
sender: {
|
|
12
|
-
attachments: false,
|
|
13
|
-
maxLength: 10000,
|
|
14
|
-
disclaimer:
|
|
15
|
-
'AI can also make mistakes, so please check carefully and use it with caution',
|
|
16
|
-
},
|
|
17
|
-
|
|
18
|
-
welcome: {
|
|
19
|
-
greeting: 'Hello, how can I help you today?',
|
|
20
|
-
description:
|
|
21
|
-
'I am a helpful assistant that can help you with your questions.',
|
|
22
|
-
avatar:
|
|
23
|
-
'https://img.alicdn.com/imgextra/i2/O1CN01lmoGYn1kjoXATy4PX_!!6000000004720-2-tps-200-200.png',
|
|
24
|
-
prompts: [
|
|
25
|
-
{
|
|
26
|
-
value: 'Hello',
|
|
27
|
-
},
|
|
28
|
-
{
|
|
29
|
-
value: 'How are you?',
|
|
30
|
-
},
|
|
31
|
-
{
|
|
32
|
-
value: 'What can you do?',
|
|
33
|
-
},
|
|
34
|
-
],
|
|
35
|
-
},
|
|
36
|
-
api: {
|
|
37
|
-
baseURL: BASE_URL,
|
|
38
|
-
token: TOKEN,
|
|
39
|
-
},
|
|
40
|
-
};
|
|
41
|
-
|
|
@@ -1,27 +0,0 @@
|
|
|
1
|
-
import { SparkSettingLine } from "@agentscope-ai/icons";
|
|
2
|
-
import { IconButton, Drawer } from "@agentscope-ai/design";
|
|
3
|
-
import { useState } from "react";
|
|
4
|
-
import OptionsEditor from "./OptionsEditor";
|
|
5
|
-
|
|
6
|
-
interface OptionsPanelProps {
|
|
7
|
-
value?: any;
|
|
8
|
-
onChange?: any;
|
|
9
|
-
}
|
|
10
|
-
|
|
11
|
-
export default function OptionsPanel(props: OptionsPanelProps) {
|
|
12
|
-
const [open, setOpen] = useState(false);
|
|
13
|
-
|
|
14
|
-
return <>
|
|
15
|
-
<IconButton onClick={() => setOpen(true)} icon={<SparkSettingLine />} bordered={false} />
|
|
16
|
-
<Drawer
|
|
17
|
-
destroyOnHidden
|
|
18
|
-
open={open}
|
|
19
|
-
onClose={() => setOpen(false)}
|
|
20
|
-
styles={{ body: { padding: 0 }, header: { padding: 8 } }}>
|
|
21
|
-
<OptionsEditor value={props.value} onChange={(v: typeof props.value) => {
|
|
22
|
-
setOpen(false);
|
|
23
|
-
props.onChange(v);
|
|
24
|
-
}} />
|
|
25
|
-
</Drawer>
|
|
26
|
-
</>
|
|
27
|
-
}
|
|
@@ -1,45 +0,0 @@
|
|
|
1
|
-
import { AgentScopeRuntimeWebUI, IAgentScopeRuntimeWebUIOptions } from '@agentscope-ai/chat';
|
|
2
|
-
import OptionsPanel from './OptionsPanel';
|
|
3
|
-
import { useMemo } from 'react';
|
|
4
|
-
import sessionApi from './sessionApi';
|
|
5
|
-
import { useLocalStorageState } from 'ahooks';
|
|
6
|
-
import defaultConfig from './OptionsPanel/defaultConfig';
|
|
7
|
-
|
|
8
|
-
export default function () {
|
|
9
|
-
const [optionsConfig, setOptionsConfig] = useLocalStorageState('agent-scope-runtime-webui-options', {
|
|
10
|
-
defaultValue: defaultConfig,
|
|
11
|
-
listenStorageChange: true,
|
|
12
|
-
});
|
|
13
|
-
|
|
14
|
-
const options = useMemo(() => {
|
|
15
|
-
const rightHeader = <OptionsPanel value={optionsConfig} onChange={(v: typeof optionsConfig) => {
|
|
16
|
-
setOptionsConfig(prev => ({
|
|
17
|
-
...prev,
|
|
18
|
-
...v,
|
|
19
|
-
}));
|
|
20
|
-
}} />;
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
return {
|
|
25
|
-
...optionsConfig,
|
|
26
|
-
session: {
|
|
27
|
-
multiple: true,
|
|
28
|
-
api: sessionApi,
|
|
29
|
-
},
|
|
30
|
-
theme: {
|
|
31
|
-
...optionsConfig.theme,
|
|
32
|
-
rightHeader,
|
|
33
|
-
},
|
|
34
|
-
};
|
|
35
|
-
}, [optionsConfig]);
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
return <div style={{ height: '100vh' }}>
|
|
41
|
-
<AgentScopeRuntimeWebUI
|
|
42
|
-
options={options as unknown as IAgentScopeRuntimeWebUIOptions}
|
|
43
|
-
/>
|
|
44
|
-
</div>;
|
|
45
|
-
}
|
|
@@ -1,53 +0,0 @@
|
|
|
1
|
-
import {
|
|
2
|
-
IAgentScopeRuntimeWebUISession,
|
|
3
|
-
IAgentScopeRuntimeWebUISessionAPI,
|
|
4
|
-
} from '@agentscope-ai/chat';
|
|
5
|
-
|
|
6
|
-
class SessionApi implements IAgentScopeRuntimeWebUISessionAPI {
|
|
7
|
-
private lsKey: string;
|
|
8
|
-
private sessionList: IAgentScopeRuntimeWebUISession[];
|
|
9
|
-
|
|
10
|
-
constructor() {
|
|
11
|
-
this.lsKey = 'agent-scope-runtime-webui-sessions';
|
|
12
|
-
this.sessionList = [];
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
async getSessionList() {
|
|
16
|
-
this.sessionList = JSON.parse(localStorage.getItem(this.lsKey) || '[]');
|
|
17
|
-
return [...this.sessionList];
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
async getSession(sessionId: string) {
|
|
21
|
-
return this.sessionList.find((session) => session.id === sessionId) as IAgentScopeRuntimeWebUISession;
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
async updateSession(session: Partial<IAgentScopeRuntimeWebUISession>) {
|
|
25
|
-
const index = this.sessionList.findIndex((item) => item.id === session.id);
|
|
26
|
-
if (index > -1) {
|
|
27
|
-
this.sessionList[index] = {
|
|
28
|
-
...this.sessionList[index],
|
|
29
|
-
...session,
|
|
30
|
-
};
|
|
31
|
-
localStorage.setItem(this.lsKey, JSON.stringify(this.sessionList));
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
return [...this.sessionList];
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
async createSession(session: Partial<IAgentScopeRuntimeWebUISession>) {
|
|
38
|
-
session.id = Date.now().toString();
|
|
39
|
-
this.sessionList.unshift(session as IAgentScopeRuntimeWebUISession);
|
|
40
|
-
localStorage.setItem(this.lsKey, JSON.stringify(this.sessionList));
|
|
41
|
-
return [...this.sessionList];
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
async removeSession(session: Partial<IAgentScopeRuntimeWebUISession>) {
|
|
45
|
-
this.sessionList = this.sessionList.filter(
|
|
46
|
-
(item) => item.id !== session.id,
|
|
47
|
-
);
|
|
48
|
-
localStorage.setItem(this.lsKey, JSON.stringify(this.sessionList));
|
|
49
|
-
return [...this.sessionList];
|
|
50
|
-
}
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
export default new SessionApi();
|
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"compilerOptions": {
|
|
3
|
-
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
|
|
4
|
-
"target": "ES2020",
|
|
5
|
-
"useDefineForClassFields": true,
|
|
6
|
-
"lib": ["ES2020", "DOM", "DOM.Iterable"],
|
|
7
|
-
"module": "ESNext",
|
|
8
|
-
"skipLibCheck": true,
|
|
9
|
-
|
|
10
|
-
"moduleResolution": "bundler",
|
|
11
|
-
"allowImportingTsExtensions": true,
|
|
12
|
-
"moduleDetection": "force",
|
|
13
|
-
"noEmit": true,
|
|
14
|
-
"jsx": "react-jsx",
|
|
15
|
-
|
|
16
|
-
"strict": true,
|
|
17
|
-
"noUnusedLocals": true,
|
|
18
|
-
"noUnusedParameters": true,
|
|
19
|
-
"erasableSyntaxOnly": true,
|
|
20
|
-
"noFallthroughCasesInSwitch": true,
|
|
21
|
-
"noUncheckedSideEffectImports": true
|
|
22
|
-
},
|
|
23
|
-
"include": ["src"]
|
|
24
|
-
}
|
|
@@ -1,22 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"compilerOptions": {
|
|
3
|
-
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo",
|
|
4
|
-
"target": "ES2022",
|
|
5
|
-
"lib": ["ES2023"],
|
|
6
|
-
"module": "ESNext",
|
|
7
|
-
"skipLibCheck": true,
|
|
8
|
-
|
|
9
|
-
"moduleResolution": "bundler",
|
|
10
|
-
"allowImportingTsExtensions": true,
|
|
11
|
-
"moduleDetection": "force",
|
|
12
|
-
"noEmit": true,
|
|
13
|
-
|
|
14
|
-
"strict": true,
|
|
15
|
-
"noUnusedLocals": true,
|
|
16
|
-
"noUnusedParameters": true,
|
|
17
|
-
"erasableSyntaxOnly": true,
|
|
18
|
-
"noFallthroughCasesInSwitch": true,
|
|
19
|
-
"noUncheckedSideEffectImports": true
|
|
20
|
-
},
|
|
21
|
-
"include": ["vite.config.ts"]
|
|
22
|
-
}
|
|
@@ -1,11 +0,0 @@
|
|
|
1
|
-
import { defineConfig } from 'vite'
|
|
2
|
-
import react from '@vitejs/plugin-react'
|
|
3
|
-
|
|
4
|
-
export default defineConfig({
|
|
5
|
-
define: {
|
|
6
|
-
BASE_URL: JSON.stringify(process.env.BASE_URL || ''),
|
|
7
|
-
TOKEN: JSON.stringify(process.env.TOKEN || ''),
|
|
8
|
-
MOBILE: false,
|
|
9
|
-
},
|
|
10
|
-
plugins: [react()],
|
|
11
|
-
})
|