V2.2.4 版本

This commit is contained in:
lq1405 2024-05-24 13:46:19 +08:00
parent 74009113fa
commit 17709c1a0b
57 changed files with 4484 additions and 2338 deletions

1
.gitignore vendored
View File

@ -5,6 +5,7 @@ resources/scripts/build*
resources/scripts/dist
resources/scripts/model
resources/scripts/Temp
resources/image/Temp*
resources/package/ffmpeg*
resources/config*
*Lai.exe*

View File

@ -5,9 +5,7 @@ import Jsx from '@vitejs/plugin-vue-jsx'
export default defineConfig({
main: {
plugins: [externalizeDepsPlugin(), bytecodePlugin({
exclude: ['src/main/discord/discordScript.js']
})]
plugins: [externalizeDepsPlugin(), bytecodePlugin()]
},
discord: {
plugins: [externalizeDepsPlugin(), bytecodePlugin()]

261
package-lock.json generated
View File

@ -1,6 +1,6 @@
{
"name": "laitool",
"version": "2.2.2",
"version": "2.2.4",
"lockfileVersion": 3,
"requires": true,
"packages": {
@ -21,6 +21,7 @@
"axios": "^1.6.5",
"compressing": "^1.10.0",
"crypto-js": "^4.2.0",
"electron-store": "^9.0.0",
"electron-updater": "^6.1.7",
"fluent-ffmpeg": "^2.1.2",
"highlight.js": "^11.9.0",
@ -2548,7 +2549,9 @@
},
"node_modules/abbrev": {
"version": "1.1.1",
"license": "ISC"
"license": "ISC",
"optional": true,
"peer": true
},
"node_modules/acorn": {
"version": "8.11.3",
@ -2594,6 +2597,42 @@
"url": "https://github.com/sponsors/epoberezkin"
}
},
"node_modules/ajv-formats": {
"version": "2.1.1",
"resolved": "https://registry.npmmirror.com/ajv-formats/-/ajv-formats-2.1.1.tgz",
"integrity": "sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==",
"dependencies": {
"ajv": "^8.0.0"
},
"peerDependencies": {
"ajv": "^8.0.0"
},
"peerDependenciesMeta": {
"ajv": {
"optional": true
}
}
},
"node_modules/ajv-formats/node_modules/ajv": {
"version": "8.13.0",
"resolved": "https://registry.npmmirror.com/ajv/-/ajv-8.13.0.tgz",
"integrity": "sha512-PRA911Blj99jR5RMeTunVbNXMF6Lp4vZXnk5GQjcnUWUTsrXtekg/pnmFFI2u/I36Y/2bITGS30GZCXei6uNkA==",
"dependencies": {
"fast-deep-equal": "^3.1.3",
"json-schema-traverse": "^1.0.0",
"require-from-string": "^2.0.2",
"uri-js": "^4.4.1"
},
"funding": {
"type": "github",
"url": "https://github.com/sponsors/epoberezkin"
}
},
"node_modules/ajv-formats/node_modules/json-schema-traverse": {
"version": "1.0.0",
"resolved": "https://registry.npmmirror.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz",
"integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug=="
},
"node_modules/ajv-keywords": {
"version": "3.5.2",
"dev": true,
@ -2612,6 +2651,7 @@
},
"node_modules/ansi-styles": {
"version": "4.3.0",
"dev": true,
"license": "MIT",
"dependencies": {
"color-convert": "^2.0.1"
@ -2894,6 +2934,15 @@
"node": ">= 4.0.0"
}
},
"node_modules/atomically": {
"version": "2.0.3",
"resolved": "https://registry.npmmirror.com/atomically/-/atomically-2.0.3.tgz",
"integrity": "sha512-kU6FmrwZ3Lx7/7y3hPS5QnbJfaohcIul5fGqf7ok+4KklIEk9tJ0C2IQPdacSbVUWv6zVHXEBWoWd6NrVMT7Cw==",
"dependencies": {
"stubborn-fs": "^1.2.5",
"when-exit": "^2.1.1"
}
},
"node_modules/awesome-js": {
"version": "2.0.0",
"resolved": "https://registry.npmmirror.com/awesome-js/-/awesome-js-2.0.0.tgz",
@ -2914,6 +2963,7 @@
},
"node_modules/balanced-match": {
"version": "1.0.2",
"devOptional": true,
"license": "MIT"
},
"node_modules/base64-js": {
@ -3014,6 +3064,7 @@
},
"node_modules/brace-expansion": {
"version": "2.0.1",
"dev": true,
"license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0"
@ -3272,6 +3323,7 @@
},
"node_modules/chalk": {
"version": "4.1.2",
"dev": true,
"license": "MIT",
"dependencies": {
"ansi-styles": "^4.1.0",
@ -3286,6 +3338,7 @@
},
"node_modules/chownr": {
"version": "2.0.0",
"devOptional": true,
"license": "ISC",
"engines": {
"node": ">=10"
@ -3298,6 +3351,7 @@
},
"node_modules/ci-info": {
"version": "3.9.0",
"dev": true,
"funding": [
{
"type": "github",
@ -3455,8 +3509,73 @@
},
"node_modules/concat-map": {
"version": "0.0.1",
"devOptional": true,
"license": "MIT"
},
"node_modules/conf": {
"version": "12.0.0",
"resolved": "https://registry.npmmirror.com/conf/-/conf-12.0.0.tgz",
"integrity": "sha512-fIWyWUXrJ45cHCIQX+Ck1hrZDIf/9DR0P0Zewn3uNht28hbt5OfGUq8rRWsxi96pZWPyBEd0eY9ama01JTaknA==",
"dependencies": {
"ajv": "^8.12.0",
"ajv-formats": "^2.1.1",
"atomically": "^2.0.2",
"debounce-fn": "^5.1.2",
"dot-prop": "^8.0.2",
"env-paths": "^3.0.0",
"json-schema-typed": "^8.0.1",
"semver": "^7.5.4",
"uint8array-extras": "^0.3.0"
},
"engines": {
"node": ">=18"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/conf/node_modules/ajv": {
"version": "8.13.0",
"resolved": "https://registry.npmmirror.com/ajv/-/ajv-8.13.0.tgz",
"integrity": "sha512-PRA911Blj99jR5RMeTunVbNXMF6Lp4vZXnk5GQjcnUWUTsrXtekg/pnmFFI2u/I36Y/2bITGS30GZCXei6uNkA==",
"dependencies": {
"fast-deep-equal": "^3.1.3",
"json-schema-traverse": "^1.0.0",
"require-from-string": "^2.0.2",
"uri-js": "^4.4.1"
},
"funding": {
"type": "github",
"url": "https://github.com/sponsors/epoberezkin"
}
},
"node_modules/conf/node_modules/env-paths": {
"version": "3.0.0",
"resolved": "https://registry.npmmirror.com/env-paths/-/env-paths-3.0.0.tgz",
"integrity": "sha512-dtJUTepzMW3Lm/NPxRf3wP4642UWhjL2sQxc+ym2YMj1m/H2zDNQOlezafzkHwn6sMstjHTwG6iQQsctDW/b1A==",
"engines": {
"node": "^12.20.0 || ^14.13.1 || >=16.0.0"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/conf/node_modules/json-schema-traverse": {
"version": "1.0.0",
"resolved": "https://registry.npmmirror.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz",
"integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug=="
},
"node_modules/conf/node_modules/semver": {
"version": "7.6.2",
"resolved": "https://registry.npmmirror.com/semver/-/semver-7.6.2.tgz",
"integrity": "sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w==",
"bin": {
"semver": "bin/semver.js"
},
"engines": {
"node": ">=10"
}
},
"node_modules/config-file-ts": {
"version": "0.2.6",
"dev": true,
@ -3676,6 +3795,20 @@
"version": "1.11.10",
"license": "MIT"
},
"node_modules/debounce-fn": {
"version": "5.1.2",
"resolved": "https://registry.npmmirror.com/debounce-fn/-/debounce-fn-5.1.2.tgz",
"integrity": "sha512-Sr4SdOZ4vw6eQDvPYNxHogvrxmCIld/VenC5JbNrFwMiwd7lY/Z18ZFfo+EWNG4DD9nFlAujWAo/wGuOPHmy5A==",
"dependencies": {
"mimic-fn": "^4.0.0"
},
"engines": {
"node": ">=12"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/debug": {
"version": "4.3.4",
"license": "MIT",
@ -3887,6 +4020,31 @@
"node_modules/dom-walk": {
"version": "0.1.2"
},
"node_modules/dot-prop": {
"version": "8.0.2",
"resolved": "https://registry.npmmirror.com/dot-prop/-/dot-prop-8.0.2.tgz",
"integrity": "sha512-xaBe6ZT4DHPkg0k4Ytbvn5xoxgpG0jOS1dYxSOwAHPuNLjP3/OzN0gH55SrLqpx8cBfSaVt91lXYkApjb+nYdQ==",
"dependencies": {
"type-fest": "^3.8.0"
},
"engines": {
"node": ">=16"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/dot-prop/node_modules/type-fest": {
"version": "3.13.1",
"resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-3.13.1.tgz",
"integrity": "sha512-tLq3bSNx+xSpwvAJnzrK0Ep5CLNWjvFTOp71URMaAEWBfRb9nnJiBoUe0tF8bI4ZFO3omgBR6NvnbzVUT3Ly4g==",
"engines": {
"node": ">=14.16"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/dotenv": {
"version": "9.0.2",
"dev": true,
@ -4109,6 +4267,32 @@
"node": ">= 10.0.0"
}
},
"node_modules/electron-store": {
"version": "9.0.0",
"resolved": "https://registry.npmmirror.com/electron-store/-/electron-store-9.0.0.tgz",
"integrity": "sha512-7LZ2dR3N3bF93G2c4x+1NZ/8fpsKv6bKrMxeOQWLqdRbxvopxVqy9QXQy9axSV2O3P1kVGTj1q2wq5/W4isiOg==",
"dependencies": {
"conf": "^12.0.0",
"type-fest": "^4.18.1"
},
"engines": {
"node": ">=20"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/electron-store/node_modules/type-fest": {
"version": "4.18.2",
"resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-4.18.2.tgz",
"integrity": "sha512-+suCYpfJLAe4OXS6+PPXjW3urOS4IoP9waSiLuXfLgqZODKw/aWwASvzqE886wA0kQgGy0mIWyhd87VpqIy6Xg==",
"engines": {
"node": ">=16"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/electron-to-chromium": {
"version": "1.4.628",
"license": "ISC"
@ -4615,7 +4799,6 @@
},
"node_modules/fast-deep-equal": {
"version": "3.1.3",
"dev": true,
"license": "MIT"
},
"node_modules/fast-diff": {
@ -4810,6 +4993,7 @@
},
"node_modules/fs-minipass": {
"version": "2.1.0",
"devOptional": true,
"license": "ISC",
"dependencies": {
"minipass": "^3.0.0"
@ -4820,6 +5004,7 @@
},
"node_modules/fs-minipass/node_modules/minipass": {
"version": "3.3.6",
"devOptional": true,
"license": "ISC",
"dependencies": {
"yallist": "^4.0.0"
@ -4830,10 +5015,12 @@
},
"node_modules/fs-minipass/node_modules/yallist": {
"version": "4.0.0",
"devOptional": true,
"license": "ISC"
},
"node_modules/fs.realpath": {
"version": "1.0.0",
"devOptional": true,
"license": "ISC"
},
"node_modules/function-bind": {
@ -4926,6 +5113,7 @@
},
"node_modules/glob": {
"version": "7.2.3",
"devOptional": true,
"license": "ISC",
"dependencies": {
"fs.realpath": "^1.0.0",
@ -4955,6 +5143,7 @@
},
"node_modules/glob/node_modules/brace-expansion": {
"version": "1.1.11",
"devOptional": true,
"license": "MIT",
"dependencies": {
"balanced-match": "^1.0.0",
@ -4963,6 +5152,7 @@
},
"node_modules/glob/node_modules/minimatch": {
"version": "3.1.2",
"devOptional": true,
"license": "ISC",
"dependencies": {
"brace-expansion": "^1.1.7"
@ -5091,6 +5281,7 @@
},
"node_modules/has-flag": {
"version": "4.0.0",
"dev": true,
"license": "MIT",
"engines": {
"node": ">=8"
@ -5155,6 +5346,7 @@
},
"node_modules/hosted-git-info": {
"version": "4.1.0",
"dev": true,
"license": "ISC",
"dependencies": {
"lru-cache": "^6.0.0"
@ -5165,6 +5357,7 @@
},
"node_modules/hosted-git-info/node_modules/lru-cache": {
"version": "6.0.0",
"dev": true,
"license": "ISC",
"dependencies": {
"yallist": "^4.0.0"
@ -5175,6 +5368,7 @@
},
"node_modules/hosted-git-info/node_modules/yallist": {
"version": "4.0.0",
"dev": true,
"license": "ISC"
},
"node_modules/html-encoding-sniffer": {
@ -5334,6 +5528,7 @@
},
"node_modules/inflight": {
"version": "1.0.6",
"devOptional": true,
"license": "ISC",
"dependencies": {
"once": "^1.3.0",
@ -5680,6 +5875,11 @@
"dev": true,
"license": "MIT"
},
"node_modules/json-schema-typed": {
"version": "8.0.1",
"resolved": "https://registry.npmmirror.com/json-schema-typed/-/json-schema-typed-8.0.1.tgz",
"integrity": "sha512-XQmWYj2Sm4kn4WeTYvmpKEbyPsL7nBsb647c7pMe6l02/yx2+Jfc4dT6UZkEXnIUb5LhD55r2HPsJ1milQ4rDg=="
},
"node_modules/json-stable-stringify-without-jsonify": {
"version": "1.0.1",
"dev": true,
@ -6017,6 +6217,17 @@
"node": ">= 0.6"
}
},
"node_modules/mimic-fn": {
"version": "4.0.0",
"resolved": "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-4.0.0.tgz",
"integrity": "sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw==",
"engines": {
"node": ">=12"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/mimic-response": {
"version": "1.0.1",
"license": "MIT",
@ -6032,6 +6243,7 @@
},
"node_modules/minimatch": {
"version": "5.1.6",
"dev": true,
"license": "ISC",
"dependencies": {
"brace-expansion": "^2.0.1"
@ -6049,6 +6261,7 @@
},
"node_modules/minipass": {
"version": "5.0.0",
"devOptional": true,
"license": "ISC",
"engines": {
"node": ">=8"
@ -6056,6 +6269,7 @@
},
"node_modules/minizlib": {
"version": "2.1.2",
"devOptional": true,
"license": "MIT",
"dependencies": {
"minipass": "^3.0.0",
@ -6067,6 +6281,7 @@
},
"node_modules/minizlib/node_modules/minipass": {
"version": "3.3.6",
"devOptional": true,
"license": "ISC",
"dependencies": {
"yallist": "^4.0.0"
@ -6077,10 +6292,12 @@
},
"node_modules/minizlib/node_modules/yallist": {
"version": "4.0.0",
"devOptional": true,
"license": "ISC"
},
"node_modules/mkdirp": {
"version": "1.0.4",
"devOptional": true,
"license": "MIT",
"bin": {
"mkdirp": "bin/cmd.js"
@ -6219,6 +6436,8 @@
"node_modules/nopt": {
"version": "5.0.0",
"license": "ISC",
"optional": true,
"peer": true,
"dependencies": {
"abbrev": "1"
},
@ -8839,6 +9058,7 @@
},
"node_modules/path-is-absolute": {
"version": "1.0.1",
"devOptional": true,
"license": "MIT",
"engines": {
"node": ">=0.10.0"
@ -9198,6 +9418,14 @@
"node": ">=0.10.0"
}
},
"node_modules/require-from-string": {
"version": "2.0.2",
"resolved": "https://registry.npmmirror.com/require-from-string/-/require-from-string-2.0.2.tgz",
"integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==",
"engines": {
"node": ">=0.10.0"
}
},
"node_modules/requires-port": {
"version": "1.0.0",
"license": "MIT"
@ -9736,6 +9964,11 @@
"url": "https://github.com/sponsors/Borewit"
}
},
"node_modules/stubborn-fs": {
"version": "1.2.5",
"resolved": "https://registry.npmmirror.com/stubborn-fs/-/stubborn-fs-1.2.5.tgz",
"integrity": "sha512-H2N9c26eXjzL/S/K+i/RHHcFanE74dptvvjM8iwzwbVcWY/zjBbgRqF3K0DY4+OD+uTTASTBvDoxPDaPN02D7g=="
},
"node_modules/sumchecker": {
"version": "3.0.1",
"license": "Apache-2.0",
@ -9748,6 +9981,7 @@
},
"node_modules/supports-color": {
"version": "7.2.0",
"dev": true,
"license": "MIT",
"dependencies": {
"has-flag": "^4.0.0"
@ -9780,6 +10014,7 @@
},
"node_modules/tar": {
"version": "6.2.0",
"devOptional": true,
"license": "ISC",
"dependencies": {
"chownr": "^2.0.0",
@ -9835,6 +10070,7 @@
},
"node_modules/tar/node_modules/yallist": {
"version": "4.0.0",
"devOptional": true,
"license": "ISC"
},
"node_modules/temp-file": {
@ -9922,6 +10158,7 @@
},
"node_modules/text-table": {
"version": "0.2.0",
"dev": true,
"license": "MIT"
},
"node_modules/timm": {
@ -10053,6 +10290,17 @@
"node": ">=14.17"
}
},
"node_modules/uint8array-extras": {
"version": "0.3.0",
"resolved": "https://registry.npmmirror.com/uint8array-extras/-/uint8array-extras-0.3.0.tgz",
"integrity": "sha512-erJsJwQ0tKdwuqI0359U8ijkFmfiTcq25JvvzRVc1VP+2son1NJRXhxcAKJmAW3ajM8JSGAfsAXye8g4s+znxA==",
"engines": {
"node": ">=18"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/undici-types": {
"version": "5.26.5",
"license": "MIT"
@ -10094,7 +10342,6 @@
},
"node_modules/uri-js": {
"version": "4.4.1",
"dev": true,
"license": "BSD-2-Clause",
"dependencies": {
"punycode": "^2.1.0"
@ -10368,8 +10615,14 @@
"webidl-conversions": "^3.0.0"
}
},
"node_modules/when-exit": {
"version": "2.1.2",
"resolved": "https://registry.npmmirror.com/when-exit/-/when-exit-2.1.2.tgz",
"integrity": "sha512-u9J+toaf3CCxCAzM/484qNAxQE75rFdVgiFEEV8Xps2gzYhf0tx73s1WXDQhkwV17E3MxRMz40m7Ekd2/121Lg=="
},
"node_modules/which": {
"version": "2.0.2",
"dev": true,
"license": "ISC",
"dependencies": {
"isexe": "^2.0.0"

View File

@ -1,6 +1,6 @@
{
"name": "laitool",
"version": "2.2.2",
"version": "2.2.4",
"description": "An Electron application with Vue",
"main": "./out/main/index.js",
"author": "example.com",
@ -29,6 +29,7 @@
"axios": "^1.6.5",
"compressing": "^1.10.0",
"crypto-js": "^4.2.0",
"electron-store": "^9.0.0",
"electron-updater": "^6.1.7",
"fluent-ffmpeg": "^2.1.2",
"highlight.js": "^11.9.0",
@ -72,7 +73,8 @@
],
"extraResources": [
"resources/package/**",
"resources/image/**",
"resources/image/style/**",
"resources/image/zhanwei.png",
"resources/scripts/model/**",
"resources/scripts/Lai.exe",
"resources/scripts/discordScript.js",

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 44 KiB

View File

@ -11,7 +11,7 @@ import shotSplit
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding="utf-8")
# sys.argv = ["C:\\Users\\27698\\Desktop\\LAITool\\resources\\scripts\\Lai.exe","-c","D:/父爱如山倒/scripts/output_crop_00001.json"]
# sys.argv = ["C:\\Users\\27698\\Desktop\\LAITool\\resources\\scripts\\Lai.exe","-c","C:/Users/27698/Desktop/测试/123/scripts/output_crop_00001.json"]
print(sys.argv)
if len(sys.argv) < 2:

View File

@ -376,11 +376,19 @@ class ImageToVideo:
proportion_height = img_height / 3200
proportion_width = img_height / 3200
if offset["name"] == "KFTypePositionY":
key_frame = offset["name"]
offset_list = ["KFTypePositionY", "KFTypePositionX"]
real_key_frame = key_frame
if key_frame == "KFTypeRandom":
# 随机获取 offset_list 中的一个数据
real_key_frame = offset_list[np.random.randint(0, 2)]
if real_key_frame == "KFTypePositionY":
offsetValue = offset["up_down"] * proportion_height
elif offset["name"] == "KFTypePositionX":
elif real_key_frame == "KFTypePositionX":
offsetValue = offset["left_right"] * proportion_width
elif offset["name"] == "KFTypeScale":
elif real_key_frame == "KFTypeScale":
offsetValue = offset["scale"]
else:
return ValueError("关键帧没有设置正确的参数")
@ -402,7 +410,7 @@ class ImageToVideo:
end_offset,
self.fps,
self.video_size,
offset["name"],
real_key_frame,
)
video_arr.append(video_path)
print(video_path)

View File

@ -0,0 +1,2 @@
@echo off
pyinstaller Lai.spec

View File

@ -4,7 +4,6 @@
{
"curveType": "Line",
"graphID": "",
"id": "389CE3AB-19DF-4EC8-B712-454C224D0D92",
"left_control": {
"x": 0.0,
"y": 0.0
@ -21,7 +20,6 @@
{
"curveType": "Line",
"graphID": "",
"id": "3F5A033A-9088-4AED-BBF6-50DEBE67BCA5",
"left_control": {
"x": 0.0,
"y": 0.0

116
src/api/apiBasic.js Normal file
View File

@ -0,0 +1,116 @@
const { net } = require('electron');
let basicApi = {
/**
* 使用electron的net模块实现的get方法
* @param {*} url 请求的url
* @param {*} headers 请求头
* @returns
*/
get: (url, headers = {}) => {
return new Promise((resolve, reject) => {
const request = net.request({ url, method: 'GET', headers });
request.on('response', (response) => {
let data = '';
response.on('data', (chunk) => {
data += chunk;
});
response.on('end', () => {
// 结束的时候检查请求的状态码,是不是成功的请求,不是返回错误,有些其他的状态码也是成功的请求,并且返回错误提示
if (response.statusCode != 200) {
reject(new Error(`请求失败,状态码:${response.statusCode},错误信息:${response.statusMessage}`));
return;
}
let parsedData;
if (response.headers['content-type'].includes('application/json')) {
parsedData = JSON.parse(data);
} else {
parsedData = data;
}
resolve({
data: parsedData,
status: response.statusCode,
statusText: response.statusMessage,
headers: response.headers,
});
});
response.on('error', (error) => {
reject(error);
});
});
request.on('error', (error) => {
console.log('request error', error);
reject(error);
});
request.end();
});
},
/**
* 使用electron的net模块实现的post方法
* @param {*} url 请求的url
* @param {*} data 传输的数据json格式
* @param {*} headers 请求头
* @returns
*/
post: (url, data = {}, headers = {}) => {
return new Promise((resolve, reject) => {
const request = net.request({
method: 'POST',
url: url,
headers: {
'Content-Type': 'application/json',
...headers
}
});
request.write(JSON.stringify(data));
request.on('response', (response) => {
let responseData = '';
response.on('data', (chunk) => {
responseData += chunk;
});
response.on('end', () => {
if (response.statusCode != 200) {
reject(new Error(`请求失败,状态码:${response.statusCode},错误信息:${response.statusMessage}`));
return;
}
let parsedData;
if (response.headers['content-type'].includes('application/json')) {
parsedData = JSON.parse(responseData);
} else {
parsedData = responseData;
}
resolve({
data: parsedData,
status: response.statusCode,
statusText: response.statusMessage,
headers: response.headers
});
});
response.on('error', (error) => {
console.log('error', error);
reject(error);
});
});
request.on('error', (error) => {
console.log('request error', error);
reject(error);
});
request.end();
});
}
}
export {
basicApi
}

90
src/api/sdApi.js Normal file
View File

@ -0,0 +1,90 @@
import { basicApi } from "./apiBasic";
import { define } from "../define/define";
import { promises as fspromises } from 'fs'
import { errorMessage, successMessage } from "../main/generalTools";
export class SdApi {
constructor() {
this.baseUrl = global.config?.webui_api_url;
this.sd_setting = null;
}
/**
* 获取当前SD的服务器中所有的lora信息
* @returns
*/
async getAllLoras(baseURL = null) {
let url = this.baseUrl + "sdapi/v1/loras";
if (baseURL != null) {
url = baseURL + "sdapi/v1/loras";
}
return await basicApi.get(url);
}
/**
* 获取当前的所有的checkpoint模型
* @param {*} baseURL
*/
async getAllSDModel(baseURL = null) {
let url = this.baseUrl + "sdapi/v1/sd-models";
if (baseURL != null) {
url = baseURL + "sdapi/v1/sd-models";
}
return await basicApi.get(url);
}
/**
* 获取当前连接的所有的samplers采样器
* @param {*} baseURL
* @returns
*/
async getAllSamplers(baseURL = null) {
try {
let url = this.baseUrl + "sdapi/v1/samplers";
if (baseURL != null) {
url = baseURL + "sdapi/v1/samplers";
}
return await basicApi.get(url);
} catch (error) {
throw error;
}
}
async txt2img(data, baseURL = null) {
try {
if (this.sd_setting == null) {
this.sd_setting = JSON.parse(await fspromises.readFile(define.sd_setting, 'utf-8'));
this.baseUrl = this.sd_setting.setting.webui_api_url;
}
// 加上通用前缀
data.prompt = this.sd_setting.webui.prompt + data.prompt
data.negative_prompt = this.sd_setting.webui.negative_prompt;
data.sampler_name = this.sd_setting.webui.sampler_name;
data.cfg_scale = this.sd_setting.webui.cfg_scale;
data.n_iter = 1;
data.steps = this.sd_setting.webui.steps;
data.save_images = false;
data.batch_size = data.batch_size ? data.batch_size : 1;
if (data.width == null) {
data.width = 512;
}
if (data.height == null) {
data.height = 512;
}
let url = this.baseUrl + "sdapi/v1/txt2img";
if (baseURL != null) {
url = baseURL + "sdapi/v1/txt2img";
}
let res = await basicApi.post(url, data);
return res;
} catch (error) {
throw error;
}
}
}

View File

@ -15,6 +15,7 @@ if (!app.isPackaged) {
scripts_path: path.join(__dirname, "../../resources/scripts"),
package_path: path.join(__dirname, "../../resources/package"),
image_path: path.join(__dirname, "../../resources/image"),
temp_sd_image: path.join(__dirname, "../../resources/image/TempSDImage"),
draft_temp_path: path.join(__dirname, "../../resources/tmp/temp.zip"),
clip_speed_temp_path: path.join(__dirname, "../../resources/tmp/Clip/speeds_tmp.json"),
add_canvases_temp_path: path.join(__dirname, "../../resources/tmp/Clip/canvases_tmp.json"),
@ -45,6 +46,7 @@ if (!app.isPackaged) {
package_path: path.join(__dirname, "../../../resources/package"),
discordScript: path.join(__dirname, "../../../resources/scripts/discordScript.js"),
image_path: path.join(__dirname, "../../../resources/image"),
temp_sd_image: path.join(__dirname, "../../../resources/image/TempSDImage"),
draft_temp_path: path.join(__dirname, "../../../resources/tmp/temp.zip"),
clip_speed_temp_path: path.join(__dirname, "../../../resources/tmp/Clip/speeds_tmp.json"),
add_canvases_temp_path: path.join(__dirname, "../../../resources/tmp/Clip/canvases_tmp.json"),

View File

@ -1,4 +1,5 @@
export const DEFINE_STRING = {
GET_FILE_BASE64: "GET_FILE_BASE64",
SAVE_DEFINE_CONFIG_JSON_BY_PROPERTY: "SAVE_DEFINE_CONFIG_JSON_BY_PROPERTY",
GET_DEFINE_CONFIG_JSON_BY_PROPERTY: "GET_DEFINE_CONFIG_JSON_BY_PROPERTY",
GET_IMAGE_GENERATE_CATEGORY: "GET_IMAGE_GENERATE_CATEGORY",
@ -134,6 +135,10 @@ export const DEFINE_STRING = {
NORMAL_PERMISSION: "NORMAL_PERMISSION",
AUTO_SAVE_IMAGE_PERMISSION: "AUTO_SAVE_IMAGE_PERMISSION",
},
SD: {
LOAD_SD_SERVICE_DATA: "LOAD_SD_SERVICE_DATA",
TXT2IMG: "TXT2IMG",
},
MJ: {
SAVE_WORD_SRT: "SAVE_WORD_SRT",
GET_MJ_CONFIG_SRT_INFORMATION: "GET_MJ_CONFIG_SRT_INFORMATION",

View File

@ -39,6 +39,9 @@ export const ClipSetting = {
}, {
label: "缩放",
value: "KFTypeScale"
}, {
label: "随机",
value: "KFTypeRandom"
}]
}
},

View File

@ -1,6 +1,9 @@
import { get } from "lodash";
import { define } from "../define";
let fspromises = require("fs").promises;
import { Tools } from "../../main/tools";
import { errorMessage } from "../../main/generalTools";
let tools = new Tools();
// Create a shared object
export const SdSettingDefine = {
@ -42,6 +45,26 @@ export const SdSettingDefine = {
return get(setting, "webui", null);
} else
return get(setting.webui, property, null);
},
/**
* 保存一个大的属性添加到配置文件中
* @param {*} property
* @param {*} value
*/
SavePropertyValue: async function (property, value) {
try {
global.fileQueue.enqueue(async () => {
// 复制文件到指定的文件夹
await tools.writeJsonFilePropertyValue(define.sd_setting, property, value);
})
} catch (error) {
throw error;
}
}
};

View File

@ -2,11 +2,14 @@
let fspromises = require('fs').promises;
import { get, cloneDeep } from 'lodash';
import { define } from './define';
import path from 'path';
import { Tools } from '../main/tools';
const { v4: uuidv4 } = require('uuid');
export class TagDefine {
constructor(global) {
this.global = global;
this.tools = new Tools();
}
/**
@ -54,6 +57,30 @@ export class TagDefine {
else {
throw new Error(`不存在的类型 : ${value}`);
}
// 返回之前,判断里面是不是有预览图片路径
if (res.hasOwnProperty("character_tags")) {
res.character_tags.forEach(item => {
if (item.show_image && item.show_image != "") {
item.show_image = path.join(define.image_path, item.show_image);
}
});
}
if (res.hasOwnProperty("scene_tags")) {
res.scene_tags.forEach(item => {
if (item.show_image && item.show_image != "") {
item.show_image = path.join(define.image_path, item.show_image);
}
});
}
if (res.hasOwnProperty("style_tags")) {
res.style_tags.forEach(item => {
if (item.show_image && item.show_image != "") {
item.show_image = path.join(define.image_path, item.show_image);
}
});
}
return {
code: 1,
data: res
@ -75,6 +102,22 @@ export class TagDefine {
try {
let property = value[1];
value = JSON.parse(value[0]);
let tmp_key = uuidv4();
// 特殊操作。为角色和场景的时候需要copy图片
if (property == "character_tags" || property == "scene_tags" || property == "style_tags") {
let show_image = value.show_image;
if (show_image && show_image != "") {
let file_name = `c_s/${value.key ? value.key : tmp_key}.png`
let new_image_path = path.join(define.image_path, file_name);
await this.tools.copyFileOrDirectory(show_image, new_image_path);
value.show_image = file_name;
value.children?.forEach(item => {
item.show_image = file_name;
});
}
}
// 获取自定义的GPT数据
let tag_setting = JSON.parse(await fspromises.readFile(define.tag_setting, 'utf-8'));
let tag = get(tag_setting, property, []);
@ -96,7 +139,7 @@ export class TagDefine {
if (tag.some(item => item.label == value.label)) {
throw new Error("已存在相同名称的数据,请修改名称后再保存");
}
value.key = uuidv4();
value.key = tmp_key;
value.value = value.key;
tag.push(value);
}
@ -107,7 +150,6 @@ export class TagDefine {
code: 1,
message: "保存成功"
}
} catch (error) {
return {
code: 0,

View File

@ -0,0 +1,34 @@
import {
ipcMain
} from "electron";
import { DEFINE_STRING } from '../../define/define_string'
import { Tools } from "../tools";
import path from "path";
import { errorMessage, successMessage } from "../generalTools";
let tools = new Tools();
function GlobalIpc() {
/**
* 将传入的文件地址修改为base64
*/
ipcMain.handle(DEFINE_STRING.GET_FILE_BASE64, async (event, value) => {
try {
value = path.normalize(value)
//检查文件或者时文件夹是不是存在
let isExists = await tools.checkExists(value);
console.log("isExists", value, isExists);
// 获取文件将其转换为base64
if (!isExists) {
throw new Error("文件不存在");
}
return successMessage(await tools.readFileBase64(value));
} catch (error) {
return errorMessage("获取文件失败" + error)
}
});
}
export {
GlobalIpc
}

View File

@ -12,6 +12,12 @@ function SdIpc() {
// 获取图片样式菜单
ipcMain.handle(DEFINE_STRING.GET_IMAGE_STYLE_MENU, async (event) => await sd.GetImageStyleMenu());
// 加载当前链接的SD服务数据
ipcMain.handle(DEFINE_STRING.SD.LOAD_SD_SERVICE_DATA, async (event, value) => await sd.LoadSDServiceData(value));
// 文生图,单张
ipcMain.handle(DEFINE_STRING.SD.TXT2IMG, async (event, value) => await sd.txt2img(value));
}
export {
SdIpc

View File

@ -9,6 +9,7 @@ import path from 'path'
import sharp from 'sharp'
import { define } from "../../define/define";
import { AwesomeRegx } from "awesome-js";
import { checkStringValueAddSuffix } from "../generalTools";
/**
* MJ原创生图的类
@ -196,13 +197,15 @@ export class MJOriginalImageGenerate {
// 判断存放的文件夹是不是存在,不存在的话创建
let outputDir = path.join(this.global.config.project_path, `data\\MJOriginalImage`);
await this.tools.checkFolderExistsOrCreate(outputDir);
let fileExist = await this.tools.checkExists(outputDir);
if (!fileExist) {
await this.tools.createDirectory(outputDir);
}
// 判断该当前tmp\output_crop_00001文件夹是不是存在不存在创建
let output_crop_00001 = path.join(this.global.config.project_path, `tmp\\output_crop_00001`);
await this.tools.checkFolderExistsOrCreate(output_crop_00001);
// 检查this.global中是不是又mj队列没有的话创建一个
if (!this.global.mjGenerateQuene) {
@ -210,7 +213,7 @@ export class MJOriginalImageGenerate {
}
let style_ids = await this.pm.GetConfigJson(JSON.stringify(["image_style", []]), false);
// let image_styles = await ImageStyleDefine.getImageStyleStringByIds(style_ids.data);
let image_styles = await ImageStyleDefine.getImageStyleStringByIds(style_ids.data);
// 替换风格的逻辑
let current_task = null;
@ -219,11 +222,17 @@ export class MJOriginalImageGenerate {
const element = data[i];
let tasK_id = `${batch}_${element.name}_${element.id}`;
let old_prompt = element.prompt;
// 拼接提示词
// 图生图的链接
// 获取风格词
let prompt = " " + image_styles + old_prompt;
this.global.mjGenerateQuene.enqueue(async () => {
try {
this.global.mjGenerateQuene.setCurrentCreateItem(element)
// 开始进行mj生图
let prompt = element.prompt;
current_task = element.name;
// 判断窗口是不是开启

View File

@ -12,7 +12,6 @@ const { spawn, exec } = require('child_process');
const execAsync = util.promisify(exec);
const { v4: uuidv4 } = require('uuid'); // 引入UUID库来生成唯一标识符
let fspromises = require("fs").promises;
import { MD5 } from "crypto-js";
import { ImageStyleDefine } from "../../define/iamgeStyleDefine";
@ -64,26 +63,21 @@ export class OriginalImageGenerate {
let style_ids = await this.pm.GetConfigJson(JSON.stringify(["image_style", []]), false);
let image_styles = await ImageStyleDefine.getImageStyleStringByIds(style_ids.data);
//
console.log(image_styles);
for (let i = 0; i < data.length; i++) {
const element = data[i];
let adetailer = element.adetailer;
let imageJson = JSON.parse(await fspromises.readFile(path.normalize(element.prompt_json), 'utf-8'));
// let prompt = image_styles + sd_setting.webui.prompt + ',' + element.prompt;
// // 添加前缀
// if (prefix_prompt) {
// prompt = prefix_prompt + ',' + prompt;
// }
// // 添加后缀
// if (suffix_prompt) {
// prompt = prompt + ',' + suffix_prompt;
// }
let prompt = imageJson.webui_config.prompt;
let prompt = sd_setting.webui.prompt + image_styles + ',' + element.prompt;
// 添加前缀
if (prefix_prompt) {
prompt = prefix_prompt + ',' + prompt;
}
// 添加后缀
if (suffix_prompt) {
prompt = prompt + ',' + suffix_prompt;
}
// let prompt = imageJson.webui_config.prompt;
this.global.requestQuene.enqueue(async () => {
try {
// 开始请求
@ -91,7 +85,7 @@ export class OriginalImageGenerate {
"prompt": prompt,
"negative_prompt": imageJson.webui_config.negative_prompt,
"seed": seed,
"sampler_name": imageJson.webui_config.sampler_name,
"sampler_name": sd_setting.webui.sampler_name,
// 提示词相关性
"cfg_scale": sd_setting.webui.cfg_scale,
"width": sd_setting.webui.width,
@ -258,7 +252,15 @@ export class OriginalImageGenerate {
if (element.outImagePath && file_regex.test(element.outImagePath)) {
// 删除 "file://" 开头
element.outImagePath = decodeURI(element.outImagePath);
element.outImagePath = element.outImagePath.replace(/^file:\/\//, '').replace(/\?time=.*$/, '');
// 判断element.outImagePath是不是不是以file://开头的,是的话,删除
if (element.outImagePath.startsWith("file://")) {
element.outImagePath = element.outImagePath.substring(7);
}
element.outImagePath = element.outImagePath.replace(/\?time=.*$/, '');
// 判断element.outImagePath是不是以/开头的,是的话,删除
if (element.outImagePath.startsWith("/")) {
element.outImagePath = element.outImagePath.substring(1);
}
}
if (element.subImagePath && element.subImagePath.length > 0) {
for (let j = 0; j < element.subImagePath.length; j++) {
@ -308,9 +310,7 @@ export class OriginalImageGenerate {
data: path.join(this.global.config.project_path, "tmp/input_crop")
}
}
let sd_config = JSON.parse(await fspromises.readFile(define.sd_setting, 'utf-8'));
for (let i = 0; i < data.length; i++) {
const element = data[i];
let name = String(element.no).padStart(5, '0') + ".png";

View File

@ -6,19 +6,112 @@ import { ImageStyleDefine } from "../../define/iamgeStyleDefine";
import { cloneDeep } from 'lodash';
let fspromises = require("fs").promises;
const sharp = require('sharp');
// const {
// createCanvas,
// loadImage
// } = require('canvas');
import { SdSettingDefine } from "../../define/setting/sdSettingDefine";
import { PublicMethod } from "./publicMethod";
import { Tools } from "../tools";
import { errorMessage, successMessage } from "../generalTools";
import { SdApi } from "../../api/sdApi";
const { v4: uuidv4 } = require('uuid');
export class SD {
constructor(global) {
this.global = global;
this.pm = new PublicMethod(global);
this.tools = new Tools();
this.sdApi = new SdApi();
}
/**
* 获取当前SD服务器所有的lora信息
*/
async GetAllLoras(baseURL = null) {
try {
let data = await this.sdApi.getAllLoras(baseURL);
return successMessage(data);
} catch (error) {
return errorMessage(error.toString());
}
}
/**
* 获取所有的checkpoint模型
* @param {*} baseURL
* @returns
*/
async GetAllSDModel(baseURL = null) {
try {
let data = await this.sdApi.getAllSDModel(baseURL);
return successMessage(data);
} catch (error) {
return errorMessage(error.toString());
}
}
/**
* 获取所有的采样器
* @param {*} baseURL
* @returns
*/
async GetAllSamplers(baseURL = null) {
try {
let data = await this.sdApi.getAllSamplers(baseURL);
return successMessage(data);
} catch (error) {
return errorMessage(error.toString());
}
}
/**
* 加载所有的SD数据
* @param {*} baseURL
* @returns
*/
async LoadSDServiceData(baseURL = null) {
try {
// 加载大模型
let sd_model = await this.GetAllSDModel(baseURL);
// 往sd_model中添加一个默认的选项
sd_model.data.data.unshift({
title: "无",
name: "无",
description: "无",
})
// 加载Lora
let lora = await this.GetAllLoras(baseURL);
lora.data.data.unshift({
Key: "无",
name: "无",
description: "无",
})
// 加载采样器
let sampler = await this.GetAllSamplers(baseURL);
sampler.data.data.unshift({
name: "无",
description: "无",
})
if (!(sd_model.code & lora.code & sampler.code)) {
throw new Error("获取SD数据错误请检查SD WEBUI链接");
}
for (let i = 0; i < lora.data.data.length; i++) {
delete lora.data.data[i].metadata;
}
let data = {
sd_model: sd_model.data.data,
lora: lora.data.data,
sampler: sampler.data.data
}
// 处理当前获取的数据,保存到配置文件中
await SdSettingDefine.SavePropertyValue("sd_model", data.sd_model);
await SdSettingDefine.SavePropertyValue("lora", data.lora);
await SdSettingDefine.SavePropertyValue("sampler", data.sampler);
return successMessage(data);
} catch (error) {
return errorMessage(error.toString());
}
}
/**
@ -34,7 +127,10 @@ export class SD {
data: style
}
} catch (error) {
return {
code: 0,
message: "不可能出现错误"
}
}
}
@ -106,6 +202,37 @@ export class SD {
}
}
/**
* 单张生图
* @param {*} value 0 生图的参数1 图片的表示用于保存 2 baseUrl
* @returns
*/
async txt2img(value) {
try {
value = JSON.parse(value);
let data = value[0];
let res = await this.sdApi.txt2img(data);
// 将base· 64的图片转换为图片
// 将当前的图片保存到指定的文件夹中然后返回文件路径并且可以复制到指定的文件删除exif信息
let image_paths = [];
for (let i = 0; res.data.images && i < res.data.images.length; i++) {
const element = res.data.images[i];
let image_data = {
base64: element
}
// 将保存图片添加到队列中
let image_name = `sd_${Date.now()}_${uuidv4()}.png`;
let image_path = path.join(define.temp_sd_image, image_name);
image_path = await this.tools.saveBase64ToImage(element, image_path);
image_data["image_path"] = image_path;
image_paths.push(image_data);
}
return successMessage(image_paths);
} catch (error) {
return errorMessage("生图失败,错误信息如下:" + error.toString());
}
}
/**
* 生成一次图片的方法可以区分模式
* @param {图片名称 } image
@ -120,7 +247,6 @@ export class SD {
let image_json = JSON.parse(await fspromises.readFile(image + '.json', 'utf-8'));
let image_path = "";
let target_image_path = "";
if (image_json.name) {
image_path = path.join(this.global.config.project_path, `tmp/${task_list.out_folder}/tmp_${image_json.name}`)
target_image_path = path.join(this.global.config.project_path, `tmp/${task_list.out_folder}/${image_json.name}`)
@ -128,25 +254,20 @@ export class SD {
image_path = image.replaceAll("input_crop", task_list.out_folder).split(".png")[0] + "_tmp.png";
target_image_path = image.replaceAll("input_crop", task_list.out_folder);
}
// let prompt = "";
// // 拼接提示词
// if (task_list.image_style != null) {
// prompt += `((${task_list.image_style})),`;
// }
// if (task_list.lora != null) {
// prompt += `${task_list.lora},`;
// }
// let image_styles = await ImageStyleDefine.getImageStyleStringByIds(task_list.image_style_list ? task_list.image_style_list : []);
// prompt = `${prompt}, ${image_styles}, ${imageJson.webui_config.prompt}`;
let prompt = imageJson.webui_config.prompt;
let image_styles = await ImageStyleDefine.getImageStyleStringByIds(task_list.image_style_list ? task_list.image_style_list : []);
let prompt = sd_setting.webui.prompt + image_styles;
// 拼接提示词
if (task_list.image_style != null) {
prompt += `((${task_list.image_style})), `;
}
if (task_list.lora != null) {
prompt += `${task_list.lora}, `;
}
prompt += imageJson.webui_config.prompt;
// 判断当前是不是有开修脸修手
let ADetailer = {
args: sd_setting.adetailer
};
if (model == "img2img") {
let web_api = this.global.config.webui_api_url + 'sdapi/v1/img2img'
let sd_config = imageJson["webui_config"];
@ -154,7 +275,6 @@ export class SD {
sd_config.seed = seed;
let im = await fspromises.readFile(image, 'binary');
sd_config.init_images = [new Buffer.from(im, 'binary').toString('base64')];
if (imageJson.adetailer) {
let ta = {
ADetailer: ADetailer
@ -163,13 +283,11 @@ export class SD {
}
sd_config.height = sd_setting.webui.height;
sd_config.width = sd_setting.webui.width;
const response = await axios.post(web_api, sd_config);
let info = JSON.parse(response.data.info);
if (seed == -1) {
seed = info.seed;
}
// 目前是单图出图
let images = response.data.images;
let imageData = Buffer.from(images[0].split(",", 1)[0], 'base64');
@ -253,22 +371,4 @@ export class SD {
throw error;
}
}
/**
*文生图
* @param {SD 请求的地址} url
* @param {SD请求的body} body
*/
async txt2img(url, body) {
}
/**
*图生图
* @param {SD 请求的地址} url
* @param {SD请求的body} body
*/
async img2img(url, body) {
}
}

View File

@ -662,12 +662,12 @@ export class ClipDraft {
* 通过 key_frame 返回关键帧数据
* @param {*} key_frame 关键帧配置
*/
async GetFrameData(key_frame) {
if (key_frame.key_frame == "KFTypePositionY") {
async GetFrameData(key_frame_key, key_frame) {
if (key_frame_key == "KFTypePositionY") {
return key_frame.up_down_key_frame;
} else if (key_frame.key_frame == "KFTypePositionX") {
} else if (key_frame_key == "KFTypePositionX") {
return key_frame.left_right_key_frame;
} else if (key_frame.key_frame == "KFTypeScale") {
} else if (key_frame_key == "KFTypeScale") {
return key_frame.scale_key_frame;
} else {
return {
@ -683,7 +683,8 @@ export class ClipDraft {
*/
async AddKeyFarme() {
let img_nodes = (await this.find_draft_node(this.draft_json.tracks, "type", "video")).segments;
let key_frame_tmp_data = JSON.parse(await fspromises.readFile(define.add_keyframe_tmp_path, "utf-8"));
let key_frame_tmp_data = await fspromises.readFile(define.add_keyframe_tmp_path, "utf-8");
// 添加关键帧
// 将最后一个数据修改为背景音乐的最后时间
this.srt_information[this.srt_information.length - 1].end_time = this.audios_duration_time / 1000;
@ -693,34 +694,42 @@ export class ClipDraft {
if (key_frame_setting == null) {
return;
}
let key_frame_pos = await this.GetFrameData(key_frame_setting);
let isFixedSpeed = key_frame_setting.isFixedSpeed;
let key_frame_time = key_frame_setting.key_frame_time * 1000000;
let isDown = true;
let scale_rate = key_frame_pos.default_scale / 100;
// 获取通用的关键帧配置,然后添加到每一个图片上面(可以设置时间。当前图片的持续实现小于设置的时间。会计算不要过快)
for (let i = 0; i < img_nodes.length; i++) {
let element = img_nodes[i];
let image_duartion = cloneDeep(element.source_timerange.duration);
let key_frame = key_frame_setting.key_frame;
// 做随机处理
let real_key_frame = key_frame;
let key_frame_list = ["KFTypePositionY", "KFTypePositionX", "KFTypeScale"];
if (key_frame == "KFTypeRandom") {
// 随机获取 key_frame_list 里面的一个数据
const randomIndex = Math.floor(Math.random() * key_frame_list.length);
real_key_frame = key_frame_list[randomIndex];
}
let key_frame_pos = await this.GetFrameData(real_key_frame, key_frame_setting);
let key_frame_time = key_frame_setting.key_frame_time * 1000000;
let scale_rate = key_frame_pos.default_scale / 100;
let up_pos = Math.abs(key_frame_pos.start_position);
let down_pos = Math.abs(key_frame_pos.end_position);
let key_frame_tmp = JSON.parse(key_frame_tmp_data)
if (key_frame_setting.key_frame == "KFTypePositionY") {
if (real_key_frame == "KFTypePositionY") {
// 勾选了匀速。需要计算时间(计算比例)
if (isFixedSpeed && image_duartion < key_frame_time) {
let time_rate = image_duartion / key_frame_time;
up_pos = up_pos * time_rate;
down_pos = down_pos * time_rate;
}
let key_frame_tmp = cloneDeep(key_frame_tmp_data)
let up_pos_rate = isDown ? (up_pos / this.draft_json.canvas_config.height) : (0 - up_pos / this.draft_json.canvas_config.height)
key_frame_tmp.id = uuidv4();
key_frame_tmp.keyframe_list[0].id == uuidv4();
key_frame_tmp.keyframe_list[0].id = uuidv4(); // Corrected assignment
key_frame_tmp.keyframe_list[0].values = [up_pos_rate];
let dow_pos_rate = isDown ? (0 - down_pos / this.draft_json.canvas_config.height) : (down_pos / this.draft_json.canvas_config.height)
@ -728,22 +737,20 @@ export class ClipDraft {
key_frame_tmp.keyframe_list[1].time_offset = image_duartion;
key_frame_tmp.keyframe_list[1].values = [dow_pos_rate];
key_frame_tmp.property_type = key_frame_setting.key_frame;
key_frame_tmp.property_type = real_key_frame;
// 修改缩放倍率
element.clip.scale.x = scale_rate;
element.clip.scale.y = scale_rate;
element.clip.transform.y = dow_pos_rate;
isDown = !isDown;
element.common_keyframes.push(key_frame_tmp);
} else if (key_frame_setting.key_frame == "KFTypePositionX") {
} else if (real_key_frame == "KFTypePositionX") {
// 勾选了匀速。需要计算时间(计算比例)
if (isFixedSpeed && image_duartion < key_frame_time) {
let time_rate = image_duartion / key_frame_time;
up_pos = up_pos * time_rate;
down_pos = down_pos * time_rate;
}
let key_frame_tmp = cloneDeep(key_frame_tmp_data)
let up_pos_rate = isDown ? (up_pos / this.draft_json.canvas_config.width) : (0 - up_pos / this.draft_json.canvas_config.width)
key_frame_tmp.id = uuidv4();
key_frame_tmp.keyframe_list[0].id == uuidv4();
@ -754,16 +761,15 @@ export class ClipDraft {
key_frame_tmp.keyframe_list[1].time_offset = image_duartion;
key_frame_tmp.keyframe_list[1].values = [dow_pos_rate];
key_frame_tmp.property_type = key_frame_setting.key_frame;
key_frame_tmp.property_type = real_key_frame;
// 修改缩放倍率
element.clip.scale.x = scale_rate;
element.clip.scale.y = scale_rate;
element.clip.transform.x = dow_pos_rate;
isDown = !isDown;
element.common_keyframes.push(key_frame_tmp);
}
else if (key_frame_setting.key_frame == "KFTypeScale") {
else if (real_key_frame == "KFTypeScale") {
if (isFixedSpeed && image_duartion < key_frame_time) {
let time_rate = image_duartion / key_frame_time;
// 计算方式和上面的不同
@ -774,7 +780,6 @@ export class ClipDraft {
}
// 修改上面的数据添加Y轴缩放
let key_frame_tmp = cloneDeep(key_frame_tmp_data)
let up_pos_rate = isDown ? up_pos / 100 : down_pos / 100;
key_frame_tmp.id = uuidv4();
key_frame_tmp.keyframe_list[0].id == uuidv4();
@ -785,7 +790,7 @@ export class ClipDraft {
key_frame_tmp.keyframe_list[1].time_offset = image_duartion;
key_frame_tmp.keyframe_list[1].values = [dow_pos_rate];
key_frame_tmp.property_type = key_frame_setting.key_frame + "X";
key_frame_tmp.property_type = real_key_frame + "X";
// 修改上面的数据添加Y轴缩放
// 修改缩放倍率
@ -794,15 +799,15 @@ export class ClipDraft {
element.clip.transform.x = 0;
element.common_keyframes.push(key_frame_tmp);
key_frame_tmp = cloneDeep(key_frame_tmp)
key_frame_tmp = JSON.parse(JSON.stringify(key_frame_tmp))
key_frame_tmp.id = uuidv4();
key_frame_tmp.keyframe_list[0].id == uuidv4();
key_frame_tmp.keyframe_list[1].id = uuidv4();
key_frame_tmp.property_type = key_frame_setting.key_frame + "Y";
element.common_keyframes.push(key_frame_tmp);
key_frame_tmp.property_type = real_key_frame + "Y";
isDown = !isDown;
}
element.common_keyframes.push(key_frame_tmp);
}
}

View File

@ -12,7 +12,6 @@ const sharp = require('sharp');
const execAsync = util.promisify(exec);
const { v4: uuidv4 } = require('uuid'); // 引入UUID库来生成唯一标识符
let fspromises = require("fs").promises;
import { MD5 } from "crypto-js";
import { ImageSetting } from "../../define/setting/imageSetting";
@ -257,7 +256,7 @@ export class ImageGenerate {
if (images.length <= 0) {
throw new Error("未检测到抽帧图片。请检查");
}
if (images.length > auto_save_image.save_match_count) {
if (auto_save_image.save_match_count && auto_save_image.auto_match && images.length > auto_save_image.save_match_count) {
png_files = await this.tools.getFilesWithExtensions(auto_save_image.main_save_folder, '.png');
}
this.global.requestQuene.enqueue(async () => {

View File

@ -1,14 +1,6 @@
const axios = require('axios');
// import { Midjourney as mApi} from "midjourney";
// const MJapi2 = require('midjourney');
// const { midjourney } = require('midjourney')
// const Midjourney = require('midjourney');
// ES5 的模块引入方式
const fetch = require("node-fetch");
// ES6 的模块引入方式
// import fetch from "node-fetch";
export class DiscordAPI {
constructor(mj_setting) {
// https://discord.com/api/v9/channels/1208362852482809939/messages?limit=20

View File

@ -134,7 +134,7 @@ async function ReGenerateImageOne(window, value) {
let image_styles = await ImageStyleDefine.getImageStyleStringByIds(value[1].image_style_list ? value[1].image_style_list : []);
let prompt = image_styles;
let prompt = sd_setting.webui.prompt + image_styles;
// 拼接提示词
if (value[1].image_style != null) {
prompt += `((${value[1].image_style})),`;
@ -143,6 +143,8 @@ async function ReGenerateImageOne(window, value) {
prompt += `${value[1].lora},`;
}
prompt += value[1].prompt;
let model = value[1].model;
// 判断当前是不是有开修脸修手
@ -624,8 +626,12 @@ async function SaveSDConfig(value) {
try {
let sd_config = JSON.parse((await fspromises.readFile(define.sd_setting, "utf-8")).toString());
global.config.webui_api_url = value.webui_api_url || value.webui_api_url == '' ? value.webui_api_url : global.config.webui_api_url;
sd_config.setting.webui_api_url = value.webui_api_url || value.webui_api_url == "" ? value.webui_api_url : sd_config.setting.webui_api_url;
sd_config.setting.type = value.type ? value.type : sd_config.setting.type;
sd_config.setting.batch_size = value.batch_size ? value.batch_size : sd_config.setting.batch_size;
sd_config.setting.style_weight = value.style_weight ? value.style_weight : sd_config.setting.style_weight;
sd_config.webui.prompt = value.prompt || value.prompt == "" ? value.prompt : sd_config.webui.prompt;
sd_config.webui.negative_prompt = value.negative_prompt || value.negative_prompt == "" ? value.negative_prompt : sd_config.webui.negative_prompt;
sd_config.webui.denoising_strength = value.denoising_strength || value.denoising_strength == "" ? value.denoising_strength : sd_config.webui.denoising_strength;
@ -635,8 +641,6 @@ async function SaveSDConfig(value) {
sd_config.webui.height = value.height ? value.height : sd_config.webui.height;
sd_config.webui.cfg_scale = value.cfg_scale ? value.cfg_scale : sd_config.webui.cfg_scale;
sd_config.webui.adetailer = value.adetailer ? value.adetailer : sd_config.webui.adetailer;
sd_config.setting.batch_size = value.batch_size ? value.batch_size : sd_config.setting.batch_size;
sd_config.setting.style_weight = value.style_weight ? value.style_weight : sd_config.setting.style_weight;
await fspromises.writeFile(define.sd_setting, JSON.stringify(sd_config));
return {

102
src/main/generalTools.js Normal file
View File

@ -0,0 +1,102 @@
/**
* 判断字符串的值是不是存在不是null不是undefined不是空字符串存在的话添加指定的后缀
* @param {*} value 要检查的字符串
* @param {*} suffix 要添加的后缀
* @returns
*/
function checkStringValueAddSuffix(value, suffix) {
if (value && value !== null && value !== undefined && value !== '') {
return value + suffix;
} else {
return '';
}
}
/**
* 判断字符串的值是不是存在不是null不是undefined不是空字符串存在的话添加指定的前缀
* @param {*} value 要检查的值
* @param {*} prefix 要添加的前缀
* @returns
*/
function checkStringValueAddPrefix(value, prefix) {
if (value && value !== null && value !== undefined && value !== '') {
return prefix + value;
} else {
return '';
}
}
/**
* 新建一个函数判断传入的字符串的值是不是存在存在的话删除后面指定数量的字符
* @param {*} value 要删除的字符串
* @param {*} suffix 删除的后缀
* @returns
*/
function checkStringValueDeleteSuffix(value, suffix) {
// 增加一个判断,当前删除的数量是不是大于字符串的长度
if (value && value !== null && value !== undefined && value !== '') {
if (suffix.length > value.length) {
return '';
} else {
return value.slice(0, value.length - suffix.length);
}
} else {
return '';
}
}
/**
* 新建一个函数判断传入的字符串的值是不是存在存在的话删除前面指定数量的字符
* @param {*} value 操作的字符串
* @param {*} prefix 要删除的字符串
* @returns
*/
function checkStringValueDeletePrefix(value, prefix) {
// 增加一个判断,当前删除的数量是不是大于字符串的长度
if (value && value !== null && value !== undefined && value !== '') {
if (prefix.length > value.length) {
return '';
} else {
return value.slice(prefix.length);
}
} else {
return '';
}
}
/**
* 返回成功的消息包含codedatamessage
* @param {*} code
* @param {*} data
* @param {*} message
* @returns
*/
function successMessage(data, message = null) {
return {
code: 1,
data: data,
message: message
}
}
/**
* 返回失败的消息包含codemessage
* @param {*} code
* @param {*} message
* @returns
*/
function errorMessage(message) {
return {
code: 0,
message: message
}
}
export {
checkStringValueAddSuffix,
checkStringValueAddPrefix,
checkStringValueDeletePrefix,
checkStringValueDeleteSuffix,
successMessage,
errorMessage
}

View File

@ -1,3 +1,8 @@
import fspromises from "fs/promises";
import { v4 as uuidv4 } from 'uuid';
import { version } from '../../package.json'
import { app, shell, BrowserWindow, ipcMain, dialog, nativeTheme } from 'electron'
import path, { join } from 'path'
import { electronApp, optimizer, is } from '@electron-toolkit/utils'
@ -6,10 +11,6 @@ import { define } from '../define/define.js'
import { func } from './func.js'
import { AsyncQueue } from "./quene.js"
import { DEFINE_STRING } from '../define/define_string.js'
const fspromises = require("fs").promises;
const { v4: uuidv4 } = require('uuid');
const { version } = require('../../package.json')
import { Tools } from './tools.js'
import { ImageGenerate } from './backPrompt/imageGenerate.js'
import { Setting } from './setting/setting.js'
@ -27,12 +28,14 @@ import { OriginalImageGenerateIpc } from './IPCEvent/originalImageGenerateIpc'
import { SdIpc } from './IPCEvent/sdIpc.js'
import { DiscordIpc, RemoveDiscordIpc } from './IPCEvent/discordIpc.js'
import { MainIpc } from './IPCEvent/mainIpc.js'
import { GlobalIpc } from "./IPCEvent/globalIpc.js";
let tools = new Tools();
let imageGenerate = new ImageGenerate(global);
let setting = new Setting(global);
async function InitData(gl) {
let res = await setting.getSettingDafultData();
gl.config = res;
@ -55,11 +58,17 @@ function removeIpcHandler(hash) {
async function createWindow(hash = "ShowMessage", data, url = null) {
// Create the browser window.
await InitData(global);
global.currentHash = hash;
// 判断当前是不是有设置的宽高,用的话记忆
let isRe = global.config.window_wh_bm_remember && hash == "ShowMessage" && global.config.window_wh_bm;
let mainWindow = new BrowserWindow({
width: 900,
height: 670,
width: isRe ? global.config.window_wh_bm.width : 900,
height: isRe ? global.config.window_wh_bm.height : 675,
x: isRe ? global.config.window_wh_bm.x : 100,
y: isRe ? global.config.window_wh_bm.y : 100,
title: 'LAITool',
icon: '../../resources/icon.ico',
show: false,
@ -104,10 +113,16 @@ async function createWindow(hash = "ShowMessage", data, url = null) {
}
}
mainWindow.on("closed", () => {
mainWindow.on("close", async () => {
// 判断指定的窗口,移除指定的监听
removeIpcHandler(hash);
global.newWindow = global.newWindow.filter(item => item.id != mainWindow.id)
// 判断当前的是不是开启了记录功能
if (global.config.window_wh_bm_remember && hash == "ShowMessage") {
let window_wh_bm = mainWindow.getBounds();
// 记录到文件中
await setting.ModifySampleSetting(JSON.stringify({ window_wh_bm: window_wh_bm }))
}
})
// 创建一个新的窗口,添加对应的监听
@ -120,7 +135,6 @@ async function createWindow(hash = "ShowMessage", data, url = null) {
})
setIpcHandler(hash);
await InitData(global);
return mainWindow;
}
@ -141,6 +155,8 @@ app.whenReady().then(async () => {
optimizer.watchWindowShortcuts(window)
})
//
global.newWindow = [];
mainWindow = createWindow('ShowMessage', null)
@ -173,9 +189,12 @@ app.whenReady().then(async () => {
}
// 判断动态文件是不是存在
tools.checkJsonFileExistsOrCreate(path.join(define.dynamic_setting));
tools.checkJsonFileExistsOrCreate(path.normalize(define.dynamic_setting));
// 判断标签文件是不是存在
tools.checkJsonFileExistsOrCreate(path.join(define.tag_setting));
tools.checkJsonFileExistsOrCreate(path.normalize(define.tag_setting));
// 判断SD图片缓存文件是不是存在不存在创建
tools.checkFolderExistsOrCreate(path.normalize(define.temp_sd_image));
tools.checkFolderExistsOrCreate(path.normalize(path.join(define.image_path, "c_s")));
app.on('activate', function () {
// On macOS it's common to re-create a window in the app when the
@ -204,6 +223,7 @@ SdIpc();
MjIpc();
MainIpc(createWindow);
OriginalImageGenerateIpc();
GlobalIpc();
ipcMain.handle('dark-mode:toggle', (event, value) => {

View File

@ -162,6 +162,9 @@ export class Setting {
seed: sd_config.setting.seed,
style_weight: sd_config.setting.style_weight,
cfg_scale: sd_config.webui.cfg_scale,
sd_model: sd_config.sd_model,
lora: sd_config.lora,
sampler: sd_config.sampler,
}
}
} catch (error) {
@ -226,6 +229,7 @@ export class Setting {
let sd_data = await fspromises.readFile(define.sd_setting, 'utf-8');
let config_json_date = JSON.parse(data);
config_json_date.webui_api_url = JSON.parse(sd_data).setting.webui_api_url;
config_json_date["space_image"] = define.zhanwei_image;
return config_json_date;
}

View File

@ -25,6 +25,21 @@ export class Tools {
}
}
/**
* 获取指定的文件然后将其转换为base64
* @param {*} filePath 文件地址
* @returns
*/
async readFileBase64(filePath) {
try {
let data = await fspromises.readFile(filePath);
return data.toString('base64');
}
catch (error) {
throw new Error(error);
}
}
/**
* 判断json文件是不是存在不存在的话协议一个空的json文件
@ -301,7 +316,6 @@ export class Tools {
method: 'GET',
url: url
});
request.on('response', (response) => {
const chunks = [];
response.on('data', (chunk) => chunks.push(chunk));
@ -323,4 +337,29 @@ export class Tools {
request.end();
});
}
/**
* 将base64保存为图片然后删除图片的exif信息
* @param {*} base64 图片的base64数据
* @param {*} filePath 保存文件地址
* @param {*} isDeleteExif 是不是要删除exif信息默认为true
*/
async saveBase64ToImage(base64, filePath, isDeleteExif = true) {
try {
let base64Data = base64.replace(/^data:image\/\w+;base64,/, "");
let dataBuffer = Buffer.from(base64Data, 'base64');
if (isDeleteExif) {
// 先将图片写道一个缓存位置清除数据的时候将原图片的exif信息删除然后将原图片复制到目标图片地址
let dir = path.dirname(filePath);
let tempFilePath = path.join(dir, `temp_${Date.now()}.png`);
await fspromises.writeFile(tempFilePath, dataBuffer);
this.deletePngAndDeleteExifData(tempFilePath, filePath);
} else {
await fspromises.writeFile(filePath, dataBuffer);
}
return filePath;
} catch (error) {
throw error;
}
}
}

View File

@ -1,9 +1,7 @@
import { contextBridge, ipcRenderer } from 'electron'
import { ipcRenderer } from 'electron'
import { DEFINE_STRING } from '../define/define_string.js';
// Custom APIs for renderer
let events = [];
const api = {
const discord = {
// 创建MJ消息
CreateMessage: (value) => ipcRenderer.send(DEFINE_STRING.DISCORD.CREATE_MESSAGE, value),
@ -13,16 +11,7 @@ const api = {
// MJ消息删除
DeleteMessage: (value) => ipcRenderer.send(DEFINE_STRING.DISCORD.DELETE_MESSAGE, value),
}
// Use `contextBridge` APIs to expose Electron APIs to
// renderer only if context isolation is enabled, otherwise
// just add to the DOM global.
if (process.contextIsolated) {
try {
contextBridge.exposeInMainWorld('api', api)
} catch (error) {
console.error(error)
}
} else {
window.api = api
}
export {
discord
}

View File

@ -1,6 +1,9 @@
import { contextBridge, ipcRenderer } from 'electron'
import { electronAPI } from '@electron-toolkit/preload'
import { DEFINE_STRING } from '../define/define_string.js';
import { discord } from './discord.js';
import { mj } from './mj.js';
import { sd } from './sd.js';
// Custom APIs for renderer
let events = [];
@ -389,56 +392,11 @@ const api = {
// 打开全局通知框
showGlobalNotificationDialog: (value) => ipcRenderer.send(DEFINE_STRING.SHOW_MAIN_NOTIFICATION, value),
// 知道文件地址获取文件base64编码
GetFileBase64: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.GET_FILE_BASE64, value)),
}
const mj = {
// 保存文案信息到mj的配置文件
SvaeMJWordSrt: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.SAVE_WORD_SRT, value)),
// 获取MJ配置文件的字幕信息
GetMJConfigSrtInformation: async (callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_MJ_CONFIG_SRT_INFORMATION)),
// 获取标签集的基础信息
GetTagDataByTypeAndProperty: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_TAG_DATA_BY_TYPE_AND_PROPERTY, value)),
// 保存数据到标签集中
SaveTagPropertyData: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.SAVE_TAG_PROPERTY_DATA, value)),
// 删除指定的标签数据
DeleteTagPropertyData: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.DELETE_TAG_PROPERTY_DATA, value)),
// 获取选择标签的模式option列表
GetTagSelectModel: async (callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_TAG_SELECT_MODEL)),
// 将翻译任务添加到后台队列中
TranslateReturnNowTask: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.TRANSLATE_RETURN_NOW_TASK, value)),
// MJ原创生图
OriginalMJImageGenerate: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.ORIGINAL_MJ_IMAGE_GENERATE, value)),
// 获取当前对话频道的机器人列表
GetChannelRobots: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_CHANNEL_ROBOTS, value)),
// 获取MJ生图的方式
GetMJGenerateCategory: async (callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_MJ_GENERATE_CATEGORY)),
// MJ生成的图片分割
ImageSplit: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.IMAGE_SPLIT, value)),
// 添加MJ敏感词
AddMjBadPrompt: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.ADD_MJ_BAD_PROMPT, value)),
// 添加MJ敏感词检查
MJBadPromptCheck: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.MJ_BAD_PROMPT_CHECK, value)),
// 获取已经生图完成的数据,并获取图片
GetGeneratedMJImageAndSplit: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_GENERATED_MJ_IMAGE_AND_SPLIT, value)),
// 给图片链接,下载指定的图片并分割保存
DownloadImageUrlAndSplit : async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.DOWNLOAD_IMAGE_URL_AND_SPLIT, value)),
}
const discord = {
// 创建MJ消息
CreateMessage: (value) => ipcRenderer.send(DEFINE_STRING.DISCORD.CREATE_MESSAGE, value),
// MJ消息更新
UpdateMessage: (value) => ipcRenderer.send(DEFINE_STRING.DISCORD.UPDATE_MESSAGE, value),
// MJ消息删除
DeleteMessage: (value) => ipcRenderer.send(DEFINE_STRING.DISCORD.DELETE_MESSAGE, value),
}
// Use `contextBridge` APIs to expose Electron APIs to
// renderer only if context isolation is enabled, otherwise
// just add to the DOM global.
@ -448,6 +406,7 @@ if (process.contextIsolated) {
contextBridge.exposeInMainWorld('api', api)
contextBridge.exposeInMainWorld('mj', mj)
contextBridge.exposeInMainWorld('discord', discord)
contextBridge.exposeInMainWorld("sd", sd)
contextBridge.exposeInMainWorld('darkMode', {
toggle: (value) => ipcRenderer.invoke('dark-mode:toggle', value),
})
@ -459,5 +418,6 @@ if (process.contextIsolated) {
window.api = api;
window.mj = mj;
window.discord = discord;
window.sd = sd;
}

44
src/preload/mj.js Normal file
View File

@ -0,0 +1,44 @@
import { ipcRenderer } from "electron";
import { DEFINE_STRING } from "../define/define_string";
const mj = {
// 保存文案信息到mj的配置文件
SvaeMJWordSrt: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.SAVE_WORD_SRT, value)),
// 获取MJ配置文件的字幕信息
GetMJConfigSrtInformation: async (callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_MJ_CONFIG_SRT_INFORMATION)),
// 获取标签集的基础信息
GetTagDataByTypeAndProperty: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_TAG_DATA_BY_TYPE_AND_PROPERTY, value)),
// 保存数据到标签集中
SaveTagPropertyData: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.SAVE_TAG_PROPERTY_DATA, value)),
// 删除指定的标签数据
DeleteTagPropertyData: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.DELETE_TAG_PROPERTY_DATA, value)),
// 获取选择标签的模式option列表
GetTagSelectModel: async (callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_TAG_SELECT_MODEL)),
// 将翻译任务添加到后台队列中
TranslateReturnNowTask: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.TRANSLATE_RETURN_NOW_TASK, value)),
// MJ原创生图
OriginalMJImageGenerate: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.ORIGINAL_MJ_IMAGE_GENERATE, value)),
// 获取当前对话频道的机器人列表
GetChannelRobots: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_CHANNEL_ROBOTS, value)),
// 获取MJ生图的方式
GetMJGenerateCategory: async (callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_MJ_GENERATE_CATEGORY)),
// MJ生成的图片分割
ImageSplit: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.IMAGE_SPLIT, value)),
// 添加MJ敏感词
AddMjBadPrompt: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.ADD_MJ_BAD_PROMPT, value)),
// 添加MJ敏感词检查
MJBadPromptCheck: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.MJ_BAD_PROMPT_CHECK, value)),
// 获取已经生图完成的数据,并获取图片
GetGeneratedMJImageAndSplit: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.GET_GENERATED_MJ_IMAGE_AND_SPLIT, value)),
// 给图片链接,下载指定的图片并分割保存
DownloadImageUrlAndSplit: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.MJ.DOWNLOAD_IMAGE_URL_AND_SPLIT, value)),
}
export {
mj
}

14
src/preload/sd.js Normal file
View File

@ -0,0 +1,14 @@
import { ipcRenderer } from "electron"
import { DEFINE_STRING } from "../define/define_string"
const sd = {
// 加载当前链接的SD服务数据
LoadSDServiceData: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.SD.LOAD_SD_SERVICE_DATA, value)),
// 文生图,单张
txt2img: async (value, callback) => callback(await ipcRenderer.invoke(DEFINE_STRING.SD.TXT2IMG, value)),
}
export {
sd
}

File diff suppressed because it is too large Load Diff

View File

@ -1,154 +1,281 @@
<template>
<n-layout id="layout_height" has-sider style="height: 300px;">
<n-layout-sider bordered :width="240" :native-scrollbar="false">
<n-menu @update:value="SelectMenu" :options="menuOptions" v-model:value="selectedKeyRef" />
</n-layout-sider>
<n-layout :native-scrollbar="false" style="margin-top: 30px;">
<div style="margin-left: 20px;">
<div>选择的风格</div>
<ShowImageTag :selectStyle="selectStyle"></ShowImageTag>
<div style="margin-top: 5px;">
<div class="image-container" style="display: flex;">
<n-image-group show-toolbar-tooltip>
<div style="display: flex; flex-wrap: wrap;">
<div v-for="(image, index) in style_image_list " style="margin: 0 5px 5px 5px;"
:key="image.id">
<n-image @contextmenu.prevent="handleContextMenu($event, image)" :width="150"
:height="150" :src="image.image" alt="图片描述" lazy style="display: block;" />
</div>
</div>
</n-image-group>
<n-layout id="layout_height" has-sider style="height: 300px">
<n-layout-sider bordered :width="240" :native-scrollbar="false">
<n-menu @update:value="SelectMenu" :options="menuOptions" v-model:value="selectedKeyRef" />
</n-layout-sider>
<n-layout :native-scrollbar="false" style="margin-top: 10px">
<n-tabs
class="card-tabs"
default-value="select"
size="large"
style="margin-left: 20px"
animated
pane-wrapper-style="margin: 0 -4px"
pane-style="padding-left: 4px; padding-right: 4px; box-sizing: border-box;"
>
<n-tab-pane name="select" tab="选择风格">
<div>
<div>选择的风格</div>
<ShowImageTag :selectStyle="selectStyle"></ShowImageTag>
<div style="margin-top: 5px">
<div class="image-container" style="display: flex">
<n-image-group show-toolbar-tooltip>
<div style="display: flex; flex-wrap: wrap">
<div
v-for="(image, index) in style_image_list"
style="margin: 0 5px 5px 5px"
:key="image.id"
>
<n-image
@contextmenu.prevent="handleContextMenu($event, image)"
:width="150"
:height="150"
:src="image.image ? image.image : image.show_image"
alt="图片描述"
lazy
style="display: block"
/>
<n-popover trigger="hover" v-if="image.type == 'style_main'">
<template #trigger>
<span
style="
display: block;
text-align: center;
width: 120px;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
"
>{{ image.prompt }}</span
>
</template>
<span>{{ image.prompt }}</span>
</n-popover>
</div>
</div>
</div>
</n-image-group>
</div>
</div>
</n-layout>
</div>
</n-tab-pane>
<n-tab-pane name="add" tab="添加风格">
<AddStyleTags
:currentStyle="null"
:initFunc="null"
:lora_options="lora_options"
></AddStyleTags>
</n-tab-pane>
</n-tabs>
</n-layout>
</n-layout>
<n-dropdown style="z-index: 100;" trigger="manual" :x="x" :y="y" :options="dropOption" :show="showDropdownRef"
:on-clickoutside="onClickoutside" @select="handleSelect" />
<n-dropdown
style="z-index: 100"
trigger="manual"
:x="x"
:y="y"
:options="dropOption"
:show="showDropdownRef"
:on-clickoutside="onClickoutside"
@select="handleSelect"
/>
</template>
<script>
import { defineComponent, ref, onMounted, watch, nextTick } from "vue";
import { NDropdown, NImage, NLayout, NLayoutContent, NLayoutSider, NMenu, NSpace, NTag, NTooltip, messageDark, useMessage } from "naive-ui"
import ShowImageTag from "./ShowImageTag.vue";
import { defineComponent, ref, onMounted, watch, nextTick } from 'vue'
import {
NDropdown,
NImage,
NLayout,
NLayoutContent,
NLayoutSider,
NMenu,
NSpace,
NTabPane,
NTabs,
NTag,
NTooltip,
messageDark,
useMessage,
NPopover
} from 'naive-ui'
import ShowImageTag from './ShowImageTag.vue'
import AddStyleTags from '../Original/Components/AddStyleTags.vue'
export default defineComponent({
components: {
NSpace, NLayout, NLayoutSider, NLayoutContent, NMenu, NTag, NTooltip, NImage, NDropdown, ShowImageTag
},
props: ["selectStyle", "height"],
setup(props) {
let current_menu = null;
let selectedKeyRef = ref(null);
components: {
NSpace,
NLayout,
NLayoutSider,
NLayoutContent,
NMenu,
NTag,
NTooltip,
NImage,
NDropdown,
ShowImageTag,
NTabs,
NTabPane,
AddStyleTags,
NPopover
},
props: ['selectStyle', 'height', 'tags'],
setup(props) {
let current_menu = null
let selectedKeyRef = ref(null)
let selectStyle = ref(props.selectStyle ? props.selectStyle : []);
let style_image_list = ref([]);
let message = useMessage();
let showDropdownRef = ref(false);
let current_img = ref(null);
let menuOptions = ref([]);
let selectStyle = ref(props.selectStyle ? props.selectStyle : [])
let style_image_list = ref([])
let message = useMessage()
let showDropdownRef = ref(false)
let current_img = ref(null)
let menuOptions = ref([])
let tags = ref(props.tags)
let lora_options = ref([])
const x = ref(0);
const y = ref(0);
const x = ref(0)
const y = ref(0)
let dropOption = [{
label: "选择风格",
key: "selectStyle"
}];
let dropOption = [
{
label: '选择风格',
key: 'selectStyle'
}
]
// props.selectStyle
watch(() => props.selectStyle, (value) => {
selectStyle.value = value;
if (!selectStyle.value) {
selectStyle.value = [];
// props.selectStyle
watch(
() => props.selectStyle,
(value) => {
selectStyle.value = value
if (!selectStyle.value) {
selectStyle.value = []
}
}
)
onMounted(async () => {
await window.api.GetDefineConfigJsonByProperty(
JSON.stringify(['sd_setting', 'lora', false, []]),
(value) => {
if (value.code == 0) {
message.error(value.message)
return
}
//
if (value.data) {
for (let i = 0; i < value.data.length; i++) {
const element = value.data[i]
lora_options.value.push({
label: element.name,
value: element.name
})
}
});
onMounted(async () => {
setLayoutHeight("layout_height");
await window.api.GetImageStyleMenu((value) => {
debugger
if (value.code == 0) {
message.error(value.message);
return;
}
menuOptions.value = value.data;
current_menu = value.data[0].id;
});
await getStyleImageSubList();
selectedKeyRef.value = current_menu;
}
}
)
setLayoutHeight('layout_height')
await window.api.GetImageStyleMenu((value) => {
debugger
if (value.code == 0) {
message.error(value.message)
return
}
menuOptions.value = value.data
menuOptions.value.unshift({
label: '自定义样式',
id: 'tag',
key: 'tag'
})
current_menu = menuOptions.value[0].id
})
await getStyleImageSubList()
selectedKeyRef.value = current_menu
})
/**
* 菜单选中事件
*/
async function SelectMenu(key, item) {
current_menu = key;
await getStyleImageSubList();
}
async function getStyleImageSubList() {
await window.api.getStyleImageSubList(current_menu, (value) => {
debugger;
console.log(value)
if (value.code == 0) {
message.error(value.message);
return;
}
style_image_list.value = value.data;
});
}
// ID
function setLayoutHeight(id) {
let layout_height = document.getElementById(id);
layout_height.style.height = props.height - 40 + "px";
}
async function closeTag(image) {
debugger;
let index = selectStyle.value.findIndex(item => item.id == image.id);
if (index != -1) {
selectStyle.value.splice(index, 1);
}
}
return {
selectStyle,
menuOptions,
SelectMenu,
style_image_list,
selectedKeyRef,
showDropdownRef,
dropOption,
current_img,
x: x,
y: y,
closeTag,
handleContextMenu(e, image) {
e.preventDefault();
current_img.value = image;
nextTick().then(() => {
showDropdownRef.value = true;
x.value = e.clientX;
y.value = e.clientY;
});
},
onClickoutside() {
showDropdownRef.value = false;
},
handleSelect(key) {
debugger
showDropdownRef.value = false;
if (key == "selectStyle") {
//
if (current_img.value) {
selectStyle.value.push(current_img.value);
}
}
}
}
/**
* 菜单选中事件
*/
async function SelectMenu(key, item) {
debugger
current_menu = key
await getStyleImageSubList()
}
async function getStyleImageSubList() {
style_image_list.value = []
if (current_menu == 'tag') {
for (let i = 0; i < tags.value.style_tags.length; i++) {
const element = tags.value.style_tags[i]
tags.value.style_tags[i].image = tags.value.style_tags[i].show_image
tags.value.style_tags[i].id = tags.value.style_tags[i].key
}
//
style_image_list.value = tags.value.style_tags
} else {
await window.api.getStyleImageSubList(current_menu, (value) => {
debugger
console.log(value)
if (value.code == 0) {
message.error(value.message)
return
}
style_image_list.value = value.data
})
}
}
// ID
function setLayoutHeight(id) {
let layout_height = document.getElementById(id)
layout_height.style.height = props.height - 40 + 'px'
}
async function closeTag(image) {
debugger
let index = selectStyle.value.findIndex((item) => item.id == image.id)
if (index != -1) {
selectStyle.value.splice(index, 1)
}
}
return {
selectStyle,
menuOptions,
SelectMenu,
style_image_list,
selectedKeyRef,
showDropdownRef,
dropOption,
current_img,
lora_options,
tags,
x: x,
y: y,
closeTag,
handleContextMenu(e, image) {
e.preventDefault()
current_img.value = image
nextTick().then(() => {
showDropdownRef.value = true
x.value = e.clientX
y.value = e.clientY
})
},
onClickoutside() {
showDropdownRef.value = false
},
handleSelect(key) {
showDropdownRef.value = false
if (key == 'selectStyle') {
debugger
//
if (current_img.value) {
selectStyle.value.push(current_img.value)
}
}
}
}
}
})
</script>

View File

@ -1,57 +1,72 @@
<template>
<div style="margin-top: 10px; min-height: 40px;">
<n-tooltip trigger="hover" v-for="item in selectStyle" placement="bottom-start">
<template #trigger>
<n-tag @close="closeTag(item)" :key="item.id" type="success"
style="margin-right: 10px; margin-bottom: 10px" closable>
{{ item.id.split("-")[0] }}
</n-tag>
</template>
<div>
<n-image :src="item.image" :width="200" :preview-disabled="true"></n-image>
</div>
</n-tooltip>
</div>
<div style="margin-top: 10px; min-height: 40px">
<n-tooltip trigger="hover" v-for="item in selectStyle" placement="bottom-start">
<template #trigger>
<n-tag
@close="closeTag(item)"
:key="item.id"
type="success"
style="margin-right: 10px; margin-bottom: 10px"
closable
>
{{
item.type && item.type == 'style_main'
? '自定义_' + item.key.split('-')[0]
: item.id.split('-')[0]
}}
</n-tag>
</template>
<div>
<n-image
:src="item.image ? item.image : item.show_image"
:width="200"
:preview-disabled="true"
></n-image>
</div>
</n-tooltip>
</div>
</template>
<script>
import { defineComponent, ref, onMounted, watch, nextTick } from "vue";
import { NImage, NTag, NTooltip, useMessage } from "naive-ui"
import { defineComponent, ref, onMounted, watch, nextTick } from 'vue'
import { NImage, NTag, NTooltip, useMessage } from 'naive-ui'
export default defineComponent({
components: {
NTag, NTooltip, NImage
},
props: ["selectStyle"],
setup(props) {
let selectStyle = ref(props.selectStyle ? props.selectStyle : []);
let message = useMessage();
components: {
NTag,
NTooltip,
NImage
},
props: ['selectStyle'],
setup(props) {
let selectStyle = ref(props.selectStyle ? props.selectStyle : [])
let message = useMessage()
// props.selectStyle
watch(() => props.selectStyle, (value) => {
selectStyle.value = value;
if (!selectStyle.value) {
selectStyle.value = [];
}
});
onMounted(async () => {
})
async function closeTag(image) {
debugger;
let index = selectStyle.value.findIndex(item => item.id == image.id);
if (index != -1) {
selectStyle.value.splice(index, 1);
}
// props.selectStyle
watch(
() => props.selectStyle,
(value) => {
selectStyle.value = value
if (!selectStyle.value) {
selectStyle.value = []
}
}
)
return {
selectStyle,
closeTag,
onMounted(async () => {})
}
async function closeTag(image) {
debugger
let index = selectStyle.value.findIndex((item) => item.id == image.id)
if (index != -1) {
selectStyle.value.splice(index, 1)
}
}
return {
selectStyle,
closeTag
}
}
})
</script>

View File

@ -1,173 +1,311 @@
<template>
<div style="width: 500px;">
<n-form ref="formRef" label-placement="top" :model="characterData" :rules="rules">
<n-form-item label="人物名称" path="label">
<n-input v-model:value="characterData.label" placeholder="请输入人物名称" />
</n-form-item>
<n-form-item label="人物别名(可多个)">
<n-dynamic-tags type="success" v-model:value="alias_tags" />
</n-form-item>
<n-form-item label="人物提示词(中文)">
<n-input type="textarea" :rows="2" v-model:value="characterData.chinese_prompt"
placeholder="请输入人物提示词" />
<n-button type="info" size="small" style="margin-left: 20px;" @click="TranslatePrompt"
:loading="loading">翻译提示词</n-button>
</n-form-item>
<n-form-item label="人物提示词(英文)" path="prompt">
<n-input type="textarea" :rows="2" v-model:value="characterData.prompt" placeholder="请输入人物提示词" />
</n-form-item>
<n-form-item label="MJ人物图片链接cref">
<n-input v-model:value="characterData.image_url" placeholder="请输入人物图片链接" style="margin-right: 20px;" />
<n-image v-if="characterData.image_url" :src="characterData.image_url"
style="width: 80px; height: 80px" />
</n-form-item>
<n-form-item label="MJ角色迁移 cw 值0-100">
<n-input-number :show-button="false" v-model:value="characterData.cref_cw" min="0" max="100"
placeholder="请输入人物风格迁移的CW值" />
</n-form-item>
<n-form-item label="SDLora选择">
<n-input v-model:value="characterData.lora" placeholder="请输入人物Lora选择" />
<!-- <n-button color="#e18a3b" style="margin-left: 20px;">选择</n-button> -->
</n-form-item>
</n-form>
<div style="text-align: right;">
<n-button type="success" @click="SaveCharacterTag">保存</n-button>
</div>
<div>
<div style="width: 500px; position: relative">
<n-form ref="formRef" label-placement="top" :model="characterData" :rules="rules">
<n-form-item label="人物名称" path="label">
<n-input v-model:value="characterData.label" placeholder="请输入人物名称" />
<n-image
style="position: absolute; left: 520px; top: 0"
width="120"
height="120"
:src="png_base64 ? png_base64 : characterData.show_image"
/>
</n-form-item>
<n-form-item label="人物别名(可多个)">
<n-dynamic-tags type="success" v-model:value="alias_tags" />
</n-form-item>
<n-form-item label="人物提示词(中文)">
<n-input
type="textarea"
:rows="2"
v-model:value="characterData.chinese_prompt"
placeholder="请输入人物提示词"
/>
<!-- 里面的按钮设置上下居中 -->
<div style="width: 100px; margin-left: 20px">
<n-button color="#a76283" size="tiny" @click="TranslatePrompt" :loading="loading"
>翻译提示词</n-button
>
<n-button
color="#e18a3b"
size="tiny"
style="margin-top: 10px"
@click="GenerateCharacterImage"
:loading="imageLoading"
>生成图片</n-button
>
</div>
</n-form-item>
<n-form-item label="人物提示词(英文)" path="prompt">
<n-input
type="textarea"
:rows="2"
v-model:value="characterData.prompt"
placeholder="请输入人物提示词"
/>
</n-form-item>
<n-form-item label="MJ人物图片链接cref">
<n-input
v-model:value="characterData.image_url"
placeholder="请输入人物图片链接"
style="margin-right: 20px"
/>
<n-image
v-if="characterData.image_url"
:src="characterData.image_url"
style="width: 80px; height: 80px"
:alt="characterData.image_url"
/>
</n-form-item>
<n-form-item label="MJ角色迁移 cw 值0-100">
<n-input-number
:show-button="false"
v-model:value="characterData.cref_cw"
min="0"
max="100"
placeholder="请输入人物风格迁移的CW值"
/>
</n-form-item>
<n-form-item label="SDLora选择">
<n-select
v-model:value="characterData.lora"
style="width: 250px"
:options="lora_options"
placeholder="选择人物lora"
/>
<n-input-number
style="width: 120px; margin-left: 10px"
placeholder="权重"
v-model:value="characterData.lora_weight"
:show-button="false"
:max="2"
:step="0.01"
:precision="2"
S
:min="0"
/>
</n-form-item>
</n-form>
<div style="text-align: right">
<n-button type="success" @click="SaveCharacterTag">保存</n-button>
</div>
</div>
</div>
</template>
<script>
import { ref, h, onMounted, defineComponent, toRaw, watch } from "vue"
import { NImage, useMessage, NButton, useDialog, NInput, NForm, NFormItem, NDynamicTags, NInputNumber } from "naive-ui";
import { DEFINE_STRING } from "../../../../../define/define_string";
import { v4 as uuidv4 } from "uuid";
import { ref, h, onMounted, defineComponent, toRaw, watch, nextTick } from 'vue'
import {
NImage,
useMessage,
NButton,
useDialog,
NInput,
NForm,
NFormItem,
NDynamicTags,
NInputNumber,
NSelect
} from 'naive-ui'
import { v4 as uuidv4 } from 'uuid'
import { isEmpty } from 'lodash'
export default defineComponent({
components: {
NImage, NButton, NInput, NForm, NFormItem, NDynamicTags, NInputNumber
},
props: ["currentCharacter", "initFunc", "currentTags"],
setup(props) {
let message = useMessage();
let alias_tags = ref(props.currentTags);
let characterData = ref(props.currentCharacter);
let loading = ref(false);
let formRef = ref(null)
components: {
NImage,
NButton,
NInput,
NForm,
NFormItem,
NDynamicTags,
NInputNumber,
NSelect
},
props: ['currentCharacter', 'initFunc', 'currentTags', 'lora_options'],
setup(props) {
let message = useMessage()
let alias_tags = ref(props.currentTags)
let characterData = ref(props.currentCharacter)
let lora_options = ref(props.lora_options)
let loading = ref(false)
let imageLoading = ref(false)
let formRef = ref(null)
//
watch(() => props.currentCharacter, (value) => {
characterData.value = value;
}, { deep: true })
let png_base64 = ref(null)
watch(() => props.currentTags, (value) => {
alias_tags.value = value;
}, { deep: true })
//
watch(
() => props.currentCharacter,
(value) => {
characterData.value = value
},
{ deep: true }
)
onMounted(async () => {
watch(
() => props.currentTags,
(value) => {
alias_tags.value = value
},
{ deep: true }
)
onMounted(async () => {})
async function SaveCharacterTag(e) {
e.preventDefault()
formRef.value?.validate(async (errors) => {
if (errors) {
message.error('请检查必填字段')
return
}
//
let children = []
alias_tags.value.forEach((item) => {
children.push({
label: item,
key: uuidv4(),
type: 'min',
chinese_prompt: characterData.value.chinese_prompt,
prompt: characterData.value.prompt,
image_url: characterData.value.image_url,
show_image: characterData.value.show_image,
cref_cw: characterData.value.cref_cw,
lora: characterData.value.lora,
lora_weight: characterData.value.lora_weight
})
})
async function SaveCharacterTag(e) {
e.preventDefault();
formRef.value?.validate(async (errors) => {
if (errors) {
message.error("请检查必填字段");
return
}
//
let children = [];
alias_tags.value.forEach((item) => {
children.push({
label: item,
key: uuidv4(),
type: "min",
chinese_prompt: characterData.value.chinese_prompt,
prompt: characterData.value.prompt,
image_url: characterData.value.image_url,
cref_cw: characterData.value.cref_cw,
lora: characterData.value.lora,
})
})
characterData.value["children"] = children;
debugger;
console.log(characterData.value);
characterData.value['type'] = "character_main";
//
await window.mj.SaveTagPropertyData([JSON.stringify(characterData.value), "character_tags"], (value) => {
debugger;
console.log(value);
if (value.code == 0) {
message.error(value.message);
return;
}
message.success(value.message);
//initFunc
props.initFunc();
//
characterData.value = {
label: null,
key: null,
type: "character_main",
prompt: null,
image_url: null,
cref_cw: 20,
lora: null,
};
alias_tags.value = [];
})
});
}
//
async function TranslatePrompt() {
loading.value = true;
if (characterData.value.chinese_prompt == "" || characterData.value.chinese_prompt == null) {
message.error("请输入中文提示词");
loading.value = false;
return;
characterData.value['children'] = children
console.log(characterData.value)
characterData.value['type'] = 'character_main'
//
await window.mj.SaveTagPropertyData(
[JSON.stringify(characterData.value), 'character_tags'],
(value) => {
debugger
console.log(value)
if (value.code == 0) {
message.error(value.message)
return
}
await window.api.TranslateReturnNow([characterData.value.chinese_prompt, 'zh', 'en', false], (value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
debugger
//
console.log(value.data);
characterData.value.prompt = value.data[0].src;
})
loading.value = false;
}
let ruleObj = (errorMessage) => {
return [{
required: true,
validator(rule, value) {
if (value == null || value == "")
return new Error(errorMessage);
return true;
},
trigger: ["input", "blur", "change"]
}]
}
let rules = {
label: ruleObj("必填人物名称"),
prompt: ruleObj("必填英文提示词")
};
return {
characterData,
alias_tags,
SaveCharacterTag,
TranslatePrompt,
loading,
rules,
formRef
}
message.success(value.message)
//initFunc
props.initFunc()
//
characterData.value = {
label: null,
key: null,
type: 'character_main',
chinese_prompt: null,
prompt: null,
image_url: null,
show_image: window.config.space_image,
cref_cw: 20,
chinese_prompt: null,
lora: '无',
lora_weight: 1
}
png_base64.value = null
alias_tags.value = []
}
)
})
}
//
async function TranslatePrompt() {
loading.value = true
if (characterData.value.chinese_prompt == '' || characterData.value.chinese_prompt == null) {
message.error('请输入中文提示词')
loading.value = false
return
}
await window.api.TranslateReturnNow(
[characterData.value.chinese_prompt, 'zh', 'en', false],
(value) => {
if (value.code == 0) {
message.error(value.message)
return
}
characterData.value.prompt = value.data[0].src
}
)
loading.value = false
}
let ruleObj = (errorMessage) => {
return [
{
required: true,
validator(rule, value) {
if (value == null || value == '') return new Error(errorMessage)
return true
},
trigger: ['input', 'blur', 'change']
}
]
}
let rules = {
label: ruleObj('必填人物名称'),
prompt: ruleObj('必填英文提示词')
}
/**
* 单张生成角色图片预览
*/
async function GenerateCharacterImage() {
// lora
let lora = ''
if (
characterData.value.lora != null &&
characterData.value.lora != '' &&
characterData.value.lora != '无'
) {
lora = `<lora:${characterData.value.lora}:${characterData.value.lora_weight}>`
}
let d = {
prompt: `${characterData.value.prompt}, ${lora}`,
width: 1024,
height: 1024
}
if (isEmpty(characterData.value.prompt)) {
message.error('请填写英文提示词')
return
}
imageLoading.value = true
await window.sd.txt2img(JSON.stringify([d]), async (value) => {
debugger
if (value.code == 0) {
message.error(value.message)
imageLoading.value = false
return
}
if (value.data && value.data.length > 0) {
let d = value.data[0]
characterData.value.show_image = d.image_path.replaceAll('\\', '/')
png_base64.value = 'data:image/png;base64,' + d.base64
console.log(characterData.value.show_image)
imageLoading.value = false
}
// 使 Vue.nextTick Vue
imageLoading.value = false
})
}
return {
characterData,
alias_tags,
SaveCharacterTag,
TranslatePrompt,
loading,
rules,
formRef,
lora_options,
imageLoading,
GenerateCharacterImage,
png_base64
}
}
})
</script>

View File

@ -1,144 +1,281 @@
<template>
<div style="width: 500px;">
<n-form label-placement="top" ref="formRef" :rules="rules" :model="styleData">
<n-form-item label="风格名称" path="label">
<n-input v-model:value="styleData.label" placeholder="请输入风格名称" />
</n-form-item>
<n-form-item label="风格提示词描述(中文)">
<n-input type="textarea" :rows="3" v-model:value="styleData.chinese_prompt"
placeholder="请输入风格提示词描述(中文)" />
<n-button type="info" size="small" style="margin-left: 20px;" @click="TranslatePrompt"
:loading="loading">翻译提示词</n-button>
</n-form-item>
<n-form-item label="风格提示词描述(英文)" path ="prompt">
<n-input type="textarea" :rows="3" v-model:value="styleData.prompt" placeholder="请输入风格提示词描述(英文)" />
</n-form-item>
<n-form-item label="MJ风格垫图链接sref">
<n-input v-model:value="styleData.image_url" placeholder="请输入风格垫图链接" style="margin-right: 20px;" />
<n-image v-if="styleData.image_url" :src="styleData.image_url" style="width: 80px; height: 80px" />
</n-form-item>
<n-form-item label="MJ风格迁移 cw 值0-1000">
<n-input-number :show-button="false" v-model:value="styleData.sref_cw" min="0" max="1000"
placeholder="请输入人物风格迁移的CW值" />
</n-form-item>
<n-form-item label="SDLora选择">
<n-input v-model:value="styleData.lora" placeholder="请输入风格Lora" />
<!-- <n-button color="#e18a3b" style="margin-left: 20px;">选择</n-button> -->
</n-form-item>
</n-form>
<div style="text-align: right;">
<n-button type="success" @click="SaveStyleTag">保存</n-button>
<div style="width: 500px">
<n-form label-placement="top" ref="formRef" :rules="rules" :model="styleData">
<n-form-item label="风格名称" path="label">
<n-input v-model:value="styleData.label" placeholder="请输入风格名称" />
<n-image
style="position: absolute; left: 520px; top: 0"
width="120"
height="120"
:src="png_base64 ? png_base64 : styleData.show_image"
/>
</n-form-item>
<n-form-item label="风格提示词描述(中文)">
<n-input
type="textarea"
:rows="2"
v-model:value="styleData.chinese_prompt"
placeholder="请输入风格提示词描述(中文)"
/>
<div style="width: 100px; margin-left: 20px">
<n-button type="info" size="tiny" @click="TranslatePrompt" :loading="loading"
>翻译提示词</n-button
>
<n-tooltip trigger="hover">
<template #trigger>
<n-button
color="#e18a3b"
size="tiny"
style="margin-top: 10px"
@click="GenerateStyleImage"
:loading="imageLoading"
>生成图片</n-button
>
</template>
使用SD出图模式生成风格图片提示词为1gril加上风格提示词
</n-tooltip>
</div>
</n-form-item>
<n-form-item label="风格提示词描述(英文)" path="prompt">
<n-input
type="textarea"
:rows="3"
v-model:value="styleData.prompt"
placeholder="请输入风格提示词描述(英文)"
/>
</n-form-item>
<n-form-item label="MJ风格垫图链接sref">
<n-input
v-model:value="styleData.image_url"
placeholder="请输入风格垫图链接"
style="margin-right: 20px"
/>
<n-image
v-if="styleData.image_url"
:src="styleData.image_url"
style="width: 80px; height: 80px"
/>
</n-form-item>
<n-form-item label="MJ风格迁移 sw 值0-1000">
<n-input-number
:show-button="false"
v-model:value="styleData.sref_sw"
min="0"
max="1000"
placeholder="请输入人物风格迁移的SW值"
/>
</n-form-item>
<n-form-item label="SDLora选择">
<n-select :options="lora_options" v-model:value="styleData.lora"> </n-select>
<n-input-number
style="width: 120px; margin-left: 10px"
placeholder="权重"
v-model:value="styleData.lora_weight"
:show-button="false"
:max="2"
:step="0.01"
:precision="2"
S
:min="0"
/>
</n-form-item>
</n-form>
<div style="text-align: right">
<n-button type="success" @click="SaveStyleTag">保存</n-button>
</div>
</div>
</template>
<script>
import { ref, h, onMounted, defineComponent, toRaw, watch } from "vue"
import { NImage, useMessage, NButton, useDialog, NInput, NForm, NFormItem, NDynamicTags, NInputNumber } from "naive-ui";
import { DEFINE_STRING } from "../../../../../define/define_string";
import { v4 as uuidv4 } from "uuid";
import { ref, h, onMounted, defineComponent, toRaw, watch } from 'vue'
import {
NImage,
useMessage,
NButton,
useDialog,
NInput,
NForm,
NFormItem,
NDynamicTags,
NInputNumber,
NSelect,
NTooltip
} from 'naive-ui'
import { isEmpty } from 'lodash'
export default defineComponent({
components: {
NImage, NButton, NInput, NForm, NFormItem, NDynamicTags, NInputNumber
},
props: ["currentStyle", "initFunc"],
setup(props) {
let message = useMessage();
let styleData = ref(props.currentStyle);
let loading = ref(false);
let formRef = ref(null)
components: {
NImage,
NButton,
NInput,
NForm,
NFormItem,
NDynamicTags,
NInputNumber,
NSelect,
NTooltip
},
props: ['currentStyle', 'initFunc', 'lora_options'],
setup(props) {
let message = useMessage()
let styleData = ref(
props.currentStyle
? props.currentStyle
: { show_image: window.config.space_image, lora_weight: 1, sref_sw: 50, lora: '无' }
)
let loading = ref(false)
let imageLoading = ref(false)
let formRef = ref(null)
let lora_options = ref(props.lora_options)
let png_base64 = ref(null)
//
watch(() => props.currentStyle, (value) => {
styleData.value = value;
}, { deep: true })
//
watch(
() => props.currentStyle,
(value) => {
styleData.value = value
},
{ deep: true }
)
onMounted(async () => {
onMounted(async () => {})
})
async function SaveStyleTag(e) {
e.preventDefault();
formRef.value?.validate(async (errors) => {
if (errors) {
message.error("请检查必填字段");
return
}
styleData.value["type"] = "style_main";
//
//
await window.mj.SaveTagPropertyData([JSON.stringify(styleData.value), "style_tags"], (value) => {
debugger;
console.log(value);
if (value.code == 0) {
message.error(value.message);
return;
}
message.success(value.message);
//initFunc
props.initFunc();
//
styleData.value = {
label: null,
key: null,
type: "style_main",
prompt: null,
image_url: null,
cref_cw: 20,
lora: null,
};
})
});
async function SaveStyleTag(e) {
e.preventDefault()
formRef.value?.validate(async (errors) => {
if (errors) {
message.error('请检查必填字段')
return
}
/**
* 翻译中文提示词
*/
async function TranslatePrompt(){
loading.value = true;
if (styleData.value.chinese_prompt == "" || styleData.value.chinese_prompt == null) {
message.error("请输入中文提示词");
loading.value = false;
return;
styleData.value['type'] = 'style_main'
//
//
await window.mj.SaveTagPropertyData(
[JSON.stringify(styleData.value), 'style_tags'],
(value) => {
debugger
console.log(value)
if (value.code == 0) {
message.error(value.message)
return
}
await window.api.TranslateReturnNow([styleData.value.chinese_prompt, 'zh', 'en', false], (value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
//
console.log(value.data);
styleData.value.prompt = value.data[0].src;
})
loading.value = false;
}
let ruleObj = (errorMessage) => {
return [{
required: true,
validator(rule, value) {
if (value == null || value == "")
return new Error(errorMessage);
return true;
},
trigger: ["input", "blur", "change"]
}]
}
let rules = {
label: ruleObj("必填人物名称"),
prompt: ruleObj("必填英文提示词")
};
return {
styleData,
SaveStyleTag,
loading,
rules,
formRef,
TranslatePrompt
}
message.success(value.message)
//initFunc
props.initFunc()
//
styleData.value = {
label: null,
key: null,
type: 'style_main',
prompt: null,
show_image: window.config.space_image,
image_url: null,
cref_cw: 20,
lora: '无'
}
png_base64.value = null
}
)
})
}
/**
* 翻译中文提示词
*/
async function TranslatePrompt() {
loading.value = true
if (styleData.value.chinese_prompt == '' || styleData.value.chinese_prompt == null) {
message.error('请输入中文提示词')
loading.value = false
return
}
await window.api.TranslateReturnNow(
[styleData.value.chinese_prompt, 'zh', 'en', false],
(value) => {
if (value.code == 0) {
message.error(value.message)
return
}
console.log(value.data)
styleData.value.prompt = value.data[0].src
}
)
loading.value = false
}
let ruleObj = (errorMessage) => {
return [
{
required: true,
validator(rule, value) {
if (value == null || value == '') return new Error(errorMessage)
return true
},
trigger: ['input', 'blur', 'change']
}
]
}
let rules = {
label: ruleObj('必填人物名称'),
prompt: ruleObj('必填英文提示词')
}
/**
* 生成风格图片
*/
async function GenerateStyleImage() {
if (isEmpty(styleData.value.prompt)) {
message.error('请输入提示词')
imageLoading.value = false
return
}
// lora
let lora = ''
if (
styleData.value.lora != null &&
styleData.value.lora != '' &&
styleData.value.lora != '无'
) {
lora = `<lora:${styleData.value.lora}:${styleData.value.lora_weight}>`
}
let d = {
prompt: `1gril, ${styleData.value.prompt}, ${lora}`,
width: 1024,
height: 1024
}
imageLoading.value = true
await window.sd.txt2img(JSON.stringify([d]), async (value) => {
debugger
if (value.code == 0) {
message.error(value.message)
imageLoading.value = false
return
}
if (value.data && value.data.length > 0) {
let d = value.data[0]
styleData.value.show_image = d.image_path.replaceAll('\\', '/')
png_base64.value = 'data:image/png;base64,' + d.base64
console.log(styleData.value.show_image)
imageLoading.value = false
}
// 使 Vue.nextTick Vue
imageLoading.value = false
})
}
return {
styleData,
SaveStyleTag,
loading,
imageLoading,
rules,
formRef,
TranslatePrompt,
lora_options,
png_base64,
GenerateStyleImage
}
}
})
</script>

View File

@ -1,354 +1,468 @@
<template>
<div>
<n-split direction="horizontal" :max="0.3" :min="0.18" :default-size="0.18">
<template #1>
<div id="tree_define_content" style="display: flex; overflow-y: auto;">
<n-tree :node-props="nodeProps" style="width: 200px;" block-line :data="treeData" />
</div>
<div>
<n-split direction="horizontal" :max="0.3" :min="0.18" :default-size="0.18">
<template #1>
<div id="tree_define_content" style="display: flex; overflow-y: auto">
<n-tree :node-props="nodeProps" style="width: 200px" block-line :data="treeData" />
</div>
</template>
<template #2>
<div style="margin-left: 20px">
<AddCharacterTag
v-if="currentType.startsWith('character')"
:currentCharacter="currentCharacter"
:currentTags="currentTags"
:initFunc="InitData"
:lora_options="lora_options"
/>
<AddStyleTags
v-else-if="currentType.startsWith('style')"
:currentStyle="currentStyle"
:initFunc="InitData"
:lora_options="lora_options"
/>
<AddSceneTags
v-else-if="currentType.startsWith('scene')"
:currentScene="currentScene"
:initFunc="InitData"
/>
<AddPrefixTags
v-else-if="currentType.startsWith('prefix')"
:currentPrefix="currentPrefix"
:initFunc="InitData"
/>
<AddSuffixTags
v-else-if="currentType.startsWith('suffix')"
:currentSuffix="currentSuffix"
:initFunc="InitData"
/>
<n-empty v-else description="是一个飞机">
<template #icon>
<n-icon>
<airplane />
</n-icon>
</template>
<template #2>
<div style="margin-left: 20px;">
<AddCharacterTag v-if="currentType.startsWith('character')" :currentCharacter="currentCharacter"
:currentTags="currentTags" :initFunc="InitData" />
<AddStyleTags v-else-if="currentType.startsWith('style')" :currentStyle="currentStyle"
:initFunc="InitData" />
<AddSceneTags v-else-if="currentType.startsWith('scene')" :currentScene="currentScene"
:initFunc="InitData" />
<AddPrefixTags v-else-if="currentType.startsWith('prefix')" :currentPrefix="currentPrefix"
:initFunc="InitData" />
<AddSuffixTags v-else-if="currentType.startsWith('suffix')" :currentSuffix="currentSuffix"
:initFunc="InitData" />
</div>
</template>
</n-split>
</div>
</n-empty>
</div>
</template>
</n-split>
</div>
</template>
<script>
import { ref, h, onMounted, defineComponent, toRaw } from "vue"
import { NImage, useMessage, NButton, useDialog, NInput, NCard, NTabs, NTabPane, NTree, NSplit, NIcon } from "naive-ui";
import { DEFINE_STRING } from "../../../../../define/define_string";
import { v4 as uuidv4 } from "uuid";
import AddCharacterTag from "./AddCharacterTag.vue";
import AddStyleTags from "./AddStyleTags.vue";
import AddSceneTags from "./AddSceneTags.vue";
import AddPrefixTags from "./AddPrefixTags.vue";
import AddSuffixTags from "./AddSuffixTags.vue";
import { Trash } from "@vicons/ionicons5"
import { ref, h, onMounted, defineComponent, toRaw } from 'vue'
import {
NImage,
useMessage,
NButton,
useDialog,
NInput,
NCard,
NTabs,
NTabPane,
NTree,
NSplit,
NIcon,
NEmpty
} from 'naive-ui'
import { DEFINE_STRING } from '../../../../../define/define_string'
import { v4 as uuidv4 } from 'uuid'
import AddCharacterTag from './AddCharacterTag.vue'
import AddStyleTags from './AddStyleTags.vue'
import AddSceneTags from './AddSceneTags.vue'
import AddPrefixTags from './AddPrefixTags.vue'
import AddSuffixTags from './AddSuffixTags.vue'
import { Trash, Airplane } from '@vicons/ionicons5'
export default defineComponent({
components: {
NImage, NButton, NInput, NCard, NTabs, NTabPane, NTree, NSplit, AddCharacterTag, AddStyleTags, AddSceneTags, AddPrefixTags, AddSuffixTags, NIcon, Trash
},
props: ["height"],
setup(props) {
let message = useMessage();
let currentCharacter = ref({});
let currentStyle = ref({});
let currentScene = ref({});
let currentPrefix = ref({});
let currentSuffix = ref({});
let currentTags = ref([]);
let currentType = ref("character");
let treeData = ref([{
label: '人物',
key: uuidv4(),
children: [],
type: "character",
suffix: () => h(NButton,
components: {
NImage,
NButton,
NInput,
NCard,
NTabs,
NTabPane,
NTree,
NSplit,
AddCharacterTag,
AddStyleTags,
AddSceneTags,
AddPrefixTags,
AddSuffixTags,
NIcon,
Trash,
NEmpty,
Airplane
},
props: ['height'],
setup(props) {
let message = useMessage()
let currentCharacter = ref({})
let currentStyle = ref({})
let currentScene = ref({})
let currentPrefix = ref({})
let currentSuffix = ref({})
let currentTags = ref([])
let currentType = ref('default')
let lora_options = ref([])
let treeData = ref([
{
label: '人物',
key: uuidv4(),
children: [],
type: 'character',
suffix: () =>
h(
NButton,
{
text: true,
color: '#e18a3b',
onClick: (e) => {
AddCharacterTags(e)
}
},
{ default: () => '新加' }
)
},
{
label: '场景',
key: uuidv4(),
children: [],
type: 'scene',
suffix: () =>
h(
NButton,
{
text: true,
color: '#e18a3b',
onClick: (e) => {
AddSceneTags(e)
}
},
{ default: () => '新加' }
)
},
{
label: '风格预设',
key: uuidv4(),
children: [],
type: 'style',
suffix: () =>
h(
NButton,
{
text: true,
color: '#e18a3b',
onClick: (e) => {
AddStyleTags(e)
}
},
{ default: () => '新加' }
)
},
{
label: '前缀',
key: uuidv4(),
children: [],
type: 'prefix',
suffix: () =>
h(
NButton,
{
text: true,
color: '#e18a3b',
onClick: (e) => {
AddPrefixTags(e)
}
},
{ default: () => '新加' }
)
},
{
label: '后缀',
key: uuidv4(),
children: [],
type: 'suffix',
suffix: () =>
h(
NButton,
{
text: true,
color: '#e18a3b',
onClick: (e) => {
AddSuffixTags(e)
}
},
{ default: () => '新加' }
)
}
])
/**
* 初始化标签数据
*/
async function InitData() {
//
await window.mj.GetTagDataByTypeAndProperty(['dynamic', null], (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
//
if (value.data.hasOwnProperty('character_tags')) {
treeData.value[0].children = value.data.character_tags
}
if (value.data.hasOwnProperty('scene_tags')) {
treeData.value[1].children = value.data.scene_tags
}
if (value.data.hasOwnProperty('style_tags')) {
treeData.value[2].children = value.data.style_tags
}
if (value.data.hasOwnProperty('prefix_tags')) {
treeData.value[3].children = value.data.prefix_tags
}
if (value.data.hasOwnProperty('suffix_tags')) {
treeData.value[4].children = value.data.suffix_tags
}
for (let i = 0; i < treeData.value.length; i++) {
treeData.value[i].children.map((item) => {
item.suffix = () =>
h(
NButton,
{
text: true, color: "#e18a3b",
onClick: (e) => { AddCharacterTags(e) }
text: true,
type: 'error',
size: 'medium',
onClick: (e) => {
DeleteTags(e, item)
}
},
{ default: () => "新加" })
},
{
label: '场景',
key: uuidv4(),
children: [],
type: "scene",
suffix: () => h(NButton,
{
text: true, color: "#e18a3b",
onClick: (e) => { AddSceneTags(e) }
},
{ default: () => "新加" })
},
{
label: '风格迁移预设',
key: uuidv4(),
children: [],
type: "style",
suffix: () => h(NButton,
{
text: true, color: "#e18a3b",
onClick: (e) => { AddStyleTags(e) }
},
{ default: () => "新加" })
},
{
label: '前缀',
key: uuidv4(),
children: [],
type: "prefix",
suffix: () => h(NButton,
{
text: true, color: "#e18a3b",
onClick: (e) => { AddPrefixTags(e) }
},
{ default: () => "新加" })
},
{
label: '后缀',
key: uuidv4(),
children: [],
type: "suffix",
suffix: () => h(NButton,
{
text: true, color: "#e18a3b",
onClick: (e) => { AddSuffixTags(e) }
},
{ default: () => "新加" })
{ default: () => h(NIcon, null, { default: () => h(Trash) }) }
)
})
}
]);
/**
* 初始化标签数据
*/
async function InitData() {
//
await window.mj.GetTagDataByTypeAndProperty(["dynamic", null], (value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
//
if (value.data.hasOwnProperty("character_tags")) {
treeData.value[0].children = value.data.character_tags;
}
if (value.data.hasOwnProperty("scene_tags")) {
treeData.value[1].children = value.data.scene_tags;
}
if (value.data.hasOwnProperty("style_tags")) {
treeData.value[2].children = value.data.style_tags;
}
if (value.data.hasOwnProperty("prefix_tags")) {
treeData.value[3].children = value.data.prefix_tags;
}
if (value.data.hasOwnProperty("suffix_tags")) {
treeData.value[4].children = value.data.suffix_tags;
}
for (let i = 0; i < treeData.value.length; i++) {
treeData.value[i].children.map(item => {
item.suffix = () => h(NButton,
{
text: true, type: "error", size: "medium",
onClick: (e) => { DeleteTags(e, item) }
},
{ default: () => h(NIcon, null, { default: () => h(Trash) }) });
});
}
})
}
onMounted(async () => {
SetAutoHeight();
await InitData();
})
//
function SetAutoHeight() {
debugger
let div = document.getElementById("tree_define_content");
div.style.height = props.height - 180 + "px";
}
/**
* 删除指定的数据
*/
async function DeleteTags(e, value) {
if (e) {
e.stopPropagation();
}
let type = value.type.replace("main", "tags");
await window.mj.DeleteTagPropertyData([value.key, type], async (value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
message.success(value.message);
await InitData();
})
}
/**
* 新加人物标签
*/
async function AddCharacterTags(e) {
if (e) {
e.stopPropagation();
}
//
currentCharacter.value = {
label: null,
key: uuidv4(),
children: [],
type: "character_main",
prompt: null,
image_url: null,
cref_cw: 50,
lora: null,
}
currentTags.value = [];
currentType.value = "character_main";
}
/**
* 新加场景标签
*/
function AddSceneTags(e) {
if (e) {
e.stopPropagation();
}
currentScene.value = {
label: null,
key: uuidv4(),
type: "scene_main",
prompt: null,
}
currentType.value = "scene_main";
}
/**
* 新加 风格迁移预设
* @param {*} e
*/
function AddStyleTags(e) {
if (e) {
e.stopPropagation();
}
currentStyle.value = {
label: null,
key: uuidv4(),
type: "style_main",
prompt: null,
image_url: null,
sref_cw: 50,
}
}
/**
* 添加前缀
*/
function AddPrefixTags(e) {
if (e) {
e.stopPropagation();
}
currentPrefix.value = {
label: null,
key: uuidv4(),
type: "prefix_main",
prompt: null,
}
}
/**
* 添加后缀
* @param {*} e
*/
function AddSuffixTags(e) {
if (e) {
e.stopPropagation();
}
currentSuffix.value = {
label: null,
key: uuidv4(),
type: "suffix_main",
prompt: null,
}
}
function nodeProps({ option }) {
return {
onClick() {
debugger;
//
if (option.type == "style") {
currentType.value = "style";
AddStyleTags(null);
return;
} else if (option.type == "scene") {
AddSceneTags(null);
currentType.value = "scene";
// message.error("");
return;
} else if (option.type == "character") {
AddCharacterTags(null);
currentType.value = "character";
return;
} else if (option.type == "prefix") {
AddPrefixTags(null);
currentType.value = "prefix";
return;
} else if (option.type == "suffix") {
AddSuffixTags(null);
currentType.value = "suffix";
return;
}
else if (option.type == "min") {
message.error("别名节点不允许操作");
} else {
debugger
console.log(option);
currentType.value = option.type;
//
if (currentType.value.startsWith("character")) {
// tags
currentCharacter.value = option;
currentTags.value = option.children.map((item) => {
return item.label;
})
} else if (currentType.value.startsWith("scene")) {
currentScene.value = option;
}
else if (currentType.value.startsWith("style")) {
currentStyle.value = option;
} else if (currentType.value.startsWith("prefix")) {
currentPrefix.value = option;
} else if (currentType.value.startsWith("suffix")) {
currentSuffix.value = option;
}
else {
message.error("未知的类型");
}
}
}
}
}
return {
treeData,
AddCharacterTags,
AddSceneTags,
AddStyleTags,
nodeProps,
currentCharacter,
currentTags,
InitData,
currentType,
currentStyle,
currentScene,
currentPrefix,
currentSuffix
})
// lora
// mj
await window.api.GetDefineConfigJsonByProperty(
JSON.stringify(['sd_setting', 'lora', false, []]),
(value) => {
if (value.code == 0) {
message.error(value.message)
return
}
//
if (value.data) {
for (let i = 0; i < value.data.length; i++) {
const element = value.data[i]
lora_options.value.push({
label: element.name,
value: element.name
})
}
}
}
)
}
onMounted(async () => {
SetAutoHeight()
await InitData()
})
//
function SetAutoHeight() {
debugger
let div = document.getElementById('tree_define_content')
div.style.height = props.height - 180 + 'px'
}
/**
* 删除指定的数据
*/
async function DeleteTags(e, value) {
if (e) {
e.stopPropagation()
}
let type = value.type.replace('main', 'tags')
await window.mj.DeleteTagPropertyData([value.key, type], async (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
message.success(value.message)
await InitData()
})
}
/**
* 新加人物标签
*/
async function AddCharacterTags(e) {
if (e) {
e.stopPropagation()
}
//
currentCharacter.value = {
label: null,
key: uuidv4(),
children: [],
type: 'character_main',
chinese_prompt: null,
prompt: null,
image_url: null,
show_image: window.config.space_image,
cref_cw: 50,
lora: '无',
lora_weight: 1
}
currentTags.value = []
currentType.value = 'character_main'
}
/**
* 新加场景标签
*/
function AddSceneTags(e) {
if (e) {
e.stopPropagation()
}
currentScene.value = {
label: null,
key: uuidv4(),
type: 'scene_main',
prompt: null
}
currentType.value = 'scene_main'
}
/**
* 新加 风格迁移预设
* @param {*} e
*/
function AddStyleTags(e) {
if (e) {
e.stopPropagation()
}
currentStyle.value = {
label: null,
key: uuidv4(),
type: 'style_main',
show_image: window.config.space_image,
prompt: null,
image_url: null,
sref_sw: 50,
lora: '无',
lora_weight: 1
}
}
/**
* 添加前缀
*/
function AddPrefixTags(e) {
if (e) {
e.stopPropagation()
}
currentPrefix.value = {
label: null,
key: uuidv4(),
type: 'prefix_main',
prompt: null
}
}
/**
* 添加后缀
* @param {*} e
*/
function AddSuffixTags(e) {
if (e) {
e.stopPropagation()
}
currentSuffix.value = {
label: null,
key: uuidv4(),
type: 'suffix_main',
prompt: null
}
}
function nodeProps({ option }) {
return {
onClick() {
debugger
//
if (option.type == 'style') {
currentType.value = 'style'
AddStyleTags(null)
return
} else if (option.type == 'scene') {
AddSceneTags(null)
currentType.value = 'scene'
// message.error("");
return
} else if (option.type == 'character') {
AddCharacterTags(null)
currentType.value = 'character'
return
} else if (option.type == 'prefix') {
AddPrefixTags(null)
currentType.value = 'prefix'
return
} else if (option.type == 'suffix') {
AddSuffixTags(null)
currentType.value = 'suffix'
return
} else if (option.type == 'min') {
message.error('别名节点不允许操作')
} else {
debugger
console.log(option)
currentType.value = option.type
//
if (currentType.value.startsWith('character')) {
// tags
currentCharacter.value = option
currentTags.value = option.children.map((item) => {
return item.label
})
} else if (currentType.value.startsWith('scene')) {
currentScene.value = option
} else if (currentType.value.startsWith('style')) {
currentStyle.value = option
} else if (currentType.value.startsWith('prefix')) {
currentPrefix.value = option
} else if (currentType.value.startsWith('suffix')) {
currentSuffix.value = option
} else {
message.error('未知的类型')
}
}
}
}
}
return {
treeData,
AddCharacterTags,
AddSceneTags,
AddStyleTags,
nodeProps,
currentCharacter,
currentTags,
InitData,
currentType,
currentStyle,
currentScene,
currentPrefix,
currentSuffix,
lora_options
}
}
})
</script>

View File

@ -1,6 +1,6 @@
<template>
<div v-if="type == 'title'" style="display: flex">
<span style="margin-right: 5px">参数</span>
<span style="margin-right: 5px">人物/场景</span>
<n-select
style="width: 75px"
size="tiny"
@ -56,6 +56,7 @@
filterable
:options="character_tags"
placeholder="选择人物"
@update:value="UpdateCharacterSelect"
/>
</div>
<div style="margin-top: 3px">
@ -69,6 +70,7 @@
filterable
:options="scene_tags"
placeholder="选择场景"
@update:value="UpdateSceneSelect"
/>
</div>
</div>
@ -236,8 +238,12 @@ export default defineComponent({
character_tags.value[index].checked = true
// select_character_tags
select_character_tags.value.push(character_tags.value[index].key)
} else {
//
row.value.character_tags.splice(i, 1)
}
}
for (let i = 0; row.value.scene_tags && i < row.value.scene_tags.length; i++) {
const element = row.value.scene_tags[i]
let index = scene_tags.value.findIndex((tag) => tag.key == element.key)
@ -245,6 +251,9 @@ export default defineComponent({
scene_tags.value[index].checked = true
// select_scene_tags
select_scene_tags.value.push(scene_tags.value[index].key)
} else {
//
row.value.scene_tags.splice(i, 1)
}
}
}
@ -265,6 +274,36 @@ export default defineComponent({
props.func.refreshTagData()
}
/**
* 更新人物选择修改对应的tag
*/
function UpdateCharacterSelect(value, option) {
console.log('人物更新下拉框', value, option)
// row
row.value.character_tags = []
// optioncheckedtruerow
for (let i = 0; i < option.length; i++) {
const element = option[i]
element.checked = true
row.value.character_tags.push(element)
}
}
/**
* 更新场景时显示的数据
*/
function UpdateSceneSelect(value, option) {
console.log('场景更新下拉框', value, option)
// row
row.value.scene_tags = []
// optioncheckedtruerow
for (let i = 0; i < option.length; i++) {
const element = option[i]
element.checked = true
row.value.scene_tags.push(element)
}
}
return {
characterData,
row,
@ -280,7 +319,9 @@ export default defineComponent({
InitCharacterAndSceneData,
UpdateImageGenerateCategory,
title_character_select_model,
RefreshTagData
RefreshTagData,
UpdateCharacterSelect,
UpdateSceneSelect
}
}
})

View File

@ -4,18 +4,18 @@
<n-popover trigger="hover">
<template #trigger>
<n-button color="#7c461e" style="margin-left: 5px" size="tiny" @click="AddPrefix"
>添加前缀</n-button
>通用前缀</n-button
>
</template>
<span>添加前缀只能添加英文并且通用前缀不为空推理会自动添加前缀</span>
<span>添加通用前缀只能添加英文并且通用前缀不为空推理会自动添加前缀</span>
</n-popover>
<n-popover trigger="hover">
<template #trigger>
<n-button color="#e18a3b" style="margin-left: 5px" size="tiny" @click="AddSuffix"
>添加后缀</n-button
>通用后缀</n-button
>
</template>
<span>添加后缀只能添加英文并且通用后缀不为空推理会自动添加后缀</span>
<span>添加通用后缀只能添加英文并且通用后缀不为空推理会自动添加后缀</span>
</n-popover>
<n-popover trigger="hover">
<template #trigger>
@ -77,10 +77,6 @@
<script>
import { ref, h, onMounted, defineComponent, toRaw, watch } from 'vue'
import { NButton, NDropdown, NInput, NPopover, useMessage } from 'naive-ui'
import { DEFINE_STRING } from '../../../../../define/define_string'
import { v4 as uuidv4 } from 'uuid'
import { cloneDeep } from 'lodash'
export default defineComponent({
components: {
NButton,
@ -192,7 +188,6 @@ export default defineComponent({
*/
async function SinglePrompt() {
await props.func.singlePrompt(row.value)
}
/**

View File

@ -1,37 +1,23 @@
<template>
<div v-if="type == 'title'">
<span style="margin-right: 5px">提示词命令</span>
<n-button size="tiny" @click="MergePrompt" color="#7c461e">合并命令</n-button>
<n-button
color="#e18a3b"
size="tiny"
style="margin-left: 5px"
@click="MJBadPromptCheck"
<n-dropdown trigger="hover" :options="MergePromptOptions" @select="MergePromptSelect">
<n-button size="tiny" @click="MergePrompt" color="#7c461e">合并命令</n-button>
</n-dropdown>
<n-button color="#e18a3b" size="tiny" style="margin-left: 5px" @click="MJBadPromptCheck"
>敏感词检查</n-button
>
</div>
<div v-else-if="type == 'data'">
<n-button
size="tiny"
color="#dd7694"
style="margin-right: 5px"
@click="SingleGenerateImage"
<n-button size="tiny" color="#dd7694" style="margin-right: 5px" @click="SingleGenerateImage"
>单句生图</n-button
>
<n-button
size="tiny"
color="#a76283"
style="margin-right: 5px"
@click="NextGenerateImage"
<n-button size="tiny" color="#a76283" style="margin-right: 5px" @click="NextGenerateImage"
>下生图</n-button
>
<n-popover trigger="hover">
<template #trigger>
<n-button
color="#a46244"
style="margin-right: 5px"
size="tiny"
@click="ImportMJImageUrl"
<n-button color="#a46244" style="margin-right: 5px" size="tiny" @click="ImportMJImageUrl"
>导入图片</n-button
>
</template>
@ -40,7 +26,6 @@
<n-input
:status="input_status"
readonly
size="tiny"
placeholder="请生成出图命令或是提示词"
type="textarea"
@ -51,25 +36,26 @@
</template>
<script>
import { ref, h, onMounted, defineComponent, toRaw, watch } from "vue";
import { NButton, NInput, NPopover, useMessage, useDialog } from "naive-ui";
import { DEFINE_STRING } from "../../../../../define/define_string";
import InputDialogContent from "./InputDialogContent.vue";
import { ref, h, onMounted, defineComponent, toRaw, watch } from 'vue'
import { NButton, NInput, NPopover, useMessage, useDialog, NDropdown } from 'naive-ui'
import { DEFINE_STRING } from '../../../../../define/define_string'
import InputDialogContent from './InputDialogContent.vue'
export default defineComponent({
components: {
NButton,
NInput,
NPopover,
NDropdown
},
props: ["type", "row", "index", "func"],
props: ['type', 'row', 'index', 'func'],
setup(props) {
let message = useMessage();
let dialog = useDialog();
let type = ref(props.type);
let row = ref(props.row);
let input_status = ref("default");
let image_url_ref = ref(null);
let message = useMessage()
let dialog = useDialog()
let type = ref(props.type)
let row = ref(props.row)
let input_status = ref('default')
let image_url_ref = ref(null)
// let input_image_
@ -79,63 +65,62 @@ export default defineComponent({
watch(
() => props.row,
(newVal) => {
row.value = newVal;
row.value = newVal
},
{ deep: true }
);
)
watch(
() => props.row.mj_message,
(newVal) => {
if (newVal && newVal["hasBadPrompt"]) {
input_status.value = "error";
if (newVal && newVal['hasBadPrompt']) {
input_status.value = 'error'
}
},
{ deep: true }
);
)
onMounted(async () => {
if (row.value["mj_message"]?.hasBadPrompt) {
input_status.value = "error";
if (row.value['mj_message']?.hasBadPrompt) {
input_status.value = 'error'
}
});
})
/**
* SD单句生图
* @param {*} row 要生图的行数据
*/
async function SingleGenerateImage() {
debugger;
let ca = window.config.image_generate_category;
let ca = window.config.image_generate_category ? window.config.image_generate_category : 'sd'
//
if (ca == "sd") {
if (ca == 'sd') {
// SD
await window.api.OriginalSDImageGenerate(
[JSON.stringify([toRaw(row.value)]), true, false],
(value) => {
if (value.code == 0) {
message.error(value.message);
return;
message.error(value.message)
return
}
message.success("添加SD生图任务成功");
message.success('添加SD生图任务成功')
}
);
} else if (ca == "mj") {
)
} else if (ca == 'mj') {
// MJ
await window.mj.OriginalMJImageGenerate(
[JSON.stringify([toRaw(row.value)]), true, false],
(value) => {
if (value.code == 0) {
message.error(value.message);
return;
message.error(value.message)
return
}
message.success("添加MJ生图任务成功");
message.success('添加MJ生图任务成功')
}
);
} else if (ca == "d3") {
)
} else if (ca == 'd3') {
// D3
} else {
message.error("生图的类别错误");
message.error('生图的类别错误')
}
}
@ -143,21 +128,22 @@ export default defineComponent({
* 下生图
*/
async function NextGenerateImage() {
props.func.nextGenerateImage(row.value, window.config.image_generate_category);
let ca = window.config.image_generate_category ? window.config.image_generate_category : 'sd'
props.func.nextGenerateImage(row.value, ca)
}
/**
* MJ敏感词检测
*/
async function MJBadPromptCheck() {
props.func.mJBadPromptCheck();
props.func.mJBadPromptCheck()
}
/**
* 合并提示词
*/
async function MergePrompt() {
props.func.mergePrompt();
props.func.mergePrompt(null)
}
/**
@ -166,43 +152,51 @@ export default defineComponent({
async function ImportMJImageUrl() {
//
//
let dialogWidth = 400;
let dialogHeight = 150;
let dialogWidth = 400
let dialogHeight = 150
dialog.create({
title: "添加图片链接",
title: '添加图片链接',
showIcon: false,
closeOnEsc: false,
content: () =>
h(InputDialogContent, {
ref: image_url_ref,
initData: null,
placeholder: "请输入图片链接",
placeholder: '请输入图片链接'
}),
style: `width : ${dialogWidth}px; min-height : ${dialogHeight}px`,
maskClosable: false,
onClose: async () => {
debugger;
let row_image_url = image_url_ref.value.data;
debugger
let row_image_url = image_url_ref.value.data
//
await window.mj.DownloadImageUrlAndSplit(
JSON.stringify([toRaw(row.value), row_image_url]),
(value) => {
if (value.code == 0) {
message.error(value.message);
return;
message.error(value.message)
return
}
//
row.value.outImagePath = value.data.outImagePath;
row.value.subImagePath = value.data.subImagePath;
row.value.mj_message = value.data.mj_message ? value.data.mj_message : {};
row.value.mj_message.image_click = value.data.image_click;
row.value.mj_message.image_path = value.data.image_path;
row.value.mj_message.progress = 100;
row.value.outImagePath = value.data.outImagePath
row.value.subImagePath = value.data.subImagePath
row.value.mj_message = value.data.mj_message ? value.data.mj_message : {}
row.value.mj_message.image_click = value.data.image_click
row.value.mj_message.image_path = value.data.image_path
row.value.mj_message.progress = 100
}
);
},
});
)
}
})
}
/**
* 点击合并下拉框实现功能的方式
* @param key
*/
async function MergePromptSelect(key) {
props.func.mergePrompt(key)
}
return {
@ -215,7 +209,12 @@ export default defineComponent({
MergePrompt,
ImportMJImageUrl,
image_url_ref,
};
},
});
MergePromptSelect,
MergePromptOptions: [
{ label: 'SD模式合并', key: 'sd_merge' },
{ label: 'MJ模式合并', key: 'mj_merge' }
]
}
}
})
</script>

View File

@ -32,8 +32,7 @@
v-if="outImagePath"
:src="outImagePath"
fit="cover"
width="120px"
height="120px"
style="width: 120px; height: 120px"
/>
</div>
<div>

View File

@ -1,11 +1,71 @@
<template>
<div style="margin-top: 30px">
<div style="display: flex; align-items: center">
<n-button color="#b6a014" size="small" @click="formateWrite">一键格式化</n-button>
<n-popover trigger="hover">
<template #trigger>
<n-button quaternary circle size="tiny" color="#b6a014" @click="AddSplitChar">
<template #icon>
<n-icon size="25"> <AddCircleOutline /> </n-icon>
</template>
</n-button>
</template>
<span>添加分割标识符</span>
</n-popover>
<n-popover trigger="hover">
<template #trigger>
<n-input-number
style="margin-left: 10px; width: 130px"
size="small"
placeholder="输入合并的行数"
:max="10000"
:min="1"
@update:value="mergeCountChange"
:show-button="false"
v-model:value="write_setting.merge_count"
></n-input-number>
</template>
<span>设置将多少行进行合并</span>
</n-popover>
<n-popover trigger="hover">
<template #trigger>
<n-input
style="margin-left: 10px; width: 80px"
size="small"
placeholder="输入中间拼接的字符"
@update:value="mergeCountChange"
:show-button="false"
v-model:value="write_setting.merge_char"
></n-input>
</template>
<span>设置中间拼接的字符</span>
</n-popover>
<n-popover trigger="hover">
<template #trigger>
<n-input
style="margin-left: 10px; width: 80px"
size="small"
placeholder="输入结尾拼接的字符"
@update:value="mergeCountChange"
:show-button="false"
v-model:value="write_setting.end_char"
></n-input>
</template>
<span>设置结尾拼接的字符</span>
</n-popover>
<n-button color="#80a492" size="small" style="margin-left: 10px" @click="mergeWord"
>合并</n-button
>
</div>
<n-input
style="margin-top: 5px"
v-model:value="word"
:autosize="{ minRows: 30, maxRows: 30 }"
type="textarea"
placeholder="输入分镜文案"
@input="ChangeWordInput"
@update:value="ChangeWordInput"
/>
<div style="margin-top: 5px; display: flex">
<div style="flex: 1"> {{ rowCount }} 行分镜当前数量不是最后数量最后会删除空行</div>
@ -15,26 +75,49 @@
<script>
import { ref, h, onMounted, defineComponent, toRaw } from 'vue'
import { NImage, useMessage, NButton, useDialog, NInput } from 'naive-ui'
import {
NImage,
useMessage,
NButton,
useDialog,
NInput,
NIcon,
NPopover,
NInputNumber
} from 'naive-ui'
import { DEFINE_STRING } from '../../../../../define/define_string'
import { AddCircleOutline } from '@vicons/ionicons5'
import InputDialogContent from './InputDialogContent.vue'
import { max } from 'lodash'
export default defineComponent({
components: {
NImage,
NButton,
NInput
NInput,
NInputNumber,
NIcon,
AddCircleOutline,
NPopover
},
props: ['initData'],
setup(props) {
let message = useMessage()
let dialog = useDialog()
let rowCount = ref(0)
let data = ref(props.initData)
let word = ref(null)
let word_arr = ref([])
let split_ref = ref(null)
let write_setting = ref({
split_char: '。,“”‘’!?【】《》()…—:;.,\'\'""!?[]<>()...-:;',
merge_count: 3,
merge_char: '',
end_char: '。'
})
onMounted(async () => {
debugger
console.log(data)
// word
let tmp_arr = []
for (let i = 0; i < data.value.length; i++) {
@ -43,6 +126,20 @@ export default defineComponent({
}
word.value = tmp_arr.join('\n')
ChangeWordInput()
//
await window.api.GetDefineConfigJsonByProperty(
JSON.stringify(['clip_setting', 'write_setting', false, null]),
(value) => {
if (value.code == 0) {
message.error(value.message)
return
}
if (value.data) {
write_setting.value = value.data
}
}
)
})
/**
@ -58,12 +155,137 @@ export default defineComponent({
}
}
/**
* 添加分割符号
*/
async function AddSplitChar() {
//
//
let dialogWidth = 400
let dialogHeight = 150
dialog.create({
title: '添加分割符',
showIcon: false,
closeOnEsc: false,
content: () =>
h(InputDialogContent, {
ref: split_ref,
initData: write_setting.value.split_char,
placeholder: '请输入分割符'
}),
style: `width : ${dialogWidth}px; min-height : ${dialogHeight}px`,
maskClosable: false,
onClose: async () => {
debugger
write_setting.value.split_char = split_ref.value.data
//
await window.api.SaveDefineConfigJsonByProperty(
JSON.stringify(['clip_setting', 'write_setting', toRaw(write_setting.value), false]),
(value) => {
debugger
console.log(value)
if (value.code == 0) {
message.error(value.message)
return
}
message.success(value.message)
}
)
}
})
}
/**
* 更新input-number事件
*/
let timer = null
async function mergeCountChange(value) {
//
clearTimeout(timer)
timer = setTimeout(async () => {
//
//
await window.api.SaveDefineConfigJsonByProperty(
JSON.stringify(['clip_setting', 'write_setting', toRaw(write_setting.value), false]),
(value) => {
debugger
console.log(value)
if (value.code == 0) {
message.error(value.message)
return
}
message.success(value.message)
}
)
}, 500)
}
/**
* 一键格式化
*/
async function formateWrite() {
debugger
let split_arr = Array.from(write_setting.value.split_char)
split_arr.forEach((item) => {
let specialCharacters = [
'.',
'*',
'?',
'+',
'^',
'$',
'[',
']',
'(',
')',
'{',
'}',
'|',
'\\'
]
let regex
if (specialCharacters.includes(item)) {
regex = new RegExp('\\' + item, 'g')
} else {
regex = new RegExp(item, 'g')
}
word.value = word.value.replace(regex, '\n')
})
//
let word_arr = word.value.split('\n')
word_arr = word_arr.filter((item) => item != '' && item != null)
word.value = word_arr.join('\n')
ChangeWordInput()
}
/**
* 合并数据
*/
async function mergeWord() {
let step = write_setting.value.merge_count ? write_setting.value.merge_count : 3
let tmp_arr = []
let word_arr = word.value.split('\n')
for (let i = 0; i < word_arr.length; i += step) {
let group = word_arr.slice(i, i + step)
tmp_arr.push(group.join(write_setting.value.merge_char) + write_setting.value.end_char)
}
word.value = tmp_arr.join('\n')
ChangeWordInput()
}
return {
data,
rowCount,
ChangeWordInput,
word,
word_arr
word_arr,
AddSplitChar,
split_ref,
write_setting,
mergeCountChange,
formateWrite,
mergeWord
}
}
})

View File

@ -50,7 +50,7 @@ export default defineComponent({
props: ['initData', 'menuType', 'height'],
setup(props) {
let hh = props.height
let maxHeight = props.height - 80
let maxHeight = props.height - 80
let data = ref(props.initData)
const message = useMessage()
let selectKey = ref([])
@ -506,11 +506,8 @@ export default defineComponent({
positiveText: '保存',
negativeText: '取消',
onPositiveClick: async () => {
debugger
console.log(editWordRef)
debugger
data.value = []
let word_arr = editWordRef.value.word.split('\n')
let new_arr = []
let lastId = ''
for (let i = 0; i < word_arr.length; i++) {
const element = word_arr[i]

View File

@ -1,12 +1,14 @@
<template>
<div>
<n-data-table
id="1111"
:max-height="maxHeight"
:row-key="rowKey"
:columns="columns"
:data="data"
:pagination="false"
:bordered="false"
:scroll-x="0"
@update:checked-row-keys="handleCheck"
/>
</div>
@ -41,13 +43,18 @@ import { DEFINE_STRING } from '../../../../define/define_string'
import ModifyPromptChinese from '../Components/ModifyPromptChinese.vue'
import InputDialogContent from './Components/InputDialogContent.vue'
import DataTableShowGenerateImage from './Components/DataTableShowGenerateImage.vue'
import { cloneDeep, debounce, min } from 'lodash'
import { cloneDeep, debounce, isEmpty, min } from 'lodash'
import SelectImageStyle from '../Components/SelectImageStyle.vue'
import DataTableCharacterAndScene from './Components/DataTableCharacterAndScene.vue'
import DataTablePromptRow from './Components/DataTablePromptRow.vue'
import DataTableGptPromptRow from './Components/DataTableGptPromptRow.vue'
import DataTableParameterRow from './Components/DataTableParameterRow.vue'
import { Reload } from '@vicons/ionicons5'
import {
checkStringValueAddPrefix,
checkStringValueAddSuffix,
checkStringValueDeletePrefix
} from '../../../../main/generalTools'
export default defineComponent({
components: {
@ -63,15 +70,15 @@ export default defineComponent({
Reload,
NIcon
},
props: ['initData', 'prefix_prompt', 'suffix_prompt'],
props: ['initData', 'prefix_prompt', 'suffix_prompt', 'tags', 'InitTags'],
setup(props) {
console.log('输出文件夹位置', props.initData)
let data = ref(props.initData)
let tags = ref(props.tags)
let message = useMessage()
let dialog = useDialog()
let selectKey = ref([])
let promptChineseRef = ref(null)
let zhAutoTranslate = ref(false)
let maxHeight = ref(null)
let prefix_prompt = ref(props.prefix_prompt)
let suffix_prompt = ref(props.suffix_prompt)
@ -86,8 +93,6 @@ export default defineComponent({
window.config.character_select_model ? window.config.character_select_model : 'tag'
)
let tags = ref({})
let AutoSaveDataJsonDebounced = debounce(AutoSaveDataJson, 3000)
let SaveDataJsonDebounced = debounce(AutoSaveDataJson, 1000)
@ -107,6 +112,23 @@ export default defineComponent({
{ deep: true }
)
watch(
() => props.tags,
async (newVal) => {
debugger
tags.value = newVal
// selectStyle.value
for (let i = 0; i < selectStyle.value.length; i++) {
let index = tags.value.style_tags?.findIndex(
(item) => item.key == selectStyle.value[i].key
)
if (selectStyle.value[i].type == 'style_main' && index != null && index != -1) {
selectStyle.value[i] = tags.value.style_tags[index]
}
}
}
)
watch(
() => props.prefix_prompt,
(newVal) => {
@ -138,20 +160,16 @@ export default defineComponent({
const createColumns = ({}) => {
return [
{
type: 'selection',
disabled(row) {
return row.name === 'Edward King 3'
}
},
{
title: '',
title: 'No.',
key: 'no',
width: 50
width: 50,
fixed: 'left'
},
{
title: '字幕',
key: 'srt',
width: '180',
fixed: 'left',
render(row) {
let tmp
if (row.subValue.length > 0) {
@ -187,10 +205,10 @@ export default defineComponent({
func: {
refreshTagData: async () => {
try {
await InitTags()
await props.InitTags()
message.success('人物角色刷新成功')
} catch (error) {
message.success('人物角色刷新失败,请重试')
message.error('人物角色刷新失败,请重试' + error.toString())
}
}
}
@ -215,6 +233,7 @@ export default defineComponent({
type: 'title',
row: row,
index: index,
tags: tags,
func: {
translateAll: TranslateAll,
addPrefix: AddPrefix,
@ -225,6 +244,9 @@ export default defineComponent({
},
key: 'gpt_prompt',
className: 'prompt_row',
resizable: true,
minWidth: 300,
width: '500',
render(row, index) {
return h(DataTableGptPromptRow, {
type: 'data',
@ -326,32 +348,31 @@ export default defineComponent({
}
}
async function InitTags() {
await window.mj.GetTagDataByTypeAndProperty(['dynamic', null], (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
tags.value = value.data
})
}
onMounted(async () => {
zhAutoTranslate.value = window.config.translation_auto
await window.api.GetConfigJson(JSON.stringify(['image_style', []]), async (value) => {
// cinfig
await window.api.GetConfigJson(JSON.stringify([null, {}]), async (value) => {
debugger
if (value.code == 0) {
message.error(value.message)
return
}
await window.api.GetImageStyleInfomation(JSON.stringify(toRaw(value.data)), (value) => {
let image_style_list = value.data.image_style
let customize_image_style_list = value.data.customize_image_style
await window.api.GetImageStyleInfomation(JSON.stringify(image_style_list), (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
selectStyle.value = value.data
})
for (let i = 0; i < customize_image_style_list.length; i++) {
const element = customize_image_style_list[i]
selectStyle.value.push({
key: element,
type: 'style_main'
})
}
})
// datachinese text
@ -410,7 +431,7 @@ export default defineComponent({
}
)
await InitTags()
await props.InitTags()
})
onBeforeUnmount(async () => {
@ -438,11 +459,9 @@ export default defineComponent({
}
delete value.type
delete value.code
data.value[index]['mj_message'] = value
} else if (value.type == 'updated') {
console.log('接收Discord的更新消息', value)
debugger
//
let index = data.value.findIndex((item) => item.mj_message?.message_id == value.message_id)
if (index < 0) {
@ -461,8 +480,6 @@ export default defineComponent({
console.log('接收Discord的删除消息', value)
} else if (value.type == 'finished') {
console.log('接收Discord的完成消息', value)
debugger
//
let index = data.value.findIndex((item) => item.mj_message?.image_id == value.image_id)
if (index < 0) {
@ -567,6 +584,7 @@ export default defineComponent({
* 选择生成图片的风格
*/
async function SelectGenerateImagesStyle() {
debugger
//
//
let dialogWidth = window.innerWidth * 0.8
@ -578,26 +596,47 @@ export default defineComponent({
content: () =>
h(SelectImageStyle, {
selectStyle: toRaw(selectStyle.value),
height: dialogHeight
height: dialogHeight,
tags: tags
}),
style: `width : ${dialogWidth}px; height : ${dialogHeight}px`,
maskClosable: false,
onClose: async () => {
//
let tmp_arr = []
let cus_arr = []
for (let i = 0; i < selectStyle.value.length; i++) {
debugger
const element = selectStyle.value[i]
tmp_arr.push(element.id)
//
if (element.type == 'style_main') {
cus_arr.push(element.key)
} else {
tmp_arr.push(element.id)
}
}
//
await window.api.SaveCopywritingInformation(
[JSON.stringify(tmp_arr), 'image_style', true],
(value) => {
async (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
message.success('保存成功')
await window.api.SaveCopywritingInformation(
[JSON.stringify(cus_arr), 'customize_image_style', true],
() => {
if (value.code == 0) {
message.error(value.message)
return
}
message.success('保存成功')
}
)
}
)
}
@ -808,12 +847,128 @@ export default defineComponent({
}
}
/**
* SD 模式合并名并
* @param character_string
* @param scene_string
* @param gpt_prompt
*/
function SDMergePrompt(element) {
let character_string = ''
element.character_tags?.forEach((item) => {
character_string += ', ' + item.prompt
if (item.lora && item.lora != '无' && item.lora_weight) {
character_string += `, <lora:${item.lora}:${item.lora_weight}>`
}
})
// character_string
character_string = checkStringValueDeletePrefix(character_string, ',')
//
let scene_string = ''
element.scene_tags?.forEach((item) => {
scene_string += ', ' + item.prompt
})
let style_string = ''
selectStyle.value.forEach((item) => {
if (item.type && item.type == 'style_main') {
style_string += ', ' + item.prompt
}
})
// scene_string
scene_string = checkStringValueDeletePrefix(scene_string, ',')
style_string = checkStringValueDeletePrefix(style_string, ',')
style_string = checkStringValueAddSuffix(style_string, ',')
return `${style_string} ${checkStringValueAddSuffix(
character_string,
', '
)}${checkStringValueAddSuffix(scene_string, ', ')}${checkStringValueAddSuffix(
element.gpt_prompt,
', '
)}`
}
/**
* 将所有的数据进行拼接
* @param prefix 前缀
* @param character_string 人物提示词
* @param scene_string 场景提示词
* @param gpt_prompt GPT提示词
* @param suffix 后缀
*/
function MJMergePrompt(prefix, element, suffix) {
debugger
let character_string = ''
let cref_url = ''
element.character_tags?.forEach((item) => {
character_string += ', ' + item.prompt
if (item.image_url && item.image_url != '' && item.cref_cw) {
cref_url += ` ${item.image_url} `
}
})
// cref_url --cref
cref_url = checkStringValueAddPrefix(cref_url, ' --cref ')
if (element.character_tags && element.character_tags.length > 0) {
cref_url = checkStringValueAddSuffix(cref_url, ` --cw ${element.character_tags[0].cref_cw} `)
}
// character_string
character_string = checkStringValueDeletePrefix(character_string, ',')
//
let scene_string = ''
element.scene_tags?.forEach((item) => {
scene_string += ', ' + item.prompt
})
// scene_string
scene_string = checkStringValueDeletePrefix(scene_string, ',')
//
let style_string = ''
let style_url = ''
let sw = 0
selectStyle.value.forEach((item) => {
if (item.type && item.type == 'style_main') {
style_string += ', ' + item.prompt
if (sw == 0) {
sw = item.sref_sw
}
if (!isEmpty(item.image_url)) {
let url = item.image_url ? item.image_url : ''
style_url += ' ' + url
}
}
})
style_url = checkStringValueAddPrefix(style_url, '--sref ')
style_url = checkStringValueAddSuffix(style_url, ` --sw ${sw}`)
style_string = checkStringValueDeletePrefix(style_string, ',')
style_string = checkStringValueAddSuffix(style_string, ', ')
return ` ${style_string} ${checkStringValueAddSuffix(
prefix,
', '
)}${checkStringValueAddSuffix(character_string, ', ')}${checkStringValueAddSuffix(
scene_string,
', '
)}${checkStringValueAddSuffix(element.gpt_prompt, ', ')} ${checkStringValueAddSuffix(
suffix,
' '
)} ${cref_url} ${style_url}`
}
/**
* 生成提示词
*/
async function MergePrompt() {
debugger
async function MergePrompt(key) {
try {
debugger
//
// ID
let style_str = ''
@ -823,29 +978,34 @@ export default defineComponent({
}
//
let prefix = prefix_prompt.value
let suffix = suffix_prompt.value
let prefix = prefix_prompt.value ? prefix_prompt.value : ''
let suffix = suffix_prompt.value ? suffix_prompt.value : ''
for (let i = 0; i < data.value.length; i++) {
const element = data.value[i]
//
//
let character_string = ''
element.character_tags?.forEach((item) => {
character_string += item.prompt + ', '
})
//
let scene_string = ''
element.scene_tags?.forEach((item) => {
scene_string += item.prompt + ', '
})
//
let end_prompt = ''
if (key == null) {
//
if (image_generate_category.value == 'sd') {
end_prompt = SDMergePrompt(element)
} else if (image_generate_category.value == 'mj') {
end_prompt = MJMergePrompt(prefix, element, suffix)
} else if (image_generate_category.value == 'd3') {
message.error('该分类暂时不可用')
return
} else {
message.error('合并提示词错误:未知的合并提示词类型')
return
}
} else if (key == 'mj_merge') {
end_prompt = MJMergePrompt(prefix, element, suffix)
} else if (key == 'sd_merge') {
end_prompt = SDMergePrompt(element)
} else {
message.error('合并提示词错误:未知的合并提示词类型')
return
}
// + + + + +
let end_prompt = ` ${style_str} ${
prefix ? prefix : ''
}, ${character_string} ${scene_string} ${element.gpt_prompt}, ${suffix ? suffix : ''}`
data.value[i].prompt = end_prompt
}
} catch (error) {
@ -910,7 +1070,6 @@ export default defineComponent({
selectKey.value = rowKeys
},
promptChineseRef,
zhAutoTranslate,
TranslatePrompt,
TranslateAll,
maxHeight,

View File

@ -1,99 +1,146 @@
<template>
<div id="menu">
<MenuButton :initData="data" :initDataFunction="InitData" :Character="AnalyzeCharacter" :treeData="TagTreeData">
</MenuButton>
</div>
<div id="data-table" style="margin-top: 10px;">
<DataTable :initData="data" :suffix_prompt="suffix_prompt" :prefix_prompt="prefix_prompt"></DataTable>
</div>
<div id="menu">
<MenuButton
:initData="data"
:initDataFunction="InitData"
:Character="AnalyzeCharacter"
:treeData="TagTreeData"
>
</MenuButton>
</div>
<div id="data-table" style="margin-top: 10px">
<DataTable
:tags="tags"
:initData="data"
:suffix_prompt="suffix_prompt"
:prefix_prompt="prefix_prompt"
:InitTags="InitTags"
></DataTable>
</div>
</template>
<script>
import { ref, h, onMounted, defineComponent, onUnmounted, toRaw } from "vue"
import { NImage, useMessage } from "naive-ui";
import { DEFINE_STRING } from "../../../../define/define_string";
import { ref, h, onMounted, defineComponent, onUnmounted, toRaw } from 'vue'
import { NImage, useMessage } from 'naive-ui'
import { DEFINE_STRING } from '../../../../define/define_string'
import MenuButton from './MenuButton.vue'
import DataTable from './DataTable.vue'
export default defineComponent({
components: {
NImage, DataTable, MenuButton
},
setup() {
let message = useMessage();
let data = ref([]);
let AnalyzeCharacter = ref("");
let TagTreeData = ref([]);
let suffix_prompt = ref(null);
let prefix_prompt = ref(null);
let promptError = true;
components: {
NImage,
DataTable,
MenuButton
},
setup() {
let message = useMessage()
let data = ref([])
let tags = ref({})
let AnalyzeCharacter = ref('')
let TagTreeData = ref([])
let suffix_prompt = ref(null)
let prefix_prompt = ref(null)
let promptError = true
async function InitData() {
//
//
// 稿
await window.api.GetProjectWord(value => {
data.value = value.data;
})
if (!data.value) {
data.value = [];
}
async function InitData() {
//
//
// 稿
await window.api.GetProjectWord((value) => {
data.value = value.data
})
if (!data.value) {
data.value = []
}
//
//
await window.api.GetConfigJson(JSON.stringify(['auto_analyze_character', ""]), (value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
AnalyzeCharacter.value = value.data;
})
//
await window.api.GetConfigJson(JSON.stringify([null, {}]), async (value) => {
debugger
if (value.code == 0) {
message.error(value.message);
return;
}
suffix_prompt.value = value.data.suffix_prompt;
prefix_prompt.value = value.data.prefix_prompt;
})
// dataprompt
for (let i = 0; i < data.value.length; i++) {
const item = data.value[i];
await window.api.GetPromptJson(item.name, (value) => {
debugger
console.log(value);
if (value.code == 0 && promptError) {
message.error(value.message);
promptError = false;
}
data.value[i].prompt = value.data.webui_config.prompt;
data.value[i].chinese_prompt = value.data?.chinese_prompt;
// data.value[i].gpt_prompt = value.data?.gpt_prompt;
data.value[i].adetailer = value.data?.adetailer;
})
}
//
//
await window.api.GetConfigJson(JSON.stringify(['auto_analyze_character', '']), (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
AnalyzeCharacter.value = value.data
})
onMounted(async () => {
await InitData();
//
await window.api.GetConfigJson(JSON.stringify([null, {}]), async (value) => {
debugger
if (value.code == 0) {
message.error(value.message)
return
}
suffix_prompt.value = value.data.suffix_prompt
prefix_prompt.value = value.data.prefix_prompt
})
// dataprompt
for (let i = 0; i < data.value.length; i++) {
const item = data.value[i]
await window.api.GetPromptJson(item.name, (value) => {
debugger
console.log(value)
if (value.code == 0 && promptError) {
message.error(value.message)
promptError = false
}
data.value[i].prompt = value.data.webui_config.prompt
data.value[i].chinese_prompt = value.data?.chinese_prompt
// data.value[i].gpt_prompt = value.data?.gpt_prompt;
data.value[i].adetailer = value.data?.adetailer
})
return {
data,
AnalyzeCharacter,
TagTreeData,
prefix_prompt,
suffix_prompt,
InitData,
}
}
}
async function InitTags() {
await window.mj.GetTagDataByTypeAndProperty(['dynamic', null], (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
debugger
tags.value = value.data
// datatag
console.log(data, tags)
for (let i = 0; i < data.value.length; i++) {
const element = data.value[i]
if (
element.character_tags &&
tags.value.character_tags &&
tags.value.character_tags.length > 0
) {
// character_tagskeytags.character_tags,
for (let j = element.character_tags.length - 1; j >= 0; j--) {
const item = element.character_tags[j]
let index = tags.value.character_tags.findIndex((tag) => tag.key == item.key)
if (index != -1) {
element.character_tags[j] = tags.value.character_tags[index]
} else {
element.character_tags.splice(j, 1)
}
}
} else {
element.character_tags = []
}
}
})
}
onMounted(async () => {
await InitData()
await InitTags()
})
return {
data,
AnalyzeCharacter,
TagTreeData,
prefix_prompt,
suffix_prompt,
InitData,
InitTags,
tags
}
}
})
</script>

View File

@ -168,7 +168,6 @@ export default defineComponent({
mjSetting.value = value.data;
modifyOption(value.data);
}
})
}

View File

@ -1,132 +1,244 @@
<template>
<n-tabs type="line" animated size="large">
<n-tab-pane name="normal setting" tab="基础设置">
<n-form ref="formRef" :label-width="80" :model="formValue">
<n-form-item label="SD请求地址" path="webui_api_url">
<n-input v-model:value="formValue.webui_api_url" placeholder="输入SD地址" />
</n-form-item>
<n-form-item label="出图方式(文生图、图生图)" path="type">
<n-select v-model:value="formValue.type" :options="modelOption" placeholder="输入采样方法" />
</n-form-item>
<n-form-item label="正向提示词和推导出来的tag进行拼接" path="prompt">
<n-input v-model:value="formValue.prompt" placeholder="输入正向提示词" />
</n-form-item>
<n-form-item label="反向提示词" path="negative_prompt">
<n-input v-model:value="formValue.negative_prompt" placeholder="输入反向提示词" />
</n-form-item>
<div style="display: flex">
<n-form-item label="CFG Scale" style="margin-right: 30px;">
<n-input-number :min="0" :max="30" :step="0.05" v-model:value="formValue.cfg_scale"
placeholder="输入重绘幅度" />
</n-form-item>
<n-form-item label="重绘幅度" path="denoising_strength">
<n-input-number :precision="2" :step="0.05" v-model:value="formValue.denoising_strength"
placeholder="输入重绘幅度" />
<n-checkbox style="margin-left: 30px;" size="large" v-model:checked="formValue.adetailer"
label="是否开启修脸/修手" />
</n-form-item>
</div>
<n-form-item label="采样方式" path="sampler_name">
<n-input v-model:value="formValue.sampler_name" placeholder="输入模型" />
</n-form-item>
<div style="display: flex;">
<n-form-item label="迭代步数" path="steps">
<n-input-number style="width: 150px;" v-model:value="formValue.steps" placeholder="输入迭代步数" />
</n-form-item>
<n-form-item label="风格词权重" path="style_weight" style="margin-left: 10px;">
<n-input-number :min="1" :max="4" :step="0.05" :precision="2" style="width: 150px;"
v-model:value="formValue.style_weight" placeholder="输入风格词权重" />
</n-form-item>
</div>
<n-form-item label="图片分辨率" path="resolution">
<n-input-number :show-button="false" v-model:value="formValue.width" placeholder="输入迭代步数" />
<span style="margin: 0 10px;"> * </span>
<n-input-number :show-button="false" v-model:value="formValue.height" placeholder="输入迭代步数" />
</n-form-item>
<n-form-item>
<n-button attr-type="button" type="primary" @click="SaveSDConfig">
保存设置
</n-button>
</n-form-item>
</n-form>
</n-tab-pane>
<n-tab-pane name="ADetailer setting" tab="ADetailer 设置">
<SDADetailerSetting />
</n-tab-pane>
</n-tabs>
<n-tabs type="line" animated size="large">
<n-tab-pane name="normal setting" tab="基础设置">
<n-form ref="formRef" :label-width="80" :model="formValue">
<n-form-item label="SD请求地址" path="webui_api_url">
<n-input
v-model:value="formValue.webui_api_url"
style="width: 250px"
placeholder="输入SD地址"
/>
<n-button style="margin-left: 10px" type="primary" @click="LoadSDServiceData"
>加载数据</n-button
>
</n-form-item>
<n-form-item label="出图方式(文生图、图生图)" path="type">
<n-select
v-model:value="formValue.type"
:options="modelOption"
placeholder="输入采样方法"
/>
</n-form-item>
<n-form-item label="正向提示词和推导出来的tag进行拼接" path="prompt">
<n-input v-model:value="formValue.prompt" placeholder="输入正向提示词" />
</n-form-item>
<n-form-item label="反向提示词" path="negative_prompt">
<n-input v-model:value="formValue.negative_prompt" placeholder="输入反向提示词" />
</n-form-item>
<div style="display: flex">
<n-form-item label="CFG Scale" style="margin-right: 30px">
<n-input-number
:min="0"
:max="30"
:step="0.05"
v-model:value="formValue.cfg_scale"
placeholder="输入重绘幅度"
/>
</n-form-item>
<n-form-item label="重绘幅度" path="denoising_strength">
<n-input-number
:precision="2"
:step="0.05"
v-model:value="formValue.denoising_strength"
placeholder="输入重绘幅度"
/>
<n-checkbox
style="margin-left: 30px"
size="large"
v-model:checked="formValue.adetailer"
label="是否开启修脸/修手"
/>
</n-form-item>
</div>
<n-form-item label="采样方式" path="sampler_name">
<n-input v-model:value="formValue.sampler_name" placeholder="输入模型" />
</n-form-item>
<div style="display: flex">
<n-form-item label="迭代步数" path="steps">
<n-input-number
style="width: 150px"
v-model:value="formValue.steps"
placeholder="输入迭代步数"
/>
</n-form-item>
<n-form-item label="风格词权重" path="style_weight" style="margin-left: 10px">
<n-input-number
:min="1"
:max="4"
:step="0.05"
:precision="2"
style="width: 150px"
v-model:value="formValue.style_weight"
placeholder="输入风格词权重"
/>
</n-form-item>
</div>
<n-form-item label="图片分辨率" path="resolution">
<n-input-number
:show-button="false"
v-model:value="formValue.width"
placeholder="输入迭代步数"
/>
<span style="margin: 0 10px"> * </span>
<n-input-number
:show-button="false"
v-model:value="formValue.height"
placeholder="输入迭代步数"
/>
</n-form-item>
<n-form-item>
<n-button attr-type="button" type="primary" @click="SaveSDConfig"> 保存设置 </n-button>
</n-form-item>
</n-form>
</n-tab-pane>
<n-tab-pane name="ADetailer setting" tab="ADetailer 设置">
<SDADetailerSetting />
</n-tab-pane>
</n-tabs>
</template>
<script>
import { defineComponent, ref, h, onMounted, toRaw } from "vue";
import { NForm, NFormItem, NInput, NButton, useMessage, NInputNumber, NSelect, NTabs, NTabPane, NCheckbox } from "naive-ui"
import SDADetailerSetting from "../Components/SDADetailerSetting.vue"
import { defineComponent, ref, h, onMounted, toRaw } from 'vue'
import {
NForm,
NFormItem,
NInput,
NButton,
useMessage,
NInputNumber,
NSelect,
NTabs,
NTabPane,
NCheckbox
} from 'naive-ui'
import SDADetailerSetting from '../Components/SDADetailerSetting.vue'
export default defineComponent({
components: {
NForm, NFormItem, NInput, NButton, NInputNumber, NSelect, NTabs, NTabPane, SDADetailerSetting, NCheckbox
},
setup() {
let formValue = ref({
webui_api_url: window.config.webui_api_url,
type: null,
prompt: null,
negative_prompt: null,
denoising_strength: 1,
sampler_name: null,
steps: 15,
width: 0,
height: 0,
adetailer: false,
style_weight: 1,
cfg_scale: 1
})
components: {
NForm,
NFormItem,
NInput,
NButton,
NInputNumber,
NSelect,
NTabs,
NTabPane,
SDADetailerSetting,
NCheckbox
},
setup() {
let formValue = ref({
webui_api_url: window.config.webui_api_url,
type: null,
prompt: null,
negative_prompt: null,
denoising_strength: 1,
sampler_name: null,
steps: 15,
width: 0,
height: 0,
adetailer: false,
style_weight: 1,
cfg_scale: 1,
sd_models: null,
lora: null
})
let modelOption = ref([{
label: "文生图",
value: "txt2img"
}, {
label: "图生图",
value: "img2img"
}]);
let samplers_options = ref([])
let lora_options = ref([])
let sd_models_options = ref([])
onMounted(async () => {
await window.api.InitSDConfig((value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
formValue.value = value.data;
})
})
let modelOption = ref([
{
label: '文生图',
value: 'txt2img'
},
{
label: '图生图',
value: 'img2img'
}
])
let message = useMessage();
/**
* 保存SD设置
*/
async function SaveSDConfig() {
// SD
await window.api.SaveSDConfig(toRaw(formValue.value), (value) => {
if (value.code == 0) {
window.api.showGlobalMessageDialog(value);
return;
} else if (value.code == 1) {
window.api.showGlobalMessageDialog(value);
window.api.getSettingDafultData((value) => {
window.config = value;
})
return
} else {
window.api.showGlobalMessageDialog({ code: 0, message: "未知错误" });
}
})
onMounted(async () => {
await window.api.InitSDConfig((value) => {
if (value.code == 0) {
message.error(value.message)
return
}
formValue.value = value.data
return {
formValue,
SaveSDConfig,
modelOption,
let samplers = value.data.samplers
for (let i = 0; i < samplers.length; i++) {
const element = samplers[i]
samplers_options.value.push({
label: element.name,
value: element.name
})
}
let loras = value.data.loras
for (let i = 0; i < loras.length; i++) {
const element = loras[i]
lora_options.value.push({
label: element.name,
value: element.name
})
}
let sd_models = value.data.sd_models
for (let i = 0; i < sd_models.length; i++) {
const element = sd_models[i]
sd_models_options.value.push({
label: element.title,
value: element.title
})
}
})
})
let message = useMessage()
/**
* 保存SD设置
*/
async function SaveSDConfig() {
// SD
await window.api.SaveSDConfig(toRaw(formValue.value), (value) => {
if (value.code == 0) {
window.api.showGlobalMessageDialog(value)
return
} else if (value.code == 1) {
window.api.showGlobalMessageDialog(value)
window.api.getSettingDafultData((value) => {
window.config = value
})
return
} else {
window.api.showGlobalMessageDialog({ code: 0, message: '未知错误' })
}
})
}
/**
* 加载SD数据
*/
async function LoadSDServiceData() {
await window.sd.LoadSDServiceData(toRaw(formValue.value).webui_api_url, (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
// formValue.value = value.data
message.success('加载成功')
})
}
return {
formValue,
SaveSDConfig,
modelOption,
LoadSDServiceData,
samplers_options,
lora_options,
sd_models_options
}
}
})
</script>

View File

@ -1,379 +1,481 @@
<template>
<NSpace vertical>
<n-spin :show="show">
<template #description>
保存中
</template>
<n-form ref="formRef" :model="formValue" :rules="rules">
<n-form-item path="draft_path" label="剪映草稿地址">
<n-input v-model:value="formValue.draft_path" @keydown.enter.prevent />
</n-form-item>
<n-form-item path="project_path" label="项目地址(帧,图片等输出的地址)">
<div>
<n-input style="width: 400px;" v-model:value="formValue.project_path" @keydown.enter.prevent />
</div>
<n-button color="#e5a84b" @click="SelectProjectFolder" style="margin-left: 10px;">
<n-icon :size="20">
<folder-open />
</n-icon>
</n-button>
</n-form-item>
<n-form-item path="project_name" label="项目名称">
<n-input v-model:value="formValue.project_name" @keydown.enter.prevent />
</n-form-item>
<n-form-item path="system_setting" label="系统设置">
<n-form-item path="task_number" label="后台并行任务数(量力而行)">
<n-input-number v-model:value="formValue.task_number" @keydown.enter.prevent :min="1" :max="10"
placeholder="输入后台并行任务数" />
</n-form-item>
<n-form-item label="系统主题" style="margin-left: 20px;">
<n-switch :rail-style="railStyle" v-model:value="formValue.theme" size="medium"
checked-value="dark" unchecked-value="light" @update:value="ChangeMode">
<template #checked>
</template>
<template #unchecked>
</template>
</n-switch>
</n-form-item>
<n-form-item label="人物场景选择模式" style="margin-left: 20px;">
<n-select size="small" placeholder="请选择" :options="character_select_model_options"
v-model:value="formValue.character_select_model" style="width: 120px;" />
</n-form-item>
</n-form-item>
<n-form-item path="gpt_setting" label="GPT设置">
<n-form-item path="gpt_business" label="GPT接口服务商">
<n-select v-model:value="formValue.gpt_business" :options="gpt_options" placeholder="GPT接口服务商"
style="width: 200px;" />
</n-form-item>
<n-button type="info" @click="openGptBuyUrl">
购买
</n-button>
<n-form-item style="width: 300px; margin-left: 30px;" path="gpt_key" label="GPT Key">
<n-input type="password" show-password-on="mousedown" v-model:value="formValue.gpt_key"
@keydown.enter.prevent placeholder="GPT Key" />
</n-form-item>
<n-form-item style="margin-left: 30px;" path="gpt_model" label="GPT模型">
<n-select v-model:value="formValue.gpt_model" :options="gpt_model_options" placeholder="GPT模型"
style="width: 160px;" />
</n-form-item>
<NSpace vertical>
<n-spin :show="show">
<template #description> 保存中 </template>
<n-form ref="formRef" :model="formValue" :rules="rules">
<n-form-item path="draft_path" label="剪映草稿地址">
<n-input v-model:value="formValue.draft_path" @keydown.enter.prevent />
</n-form-item>
<n-form-item path="project_path" label="项目地址(帧,图片等输出的地址)">
<div>
<n-input
style="width: 400px"
v-model:value="formValue.project_path"
@keydown.enter.prevent
/>
</div>
<n-button color="#e5a84b" @click="SelectProjectFolder" style="margin-left: 10px">
<n-icon :size="20">
<folder-open />
</n-icon>
</n-button>
</n-form-item>
<n-form-item path="project_name" label="项目名称">
<n-input v-model:value="formValue.project_name" @keydown.enter.prevent />
</n-form-item>
<n-form-item path="system_setting" label="系统设置">
<n-form-item path="task_number" label="后台并行任务数(量力而行)">
<n-input-number
v-model:value="formValue.task_number"
@keydown.enter.prevent
:min="1"
:max="10"
placeholder="输入后台并行任务数"
/>
</n-form-item>
<n-form-item label="系统主题" style="margin-left: 20px">
<n-switch
:rail-style="railStyle"
v-model:value="formValue.theme"
size="medium"
checked-value="dark"
unchecked-value="light"
@update:value="ChangeMode"
>
<template #checked> </template>
<template #unchecked> </template>
</n-switch>
</n-form-item>
<n-form-item label="人物场景选择模式" style="margin-left: 20px">
<n-select
size="small"
placeholder="请选择"
:options="character_select_model_options"
v-model:value="formValue.character_select_model"
style="width: 120px"
/>
</n-form-item>
<n-form-item style="margin-left: 20px"
><n-checkbox v-model:checked="formValue.window_wh_bm_remember"
>开启窗口大小/位置记忆</n-checkbox
>
</n-form-item>
</n-form-item>
<n-form-item path="gpt_setting" label="GPT设置">
<n-form-item path="gpt_business" label="GPT接口服务商">
<n-select
v-model:value="formValue.gpt_business"
:options="gpt_options"
placeholder="GPT接口服务商"
style="width: 200px"
/>
</n-form-item>
<n-button type="info" @click="openGptBuyUrl"> 购买 </n-button>
<n-form-item style="width: 300px; margin-left: 30px" path="gpt_key" label="GPT Key">
<n-input
type="password"
show-password-on="mousedown"
v-model:value="formValue.gpt_key"
@keydown.enter.prevent
placeholder="GPT Key"
/>
</n-form-item>
<n-form-item style="margin-left: 30px" path="gpt_model" label="GPT模型">
<n-select
v-model:value="formValue.gpt_model"
:options="gpt_model_options"
placeholder="GPT模型"
style="width: 160px"
/>
</n-form-item>
<n-form-item style="margin-left: 30px;" path="gpt_count" label="推理上下文行数">
<n-input-number v-model:value="formValue.gpt_count" placeholder="GPT模型" style="width: 150px;"
:show-button="false" :min="1" :max="100" />
</n-form-item>
<n-button color="#e18a3b" @click="addGptOption" style="margin-left: 10px;">
添加GPT
</n-button>
<n-button color="#779649" @click="TestGPTConnection" :loading="loading" style="margin-left: 10px;">
测试链接
</n-button>
<n-button color="#ba5b49" @click="AddGptPrompt" style="margin-left: 10px;">
添加GPT提示词
</n-button>
</n-form-item>
<n-form-item path="translation_setting" label="翻译设置">
<n-form-item path="translation_business" label="翻译服务商">
<n-select v-model:value="formValue.translation_business" :options="translation_options"
@update:value="SwitchTranslate" placeholder="翻译服务商" style="width: 200px;" />
</n-form-item>
<n-form-item path="translation_app_id" label="APP ID" style="width: 300px; margin-left: 30px;">
<n-input type="password" show-password-on="mousedown"
v-model:value="formValue.translation_app_id" @keydown.enter.prevent placeholder="APP ID" />
</n-form-item>
<n-form-item path="translation_secret" label="产品密钥" style="width: 300px; margin-left: 30px;">
<n-input type="password" show-password-on="mousedown"
v-model:value="formValue.translation_secret" placeholder="请输入产品密钥"></n-input>
</n-form-item>
<n-form-item path="translation_secret" label="自动翻译" style="width: 300px; margin-left: 30px;">
<n-checkbox v-model:checked="formValue.translation_auto" placeholder="请输入产品密钥"></n-checkbox>
</n-form-item>
</n-form-item>
<n-form-item>
<n-button
:disabled="formValue.draft_path == null || formValue.draft_path == '' || formValue.project_path == null || formValue.project_path == ''"
round type="primary" @click="handleValidateButtonClick">
保存
</n-button>
</n-form-item>
</n-form>
</n-spin>
</NSpace>
<n-form-item style="margin-left: 30px" path="gpt_count" label="推理上下文行数">
<n-input-number
v-model:value="formValue.gpt_count"
placeholder="GPT模型"
style="width: 150px"
:show-button="false"
:min="1"
:max="100"
/>
</n-form-item>
<n-button color="#e18a3b" @click="addGptOption" style="margin-left: 10px">
添加GPT
</n-button>
<n-button
color="#779649"
@click="TestGPTConnection"
:loading="loading"
style="margin-left: 10px"
>
测试链接
</n-button>
<n-button color="#ba5b49" @click="AddGptPrompt" style="margin-left: 10px">
添加GPT提示词
</n-button>
</n-form-item>
<n-form-item path="translation_setting" label="翻译设置">
<n-form-item path="translation_business" label="翻译服务商">
<n-select
v-model:value="formValue.translation_business"
:options="translation_options"
@update:value="SwitchTranslate"
placeholder="翻译服务商"
style="width: 200px"
/>
</n-form-item>
<n-form-item
path="translation_app_id"
label="APP ID"
style="width: 300px; margin-left: 30px"
>
<n-input
type="password"
show-password-on="mousedown"
v-model:value="formValue.translation_app_id"
@keydown.enter.prevent
placeholder="APP ID"
/>
</n-form-item>
<n-form-item
path="translation_secret"
label="产品密钥"
style="width: 300px; margin-left: 30px"
>
<n-input
type="password"
show-password-on="mousedown"
v-model:value="formValue.translation_secret"
placeholder="请输入产品密钥"
></n-input>
</n-form-item>
<n-form-item
path="translation_secret"
label="自动翻译"
style="width: 300px; margin-left: 30px"
>
<n-checkbox v-model:checked="formValue.translation_auto"></n-checkbox>
</n-form-item>
</n-form-item>
<n-form-item>
<n-button
:disabled="
formValue.draft_path == null ||
formValue.draft_path == '' ||
formValue.project_path == null ||
formValue.project_path == ''
"
round
type="primary"
@click="handleValidateButtonClick"
>
保存
</n-button>
</n-form-item>
</n-form>
</n-spin>
</NSpace>
</template>
<script>
import { ref, toRaw, onMounted, defineComponent, h } from "vue"
import { NForm, NFormItem, NInput, NInputNumber, useDialog, NButton, NSpace, NSpin, useMessage, NSelect, NCheckbox, NSwitch, NIcon } from "naive-ui"
import AddGptOption from "./Components/AddGptOption.vue";
import { ref, toRaw, onMounted, defineComponent, h } from 'vue'
import {
NForm,
NFormItem,
NInput,
NInputNumber,
useDialog,
NButton,
NSpace,
NSpin,
useMessage,
NSelect,
NCheckbox,
NSwitch,
NIcon
} from 'naive-ui'
import AddGptOption from './Components/AddGptOption.vue'
import { FolderOpenSharp as FolderOpen } from '@vicons/ionicons5'
import AddGptPrompts from './Components/AddGptPrompts.vue'
export default defineComponent({
components: {
NForm, NFormItem, NInput, NInputNumber, NButton, NSpace, NSpin, NSelect, NCheckbox, NSwitch, FolderOpen, NIcon
},
setup() {
let formRef = ref(null);
let message = useMessage()
let formValue = ref({
draft_path: window.config.draft_path,
project_path: window.config.project_path,
project_name: window.config.project_name,
gpt_business: window.config.gpt_business,
gpt_key: window.config.gpt_key,
gpt_model: window.config.gpt_model,
gpt_count: window.config.gpt_count,
task_number: window.config.task_number,
translation_business: window.config.translation_business,
translation_app_id: window.config.translation_app_id,
translation_secret: window.config.translation_secret,
translation_auto: window.config.translation_auto,
theme: window.config.theme,
character_select_model: window.config.character_select_model
});
let show = ref(false)
let gpt_options = ref([]);
let gpt_model_options = ref([]);
let character_select_model_options = ref([])
let dialog = useDialog();
let loading = ref(false);
components: {
NForm,
NFormItem,
NInput,
NInputNumber,
NButton,
NSpace,
NSpin,
NSelect,
NCheckbox,
NSwitch,
FolderOpen,
NIcon
},
setup() {
let formRef = ref(null)
let message = useMessage()
let formValue = ref({
draft_path: window.config.draft_path,
project_path: window.config.project_path,
project_name: window.config.project_name,
gpt_business: window.config.gpt_business,
gpt_key: window.config.gpt_key,
gpt_model: window.config.gpt_model,
gpt_count: window.config.gpt_count,
task_number: window.config.task_number,
translation_business: window.config.translation_business,
translation_app_id: window.config.translation_app_id,
translation_secret: window.config.translation_secret,
translation_auto: window.config.translation_auto,
theme: window.config.theme,
character_select_model: window.config.character_select_model,
window_wh_bm_remember: window.config.window_wh_bm_remember
})
let show = ref(false)
let gpt_options = ref([])
let gpt_model_options = ref([])
let character_select_model_options = ref([])
let dialog = useDialog()
let loading = ref(false)
/**
* 加载GPT的配置信息
*/
async function InitGptOptions() {
debugger;
await window.api.getGptBusinessOption("all", (value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
gpt_options.value = value.data;
})
await window.api.getGptModelOption("all", (value) => {
if (value.code == 0) {
message.error(value.message);
return;
}
gpt_model_options.value = value.data;
})
await window.mj.GetTagSelectModel(value => {
character_select_model_options.value = value.data;
})
/**
* 加载GPT的配置信息
*/
async function InitGptOptions() {
debugger
await window.api.getGptBusinessOption('all', (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
gpt_options.value = value.data
})
onMounted(async () => {
await InitGptOptions();
})
let translation_options = [{
label: "百度翻译",
value: "https://fanyi-api.baidu.com/api/trans/vip/translate"
}, {
label: "火山引擎",
value: "https://translate.volcengineapi.com?"
},
{
label: "腾讯翻译",
value: "https://tmt.tencentcloudapi.com"
},
{
label: "阿里翻译",
value: "https://mt.cn-hangzhou.aliyuncs.com"
await window.api.getGptModelOption('all', (value) => {
if (value.code == 0) {
message.error(value.message)
return
}
];
gpt_model_options.value = value.data
})
//
function SwitchTranslate(value, option) {
//
formValue.value.translation_app_id = null;
formValue.value.translation_secret = null;
}
let ruleObj = (errorMessage) => {
return [{
required: true,
validator(rule, value) {
if (value == null || value == "")
return new Error(errorMessage);
return true;
},
trigger: ["input", "blur", "change"]
}]
}
let rules = {
draft_path: ruleObj("必填剪映草稿地址"),
project_path: ruleObj("必填项目地址"),
project_name: ruleObj("必填项目名称"),
task_number: ruleObj("必填后台并行任务数"),
gpt_business: ruleObj("必填GPT接口服务商"),
gpt_key: ruleObj("必填GPT Key"),
gpt_model: ruleObj("必填GPT模型"),
gpt_count: ruleObj("必填自动推理上下文行数"),
translation_secret: ruleObj("必填产品密钥"),
translation_app_id: ruleObj("必填APP ID"),
translation_business: ruleObj("必填翻译服务商"),
};
/**
* 保存配置信息
*/
function handleValidateButtonClick(e) {
e.preventDefault();
formRef.value?.validate((errors) => {
if (errors) {
message.error("请检查必填字段");
return
}
//
window.api.ModifySampleSetting(JSON.stringify(toRaw(formValue.value)), (value) => {
if (value.code == 1) {
window.api.getSettingDafultData((value) => {
window.config = value;
})
window.api.showGlobalMessageDialog({ code: 1, message: value.message })
} else {
window.api.showGlobalMessageDialog({ code: 0, message: value.message })
}
})
});
}
let railStyle = (focused,
checked) => {
const style = {};
if (checked) {
style.background = "#f3a694";
if (focused) {
style.boxShadow = "0 0 0 2px #f3a694";
}
} else {
style.background = "#775039";
if (focused) {
style.boxShadow = "0 0 0 2px #775039";
}
}
return style;
}
/**
* 打开购买GPT的地址
*/
async function openGptBuyUrl() {
window.api.openGptBuyUrl(toRaw(formValue.value.gpt_business));
}
/**
* 切换主题
* @param {*} value 主题名称
*/
async function ChangeMode(value) {
debugger
console.log(value)
const isDarkMode = await window.darkMode.toggle(value)
}
/**
* 添加GPT地址或者模型
*/
async function addGptOption() {
//
//
let dialogWidth = 600;
let dialogHeight = 450;
// ImportWordAndSrt
dialog.create({
showIcon: false,
closeOnEsc: false,
title: "添加GPT地址或者模型",
content: () => h(AddGptOption, {}),
style: `width : ${dialogWidth}px; min-height : ${dialogHeight}px`,
maskClosable: false,
onClose: async () => {
//
await InitGptOptions();
}
})
}
/**
* 测试当前GPT是不是可以链接成功
*/
async function TestGPTConnection() {
loading.value = true;
//
await window.api.TestGPTConnection(JSON.stringify(formValue.value), (value) => {
loading.value = false;
if (value.code == 0) {
window.api.showGlobalMessageDialog({ code: 0, message: "GPT链接失败错误信息 " + value.message })
return;
}
window.api.showGlobalMessageDialog({ code: 1, message: "gpt链接测试成功可以正常使用" })
})
}
/**
* 选择项目文件夹
*/
async function SelectProjectFolder() {
await window.api.selectFolder({ default_paht: toRaw(formValue.value).project_path }, (value) => {
if (value.length > 0) {
formValue.value.project_path = value[0]
}
})
}
/**
* 添加GPT提示词预设
*/
async function AddGptPrompt() {
//
//
let dialogWidth = 600;
let dialogHeight = window.innerHeight * 0.9;
// ImportWordAndSrt
dialog.create({
showIcon: false,
closeOnEsc: false,
title: "添加GPT提示词预设",
content: () => h(AddGptPrompts, { height: dialogHeight }),
style: `width : ${dialogWidth}px; height : ${dialogHeight}px; padding-right : 3px `,
maskClosable: false,
onClose: () => {
}
})
}
return {
formRef,
ChangeMode,
openGptBuyUrl,
railStyle,
handleValidateButtonClick,
rules,
SwitchTranslate,
translation_options,
gpt_model_options,
gpt_options,
show,
formValue,
addGptOption,
TestGPTConnection,
loading,
SelectProjectFolder,
AddGptPrompt,
character_select_model_options,
}
await window.mj.GetTagSelectModel((value) => {
character_select_model_options.value = value.data
})
}
})
onMounted(async () => {
await InitGptOptions()
})
let translation_options = [
{
label: '百度翻译',
value: 'https://fanyi-api.baidu.com/api/trans/vip/translate'
},
{
label: '火山引擎',
value: 'https://translate.volcengineapi.com?'
},
{
label: '腾讯翻译',
value: 'https://tmt.tencentcloudapi.com'
},
{
label: '阿里翻译',
value: 'https://mt.cn-hangzhou.aliyuncs.com'
}
]
//
function SwitchTranslate(value, option) {
//
formValue.value.translation_app_id = null
formValue.value.translation_secret = null
}
let ruleObj = (errorMessage) => {
return [
{
required: true,
validator(rule, value) {
if (value == null || value == '') return new Error(errorMessage)
return true
},
trigger: ['input', 'blur', 'change']
}
]
}
let rules = {
draft_path: ruleObj('必填剪映草稿地址'),
project_path: ruleObj('必填项目地址'),
project_name: ruleObj('必填项目名称'),
task_number: ruleObj('必填后台并行任务数'),
gpt_business: ruleObj('必填GPT接口服务商'),
gpt_key: ruleObj('必填GPT Key'),
gpt_model: ruleObj('必填GPT模型'),
gpt_count: ruleObj('必填自动推理上下文行数'),
translation_secret: ruleObj('必填产品密钥'),
translation_app_id: ruleObj('必填APP ID'),
translation_business: ruleObj('必填翻译服务商')
}
/**
* 保存配置信息
*/
function handleValidateButtonClick(e) {
e.preventDefault()
formRef.value?.validate((errors) => {
if (errors) {
message.error('请检查必填字段')
return
}
//
window.api.ModifySampleSetting(JSON.stringify(toRaw(formValue.value)), (value) => {
if (value.code == 1) {
window.api.getSettingDafultData((value) => {
window.config = value
})
window.api.showGlobalMessageDialog({ code: 1, message: value.message })
} else {
window.api.showGlobalMessageDialog({ code: 0, message: value.message })
}
})
})
}
let railStyle = (focused, checked) => {
const style = {}
if (checked) {
style.background = '#f3a694'
if (focused) {
style.boxShadow = '0 0 0 2px #f3a694'
}
} else {
style.background = '#775039'
if (focused) {
style.boxShadow = '0 0 0 2px #775039'
}
}
return style
}
/**
* 打开购买GPT的地址
*/
async function openGptBuyUrl() {
window.api.openGptBuyUrl(toRaw(formValue.value.gpt_business))
}
/**
* 切换主题
* @param {*} value 主题名称
*/
async function ChangeMode(value) {
debugger
console.log(value)
const isDarkMode = await window.darkMode.toggle(value)
}
/**
* 添加GPT地址或者模型
*/
async function addGptOption() {
//
//
let dialogWidth = 600
let dialogHeight = 450
// ImportWordAndSrt
dialog.create({
showIcon: false,
closeOnEsc: false,
title: '添加GPT地址或者模型',
content: () => h(AddGptOption, {}),
style: `width : ${dialogWidth}px; min-height : ${dialogHeight}px`,
maskClosable: false,
onClose: async () => {
//
await InitGptOptions()
}
})
}
/**
* 测试当前GPT是不是可以链接成功
*/
async function TestGPTConnection() {
loading.value = true
//
await window.api.TestGPTConnection(JSON.stringify(formValue.value), (value) => {
loading.value = false
if (value.code == 0) {
window.api.showGlobalMessageDialog({
code: 0,
message: 'GPT链接失败错误信息 ' + value.message
})
return
}
window.api.showGlobalMessageDialog({ code: 1, message: 'gpt链接测试成功可以正常使用' })
})
}
/**
* 选择项目文件夹
*/
async function SelectProjectFolder() {
await window.api.selectFolder(
{ default_paht: toRaw(formValue.value).project_path },
(value) => {
if (value.length > 0) {
formValue.value.project_path = value[0]
}
}
)
}
/**
* 添加GPT提示词预设
*/
async function AddGptPrompt() {
//
//
let dialogWidth = 600
let dialogHeight = window.innerHeight * 0.9
// ImportWordAndSrt
dialog.create({
showIcon: false,
closeOnEsc: false,
title: '添加GPT提示词预设',
content: () => h(AddGptPrompts, { height: dialogHeight }),
style: `width : ${dialogWidth}px; height : ${dialogHeight}px; padding-right : 3px `,
maskClosable: false,
onClose: () => {}
})
}
return {
formRef,
ChangeMode,
openGptBuyUrl,
railStyle,
handleValidateButtonClick,
rules,
SwitchTranslate,
translation_options,
gpt_model_options,
gpt_options,
show,
formValue,
addGptOption,
TestGPTConnection,
loading,
SelectProjectFolder,
AddGptPrompt,
character_select_model_options
}
}
})
</script>