This commit is contained in:
2026-06-22 21:36:22 +08:00
parent c0b84b9ea4
commit 0fc3d6c7c8
20 changed files with 2330 additions and 518 deletions

372
package-lock.json generated
View File

@@ -9,8 +9,11 @@
"version": "0.0.0", "version": "0.0.0",
"dependencies": { "dependencies": {
"@element-plus/icons-vue": "^2.3.2", "@element-plus/icons-vue": "^2.3.2",
"axios": "^1.18.0",
"element-china-area-data": "^6.1.0",
"element-plus": "^2.14.1", "element-plus": "^2.14.1",
"vue": "^3.5.34" "vue": "^3.5.34",
"vue-router": "4"
}, },
"devDependencies": { "devDependencies": {
"@vitejs/plugin-vue": "^6.0.6", "@vitejs/plugin-vue": "^6.0.6",
@@ -549,6 +552,12 @@
"@vue/shared": "3.5.35" "@vue/shared": "3.5.35"
} }
}, },
"node_modules/@vue/devtools-api": {
"version": "6.6.4",
"resolved": "https://registry.npmjs.org/@vue/devtools-api/-/devtools-api-6.6.4.tgz",
"integrity": "sha512-sGhTPMuXqZ1rVOk32RylztWkfXTRhuS7vgAKv0zjqk8gbsHkJ7xfFf+jbySxt7tWObEJwyKaHMikV/WGDiQm8g==",
"license": "MIT"
},
"node_modules/@vue/reactivity": { "node_modules/@vue/reactivity": {
"version": "3.5.35", "version": "3.5.35",
"resolved": "https://registry.npmjs.org/@vue/reactivity/-/reactivity-3.5.35.tgz", "resolved": "https://registry.npmjs.org/@vue/reactivity/-/reactivity-3.5.35.tgz",
@@ -637,12 +646,73 @@
"vue": "^3.5.0" "vue": "^3.5.0"
} }
}, },
"node_modules/agent-base": {
"version": "6.0.2",
"resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
"integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
"license": "MIT",
"dependencies": {
"debug": "4"
},
"engines": {
"node": ">= 6.0.0"
}
},
"node_modules/async-validator": { "node_modules/async-validator": {
"version": "4.2.5", "version": "4.2.5",
"resolved": "https://registry.npmjs.org/async-validator/-/async-validator-4.2.5.tgz", "resolved": "https://registry.npmjs.org/async-validator/-/async-validator-4.2.5.tgz",
"integrity": "sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg==", "integrity": "sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/asynckit": {
"version": "0.4.0",
"resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
"integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
"license": "MIT"
},
"node_modules/axios": {
"version": "1.18.0",
"resolved": "https://registry.npmjs.org/axios/-/axios-1.18.0.tgz",
"integrity": "sha512-E32NzpYKp++W7XRe52rHiXV2ehxmh3wbdgO7MHeFM+vqxLBYHzt0ElkiImtOBxtOmyp0yoC8C6uESVV84Y2/hw==",
"license": "MIT",
"dependencies": {
"follow-redirects": "^1.16.0",
"form-data": "^4.0.5",
"https-proxy-agent": "^5.0.1",
"proxy-from-env": "^2.1.0"
}
},
"node_modules/call-bind-apply-helpers": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
"integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
"license": "MIT",
"dependencies": {
"es-errors": "^1.3.0",
"function-bind": "^1.1.2"
},
"engines": {
"node": ">= 0.4"
}
},
"node_modules/china-division": {
"version": "2.7.0",
"resolved": "https://registry.npmjs.org/china-division/-/china-division-2.7.0.tgz",
"integrity": "sha512-4uUPAT+1WfqDh5jytq7omdCmHNk3j+k76zEG/2IqaGcYB90c2SwcixttcypdsZ3T/9tN1TTpBDoeZn+Yw/qBEA==",
"license": "MIT"
},
"node_modules/combined-stream": {
"version": "1.0.8",
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
"integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
"license": "MIT",
"dependencies": {
"delayed-stream": "~1.0.0"
},
"engines": {
"node": ">= 0.8"
}
},
"node_modules/csstype": { "node_modules/csstype": {
"version": "3.2.3", "version": "3.2.3",
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz",
@@ -655,6 +725,32 @@
"integrity": "sha512-98IT+HOahAisibz/yjKbzuOBwYcjJ7BCLPzARyHiyEBmRz4fatF+KPJszEHXsGYjUG234aH/cOjW1wwTbKUZlA==", "integrity": "sha512-98IT+HOahAisibz/yjKbzuOBwYcjJ7BCLPzARyHiyEBmRz4fatF+KPJszEHXsGYjUG234aH/cOjW1wwTbKUZlA==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/debug": {
"version": "4.4.3",
"resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
"integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
"license": "MIT",
"dependencies": {
"ms": "^2.1.3"
},
"engines": {
"node": ">=6.0"
},
"peerDependenciesMeta": {
"supports-color": {
"optional": true
}
}
},
"node_modules/delayed-stream": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
"integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
"license": "MIT",
"engines": {
"node": ">=0.4.0"
}
},
"node_modules/detect-libc": { "node_modules/detect-libc": {
"version": "2.1.2", "version": "2.1.2",
"resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
@@ -665,6 +761,29 @@
"node": ">=8" "node": ">=8"
} }
}, },
"node_modules/dunder-proto": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
"integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
"license": "MIT",
"dependencies": {
"call-bind-apply-helpers": "^1.0.1",
"es-errors": "^1.3.0",
"gopd": "^1.2.0"
},
"engines": {
"node": ">= 0.4"
}
},
"node_modules/element-china-area-data": {
"version": "6.1.0",
"resolved": "https://registry.npmjs.org/element-china-area-data/-/element-china-area-data-6.1.0.tgz",
"integrity": "sha512-IkpcjwQv2A/2AxFiSoaISZ+oMw1rZCPUSOg5sOCwT5jKc96TaawmKZeY81xfxXsO0QbKxU5LLc6AirhG52hUmg==",
"license": "MIT",
"dependencies": {
"china-division": "^2.7.0"
}
},
"node_modules/element-plus": { "node_modules/element-plus": {
"version": "2.14.1", "version": "2.14.1",
"resolved": "https://registry.npmjs.org/element-plus/-/element-plus-2.14.1.tgz", "resolved": "https://registry.npmjs.org/element-plus/-/element-plus-2.14.1.tgz",
@@ -703,6 +822,51 @@
"url": "https://github.com/fb55/entities?sponsor=1" "url": "https://github.com/fb55/entities?sponsor=1"
} }
}, },
"node_modules/es-define-property": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
"integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
"license": "MIT",
"engines": {
"node": ">= 0.4"
}
},
"node_modules/es-errors": {
"version": "1.3.0",
"resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
"integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
"license": "MIT",
"engines": {
"node": ">= 0.4"
}
},
"node_modules/es-object-atoms": {
"version": "1.1.2",
"resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.2.tgz",
"integrity": "sha512-HWcBoN6NileqtSydK2FqHbS/LoDd2pqrnQHLyJzBj4kOp/ky2MWMN694xOfkK8/SnUsW2DH7EfyVlydKCsm1Zw==",
"license": "MIT",
"dependencies": {
"es-errors": "^1.3.0"
},
"engines": {
"node": ">= 0.4"
}
},
"node_modules/es-set-tostringtag": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz",
"integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==",
"license": "MIT",
"dependencies": {
"es-errors": "^1.3.0",
"get-intrinsic": "^1.2.6",
"has-tostringtag": "^1.0.2",
"hasown": "^2.0.2"
},
"engines": {
"node": ">= 0.4"
}
},
"node_modules/estree-walker": { "node_modules/estree-walker": {
"version": "2.0.2", "version": "2.0.2",
"resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz",
@@ -727,6 +891,42 @@
} }
} }
}, },
"node_modules/follow-redirects": {
"version": "1.16.0",
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.16.0.tgz",
"integrity": "sha512-y5rN/uOsadFT/JfYwhxRS5R7Qce+g3zG97+JrtFZlC9klX/W5hD7iiLzScI4nZqUS7DNUdhPgw4xI8W2LuXlUw==",
"funding": [
{
"type": "individual",
"url": "https://github.com/sponsors/RubenVerborgh"
}
],
"license": "MIT",
"engines": {
"node": ">=4.0"
},
"peerDependenciesMeta": {
"debug": {
"optional": true
}
}
},
"node_modules/form-data": {
"version": "4.0.6",
"resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.6.tgz",
"integrity": "sha512-vKatAh4SlVfgbv+YtmhiRjhEMJsYpsG1Y2rMQtR+SVSbytsSD1YGzDIcrAJmdFec88u/+VoGmxnl+80gL1tRCQ==",
"license": "MIT",
"dependencies": {
"asynckit": "^0.4.0",
"combined-stream": "^1.0.8",
"es-set-tostringtag": "^2.1.0",
"hasown": "^2.0.4",
"mime-types": "^2.1.35"
},
"engines": {
"node": ">= 6"
}
},
"node_modules/fsevents": { "node_modules/fsevents": {
"version": "2.3.3", "version": "2.3.3",
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
@@ -742,6 +942,116 @@
"node": "^8.16.0 || ^10.6.0 || >=11.0.0" "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
} }
}, },
"node_modules/function-bind": {
"version": "1.1.2",
"resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
"integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
"license": "MIT",
"funding": {
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/get-intrinsic": {
"version": "1.3.0",
"resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
"integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
"license": "MIT",
"dependencies": {
"call-bind-apply-helpers": "^1.0.2",
"es-define-property": "^1.0.1",
"es-errors": "^1.3.0",
"es-object-atoms": "^1.1.1",
"function-bind": "^1.1.2",
"get-proto": "^1.0.1",
"gopd": "^1.2.0",
"has-symbols": "^1.1.0",
"hasown": "^2.0.2",
"math-intrinsics": "^1.1.0"
},
"engines": {
"node": ">= 0.4"
},
"funding": {
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/get-proto": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
"integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
"license": "MIT",
"dependencies": {
"dunder-proto": "^1.0.1",
"es-object-atoms": "^1.0.0"
},
"engines": {
"node": ">= 0.4"
}
},
"node_modules/gopd": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
"integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
"license": "MIT",
"engines": {
"node": ">= 0.4"
},
"funding": {
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/has-symbols": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
"integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
"license": "MIT",
"engines": {
"node": ">= 0.4"
},
"funding": {
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/has-tostringtag": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz",
"integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==",
"license": "MIT",
"dependencies": {
"has-symbols": "^1.0.3"
},
"engines": {
"node": ">= 0.4"
},
"funding": {
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/hasown": {
"version": "2.0.4",
"resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.4.tgz",
"integrity": "sha512-T2UbfbBEF32wiepXIsMlTW9+dDYC6wMh/t/vYA4tuOMKqWz/n3vr1NFSxQiyP+zk2mXsoMA/i/7qV6LKut1t1A==",
"license": "MIT",
"dependencies": {
"function-bind": "^1.1.2"
},
"engines": {
"node": ">= 0.4"
}
},
"node_modules/https-proxy-agent": {
"version": "5.0.1",
"resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz",
"integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==",
"license": "MIT",
"dependencies": {
"agent-base": "6",
"debug": "4"
},
"engines": {
"node": ">= 6"
}
},
"node_modules/lightningcss": { "node_modules/lightningcss": {
"version": "1.32.0", "version": "1.32.0",
"resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz", "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz",
@@ -1035,12 +1345,48 @@
"@jridgewell/sourcemap-codec": "^1.5.5" "@jridgewell/sourcemap-codec": "^1.5.5"
} }
}, },
"node_modules/math-intrinsics": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
"integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
"license": "MIT",
"engines": {
"node": ">= 0.4"
}
},
"node_modules/memoize-one": { "node_modules/memoize-one": {
"version": "6.0.0", "version": "6.0.0",
"resolved": "https://registry.npmjs.org/memoize-one/-/memoize-one-6.0.0.tgz", "resolved": "https://registry.npmjs.org/memoize-one/-/memoize-one-6.0.0.tgz",
"integrity": "sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw==", "integrity": "sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/mime-db": {
"version": "1.52.0",
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
"license": "MIT",
"engines": {
"node": ">= 0.6"
}
},
"node_modules/mime-types": {
"version": "2.1.35",
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
"license": "MIT",
"dependencies": {
"mime-db": "1.52.0"
},
"engines": {
"node": ">= 0.6"
}
},
"node_modules/ms": {
"version": "2.1.3",
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
"license": "MIT"
},
"node_modules/nanoid": { "node_modules/nanoid": {
"version": "3.3.12", "version": "3.3.12",
"resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz", "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz",
@@ -1112,6 +1458,15 @@
"node": "^10 || ^12 || >=14" "node": "^10 || ^12 || >=14"
} }
}, },
"node_modules/proxy-from-env": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-2.1.0.tgz",
"integrity": "sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA==",
"license": "MIT",
"engines": {
"node": ">=10"
}
},
"node_modules/rolldown": { "node_modules/rolldown": {
"version": "1.0.3", "version": "1.0.3",
"resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.3.tgz", "resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.3.tgz",
@@ -1284,6 +1639,21 @@
"resolved": "https://registry.npmjs.org/vue-component-type-helpers/-/vue-component-type-helpers-3.3.3.tgz", "resolved": "https://registry.npmjs.org/vue-component-type-helpers/-/vue-component-type-helpers-3.3.3.tgz",
"integrity": "sha512-x4nsFpy5Pe8fqPzp/5vkTPeTTDBpAx4WVtV47Ejt0+2FQrq4pRRsJs7JmYRqMFzTu/LW+pCWEjQ3YVCkPV7f9g==", "integrity": "sha512-x4nsFpy5Pe8fqPzp/5vkTPeTTDBpAx4WVtV47Ejt0+2FQrq4pRRsJs7JmYRqMFzTu/LW+pCWEjQ3YVCkPV7f9g==",
"license": "MIT" "license": "MIT"
},
"node_modules/vue-router": {
"version": "4.6.4",
"resolved": "https://registry.npmjs.org/vue-router/-/vue-router-4.6.4.tgz",
"integrity": "sha512-Hz9q5sa33Yhduglwz6g9skT8OBPii+4bFn88w6J+J4MfEo4KRRpmiNG/hHHkdbRFlLBOqxN8y8gf2Fb0MTUgVg==",
"license": "MIT",
"dependencies": {
"@vue/devtools-api": "^6.6.4"
},
"funding": {
"url": "https://github.com/sponsors/posva"
},
"peerDependencies": {
"vue": "^3.5.0"
}
} }
} }
} }

View File

@@ -10,6 +10,7 @@
}, },
"dependencies": { "dependencies": {
"@element-plus/icons-vue": "^2.3.2", "@element-plus/icons-vue": "^2.3.2",
"axios": "^1.18.0",
"element-china-area-data": "^6.1.0", "element-china-area-data": "^6.1.0",
"element-plus": "^2.14.1", "element-plus": "^2.14.1",
"vue": "^3.5.34", "vue": "^3.5.34",

22
src/api/afterSales.js Normal file
View File

@@ -0,0 +1,22 @@
// api/afterSales.js —— 售后管理接口
import request from '../utils/request'
export function getAfterSalesList(params) {
return request.get('/after-sales', {params})
}
export function getAfterSalesDetail(id) {
return request.get(`/after-sales/${id}`)
}
export function createAfterSales(data) {
return request.post('/after-sales', data)
}
export function updateAfterSales(id, data) {
return request.put(`/after-sales/${id}`, data)
}
export function deleteAfterSales(id) {
return request.delete(`/after-sales/${id}`)
}

22
src/api/contract.js Normal file
View File

@@ -0,0 +1,22 @@
// api/contract.js —— 合同管理接口
import request from '../utils/request'
export function getContractList(params) {
return request.get('/contracts', {params})
}
export function getContractDetail(id) {
return request.get(`/contracts/${id}`)
}
export function createContract(data) {
return request.post('/contracts', data)
}
export function updateContract(id, data) {
return request.put(`/contracts/${id}`, data)
}
export function deleteContract(id) {
return request.delete(`/contracts/${id}`)
}

22
src/api/customer.js Normal file
View File

@@ -0,0 +1,22 @@
// api/customer.js —— 客户管理接口
import request from '../utils/request'
export function getCustomerList(params) {
return request.get('/customers', {params})
}
export function getCustomerDetail(id) {
return request.get(`/customers/${id}`)
}
export function createCustomer(data) {
return request.post('/customers', data)
}
export function updateCustomer(id, data) {
return request.put(`/customers/${id}`, data)
}
export function deleteCustomer(id) {
return request.delete(`/customers/${id}`)
}

22
src/api/employee.js Normal file
View File

@@ -0,0 +1,22 @@
// api/employee.js —— 员工管理接口
import request from '../utils/request'
export function getEmployeeList(params) {
return request.get('/employees', {params})
}
export function getEmployeeDetail(id) {
return request.get(`/employees/${id}`)
}
export function createEmployee(data) {
return request.post('/employees', data)
}
export function updateEmployee(id, data) {
return request.put(`/employees/${id}`, data)
}
export function deleteEmployee(id) {
return request.delete(`/employees/${id}`)
}

22
src/api/product.js Normal file
View File

@@ -0,0 +1,22 @@
// api/product.js —— 产品管理接口
import request from '../utils/request'
export function getProductList(params) {
return request.get('/products', {params})
}
export function getProductDetail(id) {
return request.get(`/products/${id}`)
}
export function createProduct(data) {
return request.post('/products', data)
}
export function updateProduct(id, data) {
return request.put(`/products/${id}`, data)
}
export function deleteProduct(id) {
return request.delete(`/products/${id}`)
}

43
src/api/user.js Normal file
View File

@@ -0,0 +1,43 @@
// api/user.js —— 用户相关接口
import request from '../utils/request'
// 登录
export function login(data) {
return request.post('/user/login', data)
}
// 获取当前用户信息
export function getUserInfo() {
return request.get('/user/info')
}
// 登出
export function logout() {
return request.post('/user/logout')
}
// 修改密码
export function changePassword(data) {
return request.put('/user/password', data)
}
// ========== 用户管理 CRUD管理员 ==========
export function getUserList(params) {
return request.get('/users', {params})
}
export function getUserDetail(id) {
return request.get(`/users/${id}`)
}
export function createUser(data) {
return request.post('/users', data)
}
export function updateUser(id, data) {
return request.put(`/users/${id}`, data)
}
export function deleteUser(id) {
return request.delete(`/users/${id}`)
}

View File

@@ -67,6 +67,7 @@ import {reactive, ref} from 'vue'
import {ElMessage} from 'element-plus' import {ElMessage} from 'element-plus'
import {Lock, User} from '@element-plus/icons-vue' import {Lock, User} from '@element-plus/icons-vue'
import {useRouter} from 'vue-router' import {useRouter} from 'vue-router'
import {login} from '../api/user'
const router = useRouter() const router = useRouter()
@@ -93,11 +94,22 @@ const handleLogin = async () => {
try { try {
await loginFormRef.value.validate() await loginFormRef.value.validate()
loading.value = true loading.value = true
await new Promise((r) => setTimeout(r, 800)) const res = await login({
ElMessage.success('登录成功(模拟)') username: loginForm.username,
password: loginForm.password,
})
// 保存 token 和用户信息
localStorage.setItem('token', res.data.token)
localStorage.setItem('userInfo', JSON.stringify(res.data.userInfo))
if (loginForm.remember) {
localStorage.setItem('rememberUser', loginForm.username)
} else {
localStorage.removeItem('rememberUser')
}
ElMessage.success('登录成功')
router.push('/panel') router.push('/panel')
} catch { } catch {
// 校验失败element-plus 会自动显示红色提示 // 校验失败或 API 错误,拦截器已弹提示
} finally { } finally {
loading.value = false loading.value = false
} }

View File

@@ -1,196 +1,277 @@
<script setup>
import { ref,computed } from 'vue'
import { ElMessage } from 'element-plus'
const searchKeyword = ref('')
const selectField = ref('1')
const dateRange = ref([])
const loading=ref(false)
//过滤搜索结果
const filteredContracts=computed(()=>{
})
//处理搜索
const handleSearch=()=>{
}
//日期变更处理
const handleDateChange=()=>{
handleSearch();
}
// 重置搜索
const resetSearch = () => {
searchKeyword.value = ''
searchField.value = '1'
dateRange.value = []
handleSearch()
}
// 日期快捷选项
const dateShortcuts = [
{
text: '最近一周',
value: () => {
const end = new Date()
const start = new Date()
start.setTime(start.getTime() - 7 * 24 * 3600 * 1000)
return [start, end]
}
},
{
text: '最近一个月',
value: () => {
const end = new Date()
const start = new Date()
start.setTime(start.getTime() - 30 * 24 * 3600 * 1000)
return [start, end]
}
},
{
text: '最近三个月',
value: () => {
const end = new Date()
const start = new Date()
start.setTime(start.getTime() - 90 * 24 * 3600 * 1000)
return [start, end]
}
}
]
// 格式化日期
const formatDate = (date) => {
if (!date) return ''
return date
}
</script>
<template> <template>
<div class="contract-container"> <div class="page-container">
<!-- 搜索栏区域 --> <!-- 搜索栏 -->
<div class="contract-search"> <div class="search-bar">
<div class="search-row"> <el-input v-model="search.contract_no" placeholder="合同编号" clearable style="width: 180px" @keyup.enter="handleSearch" />
<el-input v-model="searchKeyword" style="max-width: 600px" placeholder="Please input" <el-input v-model="search.customer_name" placeholder="客户名称" clearable style="width: 180px" @keyup.enter="handleSearch" />
class="contract-search-with-select"> <el-input v-model="search.employee_name" placeholder="业务员" clearable style="width: 180px" @keyup.enter="handleSearch" />
<template #prepend> <el-select v-model="search.status" placeholder="合同状态" clearable style="width: 130px">
<el-select v-model="searchField" placeholder="Select" style="width: 115px"> <el-option label="草稿" value="草稿" />
<el-option label="编号" value="1" /> <el-option label="生效" value="生效" />
<el-option label="名称" value="2" /> <el-option label="完成" value="完成" />
<el-option label="类型" value="3" /> <el-option label="作废" value="作废" />
</el-select> </el-select>
</template> <el-button type="primary" :icon="Search" @click="handleSearch">搜索</el-button>
<template #append> <el-button @click="resetSearch">重置</el-button>
<el-button :icon="Search" /> <el-button color="#E60012" @click="openAdd">新增合同</el-button>
</template>
</el-input>
</div> </div>
<div class="search-row date-range">
<span class="date-label">日期范围</span> <!-- 表格 -->
<el-date-picker <el-table :data="tableData" border v-loading="loading" stripe>
v-model="dateRange" <el-table-column prop="id" label="ID" width="60" />
type="daterange" <el-table-column prop="contract_no" label="合同编号" width="130" />
range-separator="" <el-table-column prop="contract_name" label="合同名称" min-width="150" show-overflow-tooltip />
start-placeholder="开始日期" <el-table-column prop="customer_name" label="客户" width="100" />
end-placeholder="结束日期" <el-table-column prop="employee_name" label="业务员" width="100" />
:shortcuts="dateShortcuts" <el-table-column prop="amount" label="金额" width="100">
@change="handleDateChange" <template #default="{ row }">{{ row.amount != null ? '¥' + row.amount : '-' }}</template>
</el-table-column>
<el-table-column prop="effective_date" label="生效日期" width="110">
<template #default="{ row }">{{ row.effective_date?.slice(0, 10) || '-' }}</template>
</el-table-column>
<el-table-column prop="expiry_date" label="到期日期" width="110">
<template #default="{ row }">{{ row.expiry_date?.slice(0, 10) || '-' }}</template>
</el-table-column>
<el-table-column prop="status" label="状态" width="80">
<template #default="{ row }">
<el-tag :type="statusType(row.status)">{{ row.status }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="remark" label="备注" width="120" show-overflow-tooltip />
<el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }">
<el-button link type="primary" @click="openEdit(row)">编辑</el-button>
<el-button link type="danger" @click="handleDelete(row.id)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<div class="pagination-bar">
<el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchList"
@current-change="fetchList"
/> />
</div> </div>
<el-button type="primary" @click="addCustomer" style="margin-left: 20px"> <!-- 新增/编辑弹窗 -->
新增合同 <el-dialog v-model="dialogVisible" :title="isEdit ? '编辑合同' : '新增合同'" width="650px" destroy-on-close>
</el-button> <el-form ref="formRef" :model="form" :rules="rules" label-width="90px" label-position="top">
<el-button type="default" @click="resetSearch" style="margin-left:20px"> <el-row :gutter="16">
重置 <el-col :span="12">
</el-button> <el-form-item label="合同名称" prop="contract_name">
</div> <el-input v-model="form.contract_name" placeholder="请输入合同名称" />
<!-- 合同列表 -->
<div class="contract-list">
<div v-if="loading" class="loading-container">
<el-skeleton :rows="5" animated />
</div>
<div v-else-if="filteredContracts.length===0" class="empty-container">
<el-empty description="暂无合同数据" />
</div>
<div v-else>
<div class="list-header">
<span class="total-count">共找到{{ filteredContracts.length }}条合同</span>
</div>
<div class="contract-cards">
<el-card
v-for="contract in filteredContracts"
:key="contract.id"
class="contract-card"
shadow="hover"
@click="viewDetail"
>
<div class="contract-info">
<div class="contract-no">
<el-tag size="small" type="primary">编号</el-tag>
<span class="no-value">{{ contract.contractNo }}</span>
</div>
<div class="contract-name">
<el-icon><Document /></el-icon>
<span class="name-value">{{ contract.contractName }}</span>
</div>
<div class="contract-date">
<el-icon><Calendar /></el-icon>
<span class="date-value">{{ formatDate(contract.signDate) }}</span>
</div>
</div>
</el-card>
</div>
</div>
</div>
<!-- 新增客户表单 -->
<div v-if="showAddForm" class="customer-info-label">
<el-form :model="form" label-width="auto" style="max-width: 600px" label-position="top">
<el-form-item label="姓名">
<el-input v-model="form.name" placeholder="请输入姓名" />
</el-form-item> </el-form-item>
</el-col>
<el-form-item label="电话"> <el-col :span="12">
<el-input v-model="form.phone" :controls="false" :min="0" :max="99999999999" :precision="0" <el-form-item label="合同编号">
placeholder="请输入11位手机号" style="width: 100%" /> <el-input v-model="form.contract_no" placeholder="请输入编号" />
</el-form-item> </el-form-item>
</el-col>
<el-form-item label="地区"> </el-row>
<el-cascader v-model="form.region" :options="regionData" :props="{ expandTrigger: 'hover' }" <el-row :gutter="16">
placeholder="请选择省/市/区" clearable style="width: 100%" /> <el-col :span="12">
<el-form-item label="关联客户" prop="customer_id">
<el-select v-model="form.customer_id" filterable placeholder="选择客户" style="width:100%">
<el-option v-for="c in customerOptions" :key="c.id" :label="c.name" :value="c.id" />
</el-select>
</el-form-item> </el-form-item>
</el-col>
<el-form-item label="详细地址"> <el-col :span="12">
<el-input v-model="form.address" placeholder=" 请输入详细地址" /> <el-form-item label="业务员">
<el-select v-model="form.employee_id" filterable placeholder="选择业务员" clearable style="width:100%">
<el-option v-for="e in employeeOptions" :key="e.id" :label="e.name" :value="e.id" />
</el-select>
</el-form-item> </el-form-item>
</el-col>
<el-form-item label="电子邮箱"> </el-row>
<el-input v-model="form.email" placeholder=" 请输入邮箱地址" /> <el-row :gutter="16">
<el-col :span="8">
<el-form-item label="金额">
<el-input-number v-model="form.amount" :min="0" :precision="2" controls-position="right" style="width:100%" />
</el-form-item> </el-form-item>
</el-col>
<el-form-item label="代理商类型"> <el-col :span="8">
<el-radio-group v-model="form.customer_type"> <el-form-item label="生效日期">
<el-radio value="VIP">地区总代理</el-radio> <el-date-picker v-model="form.effective_date" type="date" value-format="YYYY-MM-DD" placeholder="选择日期" style="width:100%" />
<el-radio value="Normal">普通代理</el-radio>
</el-radio-group>
</el-form-item> </el-form-item>
<el-form-item> </el-col>
<el-button type="primary" @click="saveCustomer">保存</el-button> <el-col :span="8">
<el-button type="danger" @click="cancelAdd">取消</el-button> <el-form-item label="到期日期">
<el-date-picker v-model="form.expiry_date" type="date" value-format="YYYY-MM-DD" placeholder="选择日期" style="width:100%" />
</el-form-item>
</el-col>
</el-row>
<el-form-item label="合同状态">
<el-select v-model="form.status" style="width: 200px">
<el-option label="草稿" value="草稿" />
<el-option label="生效" value="生效" />
<el-option label="完成" value="完成" />
<el-option label="作废" value="作废" />
</el-select>
</el-form-item>
<el-form-item label="合同内容">
<el-input v-model="form.contract_content" type="textarea" :rows="3" placeholder="合同条款/内容" />
</el-form-item>
<el-form-item label="备注">
<el-input v-model="form.remark" type="textarea" :rows="2" placeholder="备注信息" />
</el-form-item> </el-form-item>
</el-form> </el-form>
<template #footer>
</div> <el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" :loading="submitLoading" @click="handleSubmit">确定</el-button>
</template>
</el-dialog>
</div> </div>
</template> </template>
<style scoped> <script setup>
import {ref, reactive, onMounted} from 'vue'
import {Search} from '@element-plus/icons-vue'
import {ElMessage, ElMessageBox} from 'element-plus'
import {getContractList, createContract, updateContract, deleteContract} from '../api/contract'
import {getCustomerList} from '../api/customer'
import {getEmployeeList} from '../api/employee'
const search = reactive({contract_no: '', customer_name: '', employee_name: '', status: ''})
const loading = ref(false)
const tableData = ref([])
const pagination = reactive({page: 1, pageSize: 10, total: 0})
const dialogVisible = ref(false)
const isEdit = ref(false)
const editId = ref(null)
const submitLoading = ref(false)
const formRef = ref(null)
// 下拉选项
const customerOptions = ref([])
const employeeOptions = ref([])
const form = reactive({
contract_name: '', contract_no: '', customer_id: null, employee_id: null,
amount: null, effective_date: '', expiry_date: '', status: '生效',
contract_content: '', remark: '',
})
const rules = {
contract_name: [{required: true, message: '请输入合同名称', trigger: 'blur'}],
customer_id: [{required: true, message: '请选择客户', trigger: 'change'}],
}
const statusType = (s) => ({'草稿': 'info', '生效': 'success', '完成': '', '作废': 'danger'}[s] || 'info')
const fetchList = async () => {
loading.value = true
try {
const params = {page: pagination.page, pageSize: pagination.pageSize, ...search}
Object.keys(params).forEach(k => { if (!params[k]) delete params[k] })
const res = await getContractList(params)
tableData.value = res.data.list
pagination.total = res.data.total
} catch {} finally { loading.value = false }
}
// 加载下拉选项
const loadOptions = async () => {
try {
const [cRes, eRes] = await Promise.all([
getCustomerList({pageSize: 100}),
getEmployeeList({pageSize: 100}),
])
customerOptions.value = cRes.data.list
employeeOptions.value = eRes.data.list
} catch {}
}
const handleSearch = () => { pagination.page = 1; fetchList() }
const resetSearch = () => {
Object.assign(search, {contract_no: '', customer_name: '', employee_name: '', status: ''})
handleSearch()
}
const openAdd = () => {
isEdit.value = false
editId.value = null
Object.assign(form, {
contract_name: '', contract_no: '', customer_id: null, employee_id: null,
amount: null, effective_date: '', expiry_date: '', status: '生效',
contract_content: '', remark: '',
})
dialogVisible.value = true
}
const openEdit = (row) => {
isEdit.value = true
editId.value = row.id
Object.assign(form, {
contract_name: row.contract_name || '',
contract_no: row.contract_no || '',
customer_id: row.customer_id,
employee_id: row.employee_id,
amount: row.amount,
effective_date: row.effective_date?.slice(0, 10) || '',
expiry_date: row.expiry_date?.slice(0, 10) || '',
status: row.status || '生效',
contract_content: row.contract_content || '',
remark: row.remark || '',
})
dialogVisible.value = true
}
const handleSubmit = async () => {
if (!formRef.value) return
await formRef.value.validate()
submitLoading.value = true
try {
const payload = {...form}
if (!payload.employee_id) payload.employee_id = null
if (!payload.amount) payload.amount = null
if (isEdit.value) {
await updateContract(editId.value, payload)
ElMessage.success('更新成功')
} else {
await createContract(payload)
ElMessage.success('新增成功')
}
dialogVisible.value = false
fetchList()
} catch {} finally { submitLoading.value = false }
}
const handleDelete = async (id) => {
try {
await ElMessageBox.confirm('确定删除该合同?', '提示', {type: 'warning'})
await deleteContract(id)
ElMessage.success('删除成功')
fetchList()
} catch {}
}
onMounted(() => { fetchList(); loadOptions() })
</script>
<style scoped>
.page-container {
background: #fff;
padding: 20px;
border-radius: 8px;
}
.search-bar {
display: flex;
gap: 10px;
margin-bottom: 16px;
flex-wrap: wrap;
align-items: center;
}
.pagination-bar {
margin-top: 16px;
display: flex;
justify-content: flex-end;
}
</style> </style>

View File

@@ -1,149 +1,268 @@
<script setup>
import { Search } from '@element-plus/icons-vue'
import { ref,reactive } from 'vue'
import { regionData } from 'element-china-area-data'
import { ElMessage } from 'element-plus'
const form = reactive({
name: '', // 姓名,默认为空字符串
phone: '', // 电话,默认为空字符串
region: [], // 地区,默认为空数组(省市区三级代码)
address: '', // 详细地址,默认为空字符串
email: '', // 电子邮箱,默认为空字符串
customer_type: 'Normal' // 客户类型,默认选中"普通客户"
})
const search = ref('')
const select = ref('1')
const searchResults=ref([])
const showSearchResults=ref(false) //是否显示搜索结果
const showAddForm=ref(false) //是否显示新增表单
const isEditMode = ref(false) //是否为编辑模式
const currentCustomerId = ref(null) //当前编辑的客户的ID
// 清空表单
const resetForm = () => {
form.name = ''
form.phone = ''
form.region = []
form.address = ''
form.email = ''
form.customer_type = 'Normal'
isEditMode.value = false
currentCustomerId.value = null
}
// 显示新增表单
const addCustomer = () => {
resetForm()
showAddForm.value = true
showSearchResults.value = false
}
// 清空搜索
const clearSearch = () => {
search.value = ''
showSearchResults.value = false
showAddForm.value = false
}
</script>
<template> <template>
<div class="customer-container"> <div class="page-container">
<!-- 搜索栏区域 --> <!-- 搜索栏 -->
<div class="customer-search"> <div class="search-bar">
<el-input v-model="search" style="max-width: 600px" placeholder="Please input" <el-input
class="customer-search-with-select"> v-model="search.name"
<template #prepend> placeholder="搜索客户姓名"
<el-select v-model="select" placeholder="Select" style="width: 115px"> clearable
<el-option label="姓名" value="1" /> style="width: 200px"
<el-option label="编号" value="2" /> @keyup.enter="handleSearch"
<el-option label="电话" value="3" /> />
<el-option label="邮箱" value="4" /> <el-input
v-model="search.phone"
placeholder="搜索电话"
clearable
style="width: 200px"
@keyup.enter="handleSearch"
/>
<el-select v-model="search.customer_type" placeholder="客户类型" clearable style="width: 150px">
<el-option label="地区总代理" value="VIP" />
<el-option label="普通代理" value="Normal" />
</el-select> </el-select>
</template> <el-button type="primary" :icon="Search" @click="handleSearch">搜索</el-button>
<template #append> <el-button @click="resetSearch">重置</el-button>
<el-button :icon="Search" /> <el-button color="#E60012" @click="openAdd">新增客户</el-button>
</template>
</el-input>
<el-button color="#E60012" :dark="isDark" @click="addCustomer" style="margin-left: 20px">
新增用户
</el-button>
</div> </div>
<div v-if="showSearchResults" class="customer-table"> <!-- 表格 -->
<el-table :data="searchResults" border style="width: 100%"> <el-table :data="tableData" border v-loading="loading" stripe>
<el-table-column prop="name" label="姓名" width="120" /> <el-table-column prop="id" label="ID" width="60" />
<el-table-column prop="phone" label="电话" width="150" /> <el-table-column prop="name" label="姓名" width="100" />
<el-table-column prop="regionText" label="地区" width="200" /> <el-table-column prop="phone" label="电话" width="130" />
<el-table-column prop="address" label="详细地址" width="200" /> <el-table-column label="地区" width="200">
<el-table-column prop="email" label="电子邮箱" width="200" /> <template #default="{ row }">
<el-table-column prop="customer_type" label="代理商类型" width="120"> {{ [row.province, row.city, row.district].filter(Boolean).join(' / ') || '-' }}
</template>
</el-table-column>
<el-table-column prop="address" label="详细地址" min-width="150" show-overflow-tooltip />
<el-table-column prop="email" label="邮箱" width="160" show-overflow-tooltip />
<el-table-column prop="customer_type" label="类型" width="100">
<template #default="{ row }"> <template #default="{ row }">
<el-tag :type="row.customer_type === 'VIP' ? 'danger' : 'info'"> <el-tag :type="row.customer_type === 'VIP' ? 'danger' : 'info'">
{{ row.customer_type === 'VIP' ? '地区总代理' : '普通代理' }} {{ row.customer_type === 'VIP' ? '总代理' : '普通代理' }}
</el-tag> </el-tag>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column prop="remark" label="备注" width="120" show-overflow-tooltip />
<el-table-column label="操作" width="150" fixed="right"> <el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }"> <template #default="{ row }">
<el-button link type="primary" @click="editCustomer(row)">编辑</el-button> <el-button link type="primary" @click="openEdit(row)">编辑</el-button>
<el-button link type="danger" @click="deleteCustomer(row.id)">删除</el-button> <el-button link type="danger" @click="handleDelete(row.id)">删除</el-button>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<div style="margin-top: 10px; color: #909399; font-size: 14px;">
找到 {{ searchResults.length }} 条结果 <!-- 分页 -->
<el-button link type="primary" @click="clearSearch">清空搜索</el-button> <div class="pagination-bar">
</div> <el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchList"
@current-change="fetchList"
/>
</div> </div>
<!-- 新增客户表单 --> <!-- 新增/编辑弹窗 -->
<div v-if="showAddForm" class="customer-info-label"> <el-dialog
<el-form :model="form" label-width="auto" style="max-width: 600px" label-position="top"> v-model="dialogVisible"
<el-form-item label="姓名"> :title="isEdit ? '编辑客户' : '新增客户'"
width="600px"
destroy-on-close
>
<el-form ref="formRef" :model="form" :rules="rules" label-width="80px" label-position="top">
<el-form-item label="姓名" prop="name">
<el-input v-model="form.name" placeholder="请输入姓名" /> <el-input v-model="form.name" placeholder="请输入姓名" />
</el-form-item> </el-form-item>
<el-form-item label="电话"> <el-form-item label="电话">
<el-input v-model="form.phone" :controls="false" :min="0" :max="99999999999" :precision="0" <el-input v-model="form.phone" placeholder="请输入电话" />
placeholder="请输入11位手机号" style="width: 100%" />
</el-form-item> </el-form-item>
<el-form-item label="地区"> <el-form-item label="地区">
<el-cascader v-model="form.region" :options="regionData" :props="{ expandTrigger: 'hover' }" <el-cascader
placeholder="请选择省/市/区" clearable style="width: 100%" /> v-model="form.region"
:options="regionData"
:props="{ expandTrigger: 'hover' }"
placeholder="请选择省/市/区"
clearable
style="width: 100%"
/>
</el-form-item> </el-form-item>
<el-form-item label="详细地址"> <el-form-item label="详细地址">
<el-input v-model="form.address" placeholder="请输入详细地址" /> <el-input v-model="form.address" placeholder="请输入详细地址" />
</el-form-item> </el-form-item>
<el-form-item label="邮箱">
<el-form-item label="电子邮箱"> <el-input v-model="form.email" placeholder="请输入邮箱" />
<el-input v-model="form.email" placeholder=" 请输入邮箱地址" />
</el-form-item> </el-form-item>
<el-form-item label="客户类型">
<el-form-item label="代理商类型">
<el-radio-group v-model="form.customer_type"> <el-radio-group v-model="form.customer_type">
<el-radio value="VIP">地区总代理</el-radio> <el-radio value="VIP">地区总代理</el-radio>
<el-radio value="Normal">普通代理</el-radio> <el-radio value="Normal">普通代理</el-radio>
</el-radio-group> </el-radio-group>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item label="备注">
<el-button type="primary" @click="saveCustomer">保存</el-button> <el-input v-model="form.remark" type="textarea" :rows="2" placeholder="备注信息" />
<el-button type="danger" @click="cancelAdd">取消</el-button>
</el-form-item> </el-form-item>
</el-form> </el-form>
<template #footer>
</div> <el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" :loading="submitLoading" @click="handleSubmit">确定</el-button>
</template>
</el-dialog>
</div> </div>
</template> </template>
<style scoped> <script setup>
import {ref, reactive, onMounted} from 'vue'
import {Search} from '@element-plus/icons-vue'
import {ElMessage, ElMessageBox} from 'element-plus'
import {regionData} from 'element-china-area-data'
import {getCustomerList, createCustomer, updateCustomer, deleteCustomer} from '../api/customer'
// 搜索
const search = reactive({name: '', phone: '', customer_type: ''})
const loading = ref(false)
const tableData = ref([])
const pagination = reactive({page: 1, pageSize: 10, total: 0})
// 弹窗
const dialogVisible = ref(false)
const isEdit = ref(false)
const editId = ref(null)
const submitLoading = ref(false)
const formRef = ref(null)
const form = reactive({
name: '', phone: '', region: [], address: '', email: '', customer_type: 'Normal', remark: '',
})
const rules = {
name: [{required: true, message: '请输入客户姓名', trigger: 'blur'}],
}
// 获取列表
const fetchList = async () => {
loading.value = true
try {
const params = {
page: pagination.page,
pageSize: pagination.pageSize,
...search,
}
// 清空空值
Object.keys(params).forEach(k => { if (!params[k]) delete params[k] })
const res = await getCustomerList(params)
tableData.value = res.data.list
pagination.total = res.data.total
} catch {} finally {
loading.value = false
}
}
const handleSearch = () => {
pagination.page = 1
fetchList()
}
const resetSearch = () => {
search.name = ''
search.phone = ''
search.customer_type = ''
handleSearch()
}
// 打开新增
const openAdd = () => {
isEdit.value = false
editId.value = null
Object.assign(form, {name: '', phone: '', region: [], address: '', email: '', customer_type: 'Normal', remark: ''})
dialogVisible.value = true
}
// 打开编辑
const openEdit = (row) => {
isEdit.value = true
editId.value = row.id
Object.assign(form, {
name: row.name || '',
phone: row.phone || '',
region: [],
address: row.address || '',
email: row.email || '',
customer_type: row.customer_type || 'Normal',
remark: row.remark || '',
})
dialogVisible.value = true
}
// 提交
const handleSubmit = async () => {
if (!formRef.value) return
await formRef.value.validate()
submitLoading.value = true
try {
// 从 region cascader 拆出省市区
const [province, city, district] = form.region
const payload = {
name: form.name,
phone: form.phone || null,
province: province || null,
city: city || null,
district: district || null,
address: form.address || null,
email: form.email || null,
customer_type: form.customer_type,
remark: form.remark || null,
}
if (isEdit.value) {
await updateCustomer(editId.value, payload)
ElMessage.success('更新成功')
} else {
await createCustomer(payload)
ElMessage.success('新增成功')
}
dialogVisible.value = false
fetchList()
} catch {} finally {
submitLoading.value = false
}
}
// 删除
const handleDelete = async (id) => {
try {
await ElMessageBox.confirm('确定删除该客户?', '提示', {type: 'warning'})
await deleteCustomer(id)
ElMessage.success('删除成功')
fetchList()
} catch {}
}
onMounted(() => {
fetchList()
})
</script>
<style scoped>
.page-container {
background: #fff;
padding: 20px;
border-radius: 8px;
}
.search-bar {
display: flex;
gap: 10px;
margin-bottom: 16px;
flex-wrap: wrap;
align-items: center;
}
.pagination-bar {
margin-top: 16px;
display: flex;
justify-content: flex-end;
}
</style> </style>

266
src/components/employee.vue Normal file
View File

@@ -0,0 +1,266 @@
<template>
<div class="page-container">
<!-- 搜索栏 -->
<div class="search-bar">
<el-input v-model="search.name" placeholder="员工姓名" clearable style="width: 200px" @keyup.enter="handleSearch" />
<el-input v-model="search.department" placeholder="部门" clearable style="width: 180px" @keyup.enter="handleSearch" />
<el-select v-model="search.status" placeholder="在职状态" clearable style="width: 130px">
<el-option label="在职" :value="1" />
<el-option label="离职" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="handleSearch">搜索</el-button>
<el-button @click="resetSearch">重置</el-button>
<el-button color="#E60012" @click="openAdd">新增员工</el-button>
</div>
<!-- 表格 -->
<el-table :data="tableData" border v-loading="loading" stripe>
<el-table-column prop="id" label="ID" width="60" />
<el-table-column prop="name" label="姓名" width="90" />
<el-table-column prop="gender" label="性别" width="60" />
<el-table-column prop="age" label="年龄" width="60" />
<el-table-column prop="education" label="学历" width="80" />
<el-table-column prop="department" label="部门" width="100" />
<el-table-column prop="position" label="职务" width="100" />
<el-table-column prop="entry_date" label="入职日期" width="110">
<template #default="{ row }">{{ row.entry_date?.slice(0, 10) || '-' }}</template>
</el-table-column>
<el-table-column prop="salary" label="工资" width="90">
<template #default="{ row }">{{ row.salary != null ? '¥' + row.salary : '-' }}</template>
</el-table-column>
<el-table-column prop="phone" label="电话" width="120" />
<el-table-column prop="email" label="邮箱" width="160" show-overflow-tooltip />
<el-table-column prop="status" label="状态" width="70">
<template #default="{ row }">
<el-tag :type="row.status === 1 ? 'success' : 'danger'">{{ row.status === 1 ? '在职' : '离职' }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="remark" label="备注" width="120" show-overflow-tooltip />
<el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }">
<el-button link type="primary" @click="openEdit(row)">编辑</el-button>
<el-button link type="danger" @click="handleDelete(row.id)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<div class="pagination-bar">
<el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchList"
@current-change="fetchList"
/>
</div>
<!-- 新增/编辑弹窗 -->
<el-dialog v-model="dialogVisible" :title="isEdit ? '编辑员工' : '新增员工'" width="650px" destroy-on-close>
<el-form ref="formRef" :model="form" :rules="rules" label-width="80px" label-position="top">
<el-row :gutter="16">
<el-col :span="8">
<el-form-item label="姓名" prop="name">
<el-input v-model="form.name" placeholder="姓名" />
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="性别">
<el-select v-model="form.gender" placeholder="选择" style="width:100%">
<el-option label="男" value="男" />
<el-option label="女" value="女" />
</el-select>
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="年龄">
<el-input-number v-model="form.age" :min="18" :max="70" controls-position="right" style="width:100%" />
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="16">
<el-col :span="8">
<el-form-item label="学历">
<el-select v-model="form.education" placeholder="选择" clearable style="width:100%">
<el-option label="高中" value="高中" />
<el-option label="专科" value="专科" />
<el-option label="本科" value="本科" />
<el-option label="硕士" value="硕士" />
<el-option label="博士" value="博士" />
</el-select>
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="部门">
<el-input v-model="form.department" placeholder="所属部门" />
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="职务">
<el-input v-model="form.position" placeholder="职务/岗位" />
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="16">
<el-col :span="8">
<el-form-item label="入职日期">
<el-date-picker v-model="form.entry_date" type="date" value-format="YYYY-MM-DD" placeholder="选择日期" style="width:100%" />
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="工资">
<el-input-number v-model="form.salary" :min="0" :precision="2" controls-position="right" style="width:100%" />
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="在职状态">
<el-select v-model="form.status" style="width:100%">
<el-option label="在职" :value="1" />
<el-option label="离职" :value="0" />
</el-select>
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="电话">
<el-input v-model="form.phone" placeholder="联系电话" />
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="邮箱">
<el-input v-model="form.email" placeholder="电子邮箱" />
</el-form-item>
</el-col>
</el-row>
<el-form-item label="备注">
<el-input v-model="form.remark" type="textarea" :rows="2" placeholder="备注信息" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" :loading="submitLoading" @click="handleSubmit">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import {ref, reactive, onMounted} from 'vue'
import {Search} from '@element-plus/icons-vue'
import {ElMessage, ElMessageBox} from 'element-plus'
import {getEmployeeList, createEmployee, updateEmployee, deleteEmployee} from '../api/employee'
const search = reactive({name: '', department: '', status: ''})
const loading = ref(false)
const tableData = ref([])
const pagination = reactive({page: 1, pageSize: 10, total: 0})
const dialogVisible = ref(false)
const isEdit = ref(false)
const editId = ref(null)
const submitLoading = ref(false)
const formRef = ref(null)
const defaultForm = () => ({
name: '', gender: '男', age: null, education: '', department: '',
entry_date: '', position: '', salary: null, phone: '', email: '',
status: 1, remark: '',
})
const form = reactive(defaultForm())
const rules = {
name: [{required: true, message: '请输入员工姓名', trigger: 'blur'}],
}
const fetchList = async () => {
loading.value = true
try {
const params = {page: pagination.page, pageSize: pagination.pageSize, ...search}
Object.keys(params).forEach(k => { if (params[k] === '' || params[k] === null) delete params[k] })
const res = await getEmployeeList(params)
tableData.value = res.data.list
pagination.total = res.data.total
} catch {} finally { loading.value = false }
}
const handleSearch = () => { pagination.page = 1; fetchList() }
const resetSearch = () => {
Object.assign(search, {name: '', department: '', status: ''})
handleSearch()
}
const openAdd = () => {
isEdit.value = false
editId.value = null
Object.assign(form, defaultForm())
dialogVisible.value = true
}
const openEdit = (row) => {
isEdit.value = true
editId.value = row.id
Object.assign(form, {
name: row.name || '', gender: row.gender || '男', age: row.age,
education: row.education || '', department: row.department || '',
entry_date: row.entry_date?.slice(0, 10) || '', position: row.position || '',
salary: row.salary, phone: row.phone || '', email: row.email || '',
status: row.status ?? 1, remark: row.remark || '',
})
dialogVisible.value = true
}
const handleSubmit = async () => {
if (!formRef.value) return
await formRef.value.validate()
submitLoading.value = true
try {
const payload = {...form}
if (!payload.age) payload.age = null
if (!payload.entry_date) payload.entry_date = null
if (isEdit.value) {
await updateEmployee(editId.value, payload)
ElMessage.success('更新成功')
} else {
await createEmployee(payload)
ElMessage.success('新增成功')
}
dialogVisible.value = false
fetchList()
} catch {} finally { submitLoading.value = false }
}
const handleDelete = async (id) => {
try {
await ElMessageBox.confirm('确定删除该员工?', '提示', {type: 'warning'})
await deleteEmployee(id)
ElMessage.success('删除成功')
fetchList()
} catch {}
}
onMounted(() => { fetchList() })
</script>
<style scoped>
.page-container {
background: #fff;
padding: 20px;
border-radius: 8px;
}
.search-bar {
display: flex;
gap: 10px;
margin-bottom: 16px;
flex-wrap: wrap;
align-items: center;
}
.pagination-bar {
margin-top: 16px;
display: flex;
justify-content: flex-end;
}
</style>

View File

@@ -1,13 +1,42 @@
<script setup> <script setup>
import {ref} from 'vue' import {ref, computed} from 'vue'
import {useRoute} from 'vue-router' import {useRoute, useRouter} from 'vue-router'
import {ElMessageBox} from 'element-plus'
const route = useRoute() const route = useRoute()
const router = useRouter()
const isCollapse = ref(false) const isCollapse = ref(false)
const switchFold = () => { const switchFold = () => {
isCollapse.value = !isCollapse.value isCollapse.value = !isCollapse.value
} }
// 获取当前用户信息
const userInfo = computed(() => {
try {
return JSON.parse(localStorage.getItem('userInfo') || '{}')
} catch {
return {}
}
})
const isAdmin = computed(() => userInfo.value.role === 'admin')
// 退出登录
const handleLogout = async () => {
try {
await ElMessageBox.confirm('确定退出登录?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
})
localStorage.removeItem('token')
localStorage.removeItem('userInfo')
router.push('/login')
} catch {
// 取消
}
}
</script> </script>
<template> <template>
@@ -26,11 +55,14 @@ const switchFold = () => {
<h1 class="logo-title">蜜雪冰城管理系统</h1> <h1 class="logo-title">蜜雪冰城管理系统</h1>
</el-menu-item> </el-menu-item>
<!-- 右侧退出 --> <!-- 右侧用户信息 + 退出 -->
<el-menu-item index="/login" class="logout-item"> <div class="header-right">
<span class="user-name">{{ userInfo.real_name || userInfo.username || '用户' }}</span>
<el-button type="danger" text @click="handleLogout">
<el-icon><SwitchButton /></el-icon> <el-icon><SwitchButton /></el-icon>
<template #title>退出登录</template> 退出登录
</el-menu-item> </el-button>
</div>
</el-menu> </el-menu>
</el-header> </el-header>
@@ -63,16 +95,22 @@ const switchFold = () => {
<template #title>售后管理</template> <template #title>售后管理</template>
</el-menu-item> </el-menu-item>
<el-menu-item index="/panel/page3"> <el-menu-item index="/panel/product">
<el-icon><IceTea /></el-icon> <el-icon><Goods /></el-icon>
<template #title>产品管理</template> <template #title>产品管理</template>
</el-menu-item> </el-menu-item>
<el-menu-item index="/panel/page3"> <el-menu-item index="/panel/employee">
<el-icon><User /></el-icon> <el-icon><UserFilled /></el-icon>
<template #title>员工管理</template> <template #title>员工管理</template>
</el-menu-item> </el-menu-item>
<!-- 用户管理仅管理员可见 -->
<el-menu-item v-if="isAdmin" index="/panel/user">
<el-icon><User /></el-icon>
<template #title>用户管理</template>
</el-menu-item>
<!-- 底部收缩按钮 --> <!-- 底部收缩按钮 -->
<el-menu-item index="" class="collapse-btn" @click="switchFold"> <el-menu-item index="" class="collapse-btn" @click="switchFold">
<el-icon :class="{'rotate-180-animation':!isCollapse,'rotate-180-animation-reverse':isCollapse}"> <el-icon :class="{'rotate-180-animation':!isCollapse,'rotate-180-animation-reverse':isCollapse}">
@@ -106,6 +144,8 @@ const switchFold = () => {
.header-menu { .header-menu {
flex: 1; flex: 1;
border-bottom: none !important; border-bottom: none !important;
display: flex;
align-items: center;
} }
.logo-item { .logo-item {
@@ -128,13 +168,17 @@ const switchFold = () => {
white-space: nowrap; white-space: nowrap;
} }
.logout-item { .header-right {
margin-left: auto !important; margin-left: auto;
color: #909399; display: flex;
align-items: center;
gap: 16px;
padding-right: 20px;
} }
.logout-item:hover { .user-name {
color: #E60012 !important; font-size: 14px;
color: #606266;
} }
/* ========== 侧边栏 ========== */ /* ========== 侧边栏 ========== */

213
src/components/product.vue Normal file
View File

@@ -0,0 +1,213 @@
<template>
<div class="page-container">
<!-- 搜索栏 -->
<div class="search-bar">
<el-input v-model="search.name" placeholder="产品名称" clearable style="width: 200px" @keyup.enter="handleSearch" />
<el-input v-model="search.type" placeholder="产品类型" clearable style="width: 180px" @keyup.enter="handleSearch" />
<el-input v-model="search.supplier" placeholder="供应商" clearable style="width: 180px" @keyup.enter="handleSearch" />
<el-button type="primary" :icon="Search" @click="handleSearch">搜索</el-button>
<el-button @click="resetSearch">重置</el-button>
<el-button color="#E60012" @click="openAdd">新增产品</el-button>
</div>
<!-- 表格 -->
<el-table :data="tableData" border v-loading="loading" stripe>
<el-table-column prop="id" label="ID" width="60" />
<el-table-column prop="name" label="产品名称" min-width="150" show-overflow-tooltip />
<el-table-column prop="type" label="类型" width="100" />
<el-table-column prop="quantity" label="库存" width="80" />
<el-table-column prop="price" label="单价" width="90">
<template #default="{ row }">¥{{ row.price }}</template>
</el-table-column>
<el-table-column prop="unit" label="单位" width="60" />
<el-table-column prop="specification" label="规格" width="120" show-overflow-tooltip />
<el-table-column prop="supplier" label="供应商" width="130" show-overflow-tooltip />
<el-table-column prop="remark" label="备注" width="120" show-overflow-tooltip />
<el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }">
<el-button link type="primary" @click="openEdit(row)">编辑</el-button>
<el-button link type="danger" @click="handleDelete(row.id)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<div class="pagination-bar">
<el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchList"
@current-change="fetchList"
/>
</div>
<!-- 新增/编辑弹窗 -->
<el-dialog v-model="dialogVisible" :title="isEdit ? '编辑产品' : '新增产品'" width="600px" destroy-on-close>
<el-form ref="formRef" :model="form" :rules="rules" label-width="80px" label-position="top">
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="产品名称" prop="name">
<el-input v-model="form.name" placeholder="产品名称" />
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="类型">
<el-input v-model="form.type" placeholder="产品类型" />
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="16">
<el-col :span="8">
<el-form-item label="库存">
<el-input-number v-model="form.quantity" :min="0" controls-position="right" style="width:100%" />
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="单价">
<el-input-number v-model="form.price" :min="0" :precision="2" controls-position="right" style="width:100%" />
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item label="单位">
<el-input v-model="form.unit" placeholder="件" />
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="规格">
<el-input v-model="form.specification" placeholder="产品规格/型号" />
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="供应商">
<el-input v-model="form.supplier" placeholder="供应商名称" />
</el-form-item>
</el-col>
</el-row>
<el-form-item label="备注">
<el-input v-model="form.remark" type="textarea" :rows="2" placeholder="备注信息" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" :loading="submitLoading" @click="handleSubmit">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import {ref, reactive, onMounted} from 'vue'
import {Search} from '@element-plus/icons-vue'
import {ElMessage, ElMessageBox} from 'element-plus'
import {getProductList, createProduct, updateProduct, deleteProduct} from '../api/product'
const search = reactive({name: '', type: '', supplier: ''})
const loading = ref(false)
const tableData = ref([])
const pagination = reactive({page: 1, pageSize: 10, total: 0})
const dialogVisible = ref(false)
const isEdit = ref(false)
const editId = ref(null)
const submitLoading = ref(false)
const formRef = ref(null)
const defaultForm = () => ({
name: '', type: '', quantity: 0, price: 0, unit: '件',
specification: '', supplier: '', remark: '',
})
const form = reactive(defaultForm())
const rules = {
name: [{required: true, message: '请输入产品名称', trigger: 'blur'}],
}
const fetchList = async () => {
loading.value = true
try {
const params = {page: pagination.page, pageSize: pagination.pageSize, ...search}
Object.keys(params).forEach(k => { if (!params[k]) delete params[k] })
const res = await getProductList(params)
tableData.value = res.data.list
pagination.total = res.data.total
} catch {} finally { loading.value = false }
}
const handleSearch = () => { pagination.page = 1; fetchList() }
const resetSearch = () => {
Object.assign(search, {name: '', type: '', supplier: ''})
handleSearch()
}
const openAdd = () => {
isEdit.value = false
editId.value = null
Object.assign(form, defaultForm())
dialogVisible.value = true
}
const openEdit = (row) => {
isEdit.value = true
editId.value = row.id
Object.assign(form, {
name: row.name || '', type: row.type || '', quantity: row.quantity ?? 0,
price: row.price ?? 0, unit: row.unit || '件', specification: row.specification || '',
supplier: row.supplier || '', remark: row.remark || '',
})
dialogVisible.value = true
}
const handleSubmit = async () => {
if (!formRef.value) return
await formRef.value.validate()
submitLoading.value = true
try {
if (isEdit.value) {
await updateProduct(editId.value, {...form})
ElMessage.success('更新成功')
} else {
await createProduct({...form})
ElMessage.success('新增成功')
}
dialogVisible.value = false
fetchList()
} catch {} finally { submitLoading.value = false }
}
const handleDelete = async (id) => {
try {
await ElMessageBox.confirm('确定删除该产品?', '提示', {type: 'warning'})
await deleteProduct(id)
ElMessage.success('删除成功')
fetchList()
} catch {}
}
onMounted(() => { fetchList() })
</script>
<style scoped>
.page-container {
background: #fff;
padding: 20px;
border-radius: 8px;
}
.search-bar {
display: flex;
gap: 10px;
margin-bottom: 16px;
flex-wrap: wrap;
align-items: center;
}
.pagination-bar {
margin-top: 16px;
display: flex;
justify-content: flex-end;
}
</style>

View File

@@ -1,11 +1,237 @@
<script setup>
</script>
<template> <template>
<h1>hello ,im 3</h1> <div class="page-container">
<!-- 搜索栏 -->
<div class="search-bar">
<el-input v-model="search.customer_name" placeholder="客户名称" clearable style="width: 200px" @keyup.enter="handleSearch" />
<el-select v-model="search.handle_status" placeholder="处理状态" clearable style="width: 150px">
<el-option label="待处理" value="待处理" />
<el-option label="处理中" value="处理中" />
<el-option label="已完成" value="已完成" />
</el-select>
<el-button type="primary" :icon="Search" @click="handleSearch">搜索</el-button>
<el-button @click="resetSearch">重置</el-button>
<el-button color="#E60012" @click="openAdd">新增售后</el-button>
</div>
<!-- 表格 -->
<el-table :data="tableData" border v-loading="loading" stripe>
<el-table-column prop="id" label="ID" width="60" />
<el-table-column prop="customer_name" label="客户" width="100" />
<el-table-column prop="feedback" label="反馈内容" min-width="200" show-overflow-tooltip />
<el-table-column prop="employee_name" label="处理人" width="100" />
<el-table-column prop="handle_method" label="处理方式" width="150" show-overflow-tooltip />
<el-table-column prop="handle_status" label="状态" width="100">
<template #default="{ row }">
<el-tag :type="statusType(row.handle_status)">{{ row.handle_status }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="service_date" label="售后日期" width="110">
<template #default="{ row }">{{ row.service_date?.slice(0, 10) || '-' }}</template>
</el-table-column>
<el-table-column prop="remark" label="备注" width="120" show-overflow-tooltip />
<el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }">
<el-button link type="primary" @click="openEdit(row)">编辑</el-button>
<el-button link type="danger" @click="handleDelete(row.id)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<div class="pagination-bar">
<el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchList"
@current-change="fetchList"
/>
</div>
<!-- 新增/编辑弹窗 -->
<el-dialog v-model="dialogVisible" :title="isEdit ? '编辑售后' : '新增售后'" width="600px" destroy-on-close>
<el-form ref="formRef" :model="form" :rules="rules" label-width="90px" label-position="top">
<el-form-item label="客户" prop="customer_id">
<el-select v-model="form.customer_id" filterable placeholder="选择客户" style="width:100%">
<el-option v-for="c in customerOptions" :key="c.id" :label="c.name" :value="c.id" />
</el-select>
</el-form-item>
<el-form-item label="反馈内容" prop="feedback">
<el-input v-model="form.feedback" type="textarea" :rows="3" placeholder="客户反馈内容" />
</el-form-item>
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="处理人">
<el-select v-model="form.employee_id" filterable placeholder="选择处理人" clearable style="width:100%">
<el-option v-for="e in employeeOptions" :key="e.id" :label="e.name" :value="e.id" />
</el-select>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="处理状态">
<el-select v-model="form.handle_status" style="width:100%">
<el-option label="待处理" value="待处理" />
<el-option label="处理中" value="处理中" />
<el-option label="已完成" value="已完成" />
</el-select>
</el-form-item>
</el-col>
</el-row>
<el-form-item label="售后日期">
<el-date-picker v-model="form.service_date" type="date" value-format="YYYY-MM-DD" placeholder="选择日期" style="width:100%" />
</el-form-item>
<el-form-item label="处理方式">
<el-input v-model="form.handle_method" type="textarea" :rows="2" placeholder="处理方式/解决方案" />
</el-form-item>
<el-form-item label="备注">
<el-input v-model="form.remark" type="textarea" :rows="2" placeholder="备注信息" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" :loading="submitLoading" @click="handleSubmit">确定</el-button>
</template>
</el-dialog>
</div>
</template> </template>
<style scoped> <script setup>
import {ref, reactive, onMounted} from 'vue'
import {Search} from '@element-plus/icons-vue'
import {ElMessage, ElMessageBox} from 'element-plus'
import {getAfterSalesList, createAfterSales, updateAfterSales, deleteAfterSales} from '../api/afterSales'
import {getCustomerList} from '../api/customer'
import {getEmployeeList} from '../api/employee'
const search = reactive({customer_name: '', handle_status: ''})
const loading = ref(false)
const tableData = ref([])
const pagination = reactive({page: 1, pageSize: 10, total: 0})
const dialogVisible = ref(false)
const isEdit = ref(false)
const editId = ref(null)
const submitLoading = ref(false)
const formRef = ref(null)
const customerOptions = ref([])
const employeeOptions = ref([])
const form = reactive({
customer_id: null, feedback: '', employee_id: null,
handle_method: '', handle_status: '待处理', service_date: '', remark: '',
})
const rules = {
customer_id: [{required: true, message: '请选择客户', trigger: 'change'}],
feedback: [{required: true, message: '请输入反馈内容', trigger: 'blur'}],
}
const statusType = (s) => ({'待处理': 'warning', '处理中': 'info', '已完成': 'success'}[s] || 'info')
const fetchList = async () => {
loading.value = true
try {
const params = {page: pagination.page, pageSize: pagination.pageSize, ...search}
Object.keys(params).forEach(k => { if (!params[k]) delete params[k] })
const res = await getAfterSalesList(params)
tableData.value = res.data.list
pagination.total = res.data.total
} catch {} finally { loading.value = false }
}
const loadOptions = async () => {
try {
const [cRes, eRes] = await Promise.all([
getCustomerList({pageSize: 100}),
getEmployeeList({pageSize: 100}),
])
customerOptions.value = cRes.data.list
employeeOptions.value = eRes.data.list
} catch {}
}
const handleSearch = () => { pagination.page = 1; fetchList() }
const resetSearch = () => {
Object.assign(search, {customer_name: '', handle_status: ''})
handleSearch()
}
const openAdd = () => {
isEdit.value = false
editId.value = null
Object.assign(form, {
customer_id: null, feedback: '', employee_id: null,
handle_method: '', handle_status: '待处理', service_date: '', remark: '',
})
dialogVisible.value = true
}
const openEdit = (row) => {
isEdit.value = true
editId.value = row.id
Object.assign(form, {
customer_id: row.customer_id,
feedback: row.feedback || '',
employee_id: row.employee_id,
handle_method: row.handle_method || '',
handle_status: row.handle_status || '待处理',
service_date: row.service_date?.slice(0, 10) || '',
remark: row.remark || '',
})
dialogVisible.value = true
}
const handleSubmit = async () => {
if (!formRef.value) return
await formRef.value.validate()
submitLoading.value = true
try {
const payload = {...form}
if (!payload.employee_id) payload.employee_id = null
if (!payload.service_date) payload.service_date = null
if (isEdit.value) {
await updateAfterSales(editId.value, payload)
ElMessage.success('更新成功')
} else {
await createAfterSales(payload)
ElMessage.success('新增成功')
}
dialogVisible.value = false
fetchList()
} catch {} finally { submitLoading.value = false }
}
const handleDelete = async (id) => {
try {
await ElMessageBox.confirm('确定删除该售后记录?', '提示', {type: 'warning'})
await deleteAfterSales(id)
ElMessage.success('删除成功')
fetchList()
} catch {}
}
onMounted(() => { fetchList(); loadOptions() })
</script>
<style scoped>
.page-container {
background: #fff;
padding: 20px;
border-radius: 8px;
}
.search-bar {
display: flex;
gap: 10px;
margin-bottom: 16px;
flex-wrap: wrap;
align-items: center;
}
.pagination-bar {
margin-top: 16px;
display: flex;
justify-content: flex-end;
}
</style> </style>

216
src/components/user.vue Normal file
View File

@@ -0,0 +1,216 @@
<template>
<div class="page-container">
<!-- 搜索栏 -->
<div class="search-bar">
<el-input v-model="search.username" placeholder="用户名" clearable style="width: 200px" @keyup.enter="handleSearch" />
<el-select v-model="search.role" placeholder="角色" clearable style="width: 130px">
<el-option label="管理员" value="admin" />
<el-option label="普通用户" value="user" />
</el-select>
<el-select v-model="search.status" placeholder="状态" clearable style="width: 130px">
<el-option label="启用" :value="1" />
<el-option label="禁用" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="handleSearch">搜索</el-button>
<el-button @click="resetSearch">重置</el-button>
<el-button color="#E60012" @click="openAdd">新增用户</el-button>
</div>
<!-- 表格 -->
<el-table :data="tableData" border v-loading="loading" stripe>
<el-table-column prop="id" label="ID" width="60" />
<el-table-column prop="username" label="用户名" width="120" />
<el-table-column prop="real_name" label="真实姓名" width="120" />
<el-table-column prop="role" label="角色" width="100">
<template #default="{ row }">
<el-tag :type="row.role === 'admin' ? 'danger' : 'info'">
{{ row.role === 'admin' ? '管理员' : '普通用户' }}
</el-tag>
</template>
</el-table-column>
<el-table-column prop="status" label="状态" width="80">
<template #default="{ row }">
<el-tag :type="row.status === 1 ? 'success' : 'danger'">{{ row.status === 1 ? '启用' : '禁用' }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="created_at" label="创建时间" width="170">
<template #default="{ row }">{{ row.created_at?.slice(0, 19)?.replace('T', ' ') || '-' }}</template>
</el-table-column>
<el-table-column label="操作" width="200" fixed="right">
<template #default="{ row }">
<el-button link type="primary" @click="openEdit(row)">编辑</el-button>
<el-button link type="warning" @click="resetPwd(row.id)">重置密码</el-button>
<el-button link type="danger" @click="handleDelete(row.id)">删除</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<div class="pagination-bar">
<el-pagination
v-model:current-page="pagination.page"
v-model:page-size="pagination.pageSize"
:total="pagination.total"
:page-sizes="[10, 20, 50]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="fetchList"
@current-change="fetchList"
/>
</div>
<!-- 新增/编辑弹窗 -->
<el-dialog v-model="dialogVisible" :title="isEdit ? '编辑用户' : '新增用户'" width="500px" destroy-on-close>
<el-form ref="formRef" :model="form" :rules="rules" label-width="90px" label-position="top">
<el-form-item label="用户名" prop="username">
<el-input v-model="form.username" placeholder="3-50个字符" :disabled="isEdit" />
</el-form-item>
<el-form-item v-if="!isEdit" label="密码" prop="password">
<el-input v-model="form.password" type="password" show-password placeholder="至少6位" />
</el-form-item>
<el-form-item label="真实姓名">
<el-input v-model="form.real_name" placeholder="真实姓名" />
</el-form-item>
<el-row :gutter="16">
<el-col :span="12">
<el-form-item label="角色">
<el-select v-model="form.role" style="width:100%">
<el-option label="管理员" value="admin" />
<el-option label="普通用户" value="user" />
</el-select>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="状态">
<el-select v-model="form.status" style="width:100%">
<el-option label="启用" :value="1" />
<el-option label="禁用" :value="0" />
</el-select>
</el-form-item>
</el-col>
</el-row>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" :loading="submitLoading" @click="handleSubmit">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup>
import {ref, reactive, onMounted} from 'vue'
import {Search} from '@element-plus/icons-vue'
import {ElMessage, ElMessageBox} from 'element-plus'
import {getUserList, createUser, updateUser, deleteUser} from '../api/user'
const search = reactive({username: '', role: '', status: ''})
const loading = ref(false)
const tableData = ref([])
const pagination = reactive({page: 1, pageSize: 10, total: 0})
const dialogVisible = ref(false)
const isEdit = ref(false)
const editId = ref(null)
const submitLoading = ref(false)
const formRef = ref(null)
const form = reactive({
username: '', password: '', real_name: '', role: 'user', status: 1,
})
const rules = {
username: [{required: true, message: '请输入用户名', trigger: 'blur'}, {min: 3, max: 50, message: '3-50个字符', trigger: 'blur'}],
password: [{required: true, message: '请输入密码', trigger: 'blur'}, {min: 6, message: '至少6位', trigger: 'blur'}],
}
const fetchList = async () => {
loading.value = true
try {
const params = {page: pagination.page, pageSize: pagination.pageSize, ...search}
Object.keys(params).forEach(k => { if (params[k] === '' || params[k] === null) delete params[k] })
const res = await getUserList(params)
tableData.value = res.data.list
pagination.total = res.data.total
} catch {} finally { loading.value = false }
}
const handleSearch = () => { pagination.page = 1; fetchList() }
const resetSearch = () => {
Object.assign(search, {username: '', role: '', status: ''})
handleSearch()
}
const openAdd = () => {
isEdit.value = false
editId.value = null
Object.assign(form, {username: '', password: '', real_name: '', role: 'user', status: 1})
dialogVisible.value = true
}
const openEdit = (row) => {
isEdit.value = true
editId.value = row.id
Object.assign(form, {
username: row.username, password: '', real_name: row.real_name || '',
role: row.role, status: row.status,
})
dialogVisible.value = true
}
const handleSubmit = async () => {
if (!formRef.value) return
await formRef.value.validate()
submitLoading.value = true
try {
if (isEdit.value) {
const payload = {real_name: form.real_name, role: form.role, status: form.status}
await updateUser(editId.value, payload)
ElMessage.success('更新成功')
} else {
await createUser({username: form.username, password: form.password, real_name: form.real_name, role: form.role, status: form.status})
ElMessage.success('新增成功')
}
dialogVisible.value = false
fetchList()
} catch {} finally { submitLoading.value = false }
}
const resetPwd = async (id) => {
try {
await ElMessageBox.confirm('将密码重置为 123456确定', '重置密码', {type: 'warning'})
await updateUser(id, {password: '123456'})
ElMessage.success('密码已重置为 123456')
} catch {}
}
const handleDelete = async (id) => {
try {
await ElMessageBox.confirm('确定删除该用户?', '提示', {type: 'warning'})
await deleteUser(id)
ElMessage.success('删除成功')
fetchList()
} catch {}
}
onMounted(() => { fetchList() })
</script>
<style scoped>
.page-container {
background: #fff;
padding: 20px;
border-radius: 8px;
}
.search-bar {
display: flex;
gap: 10px;
margin-bottom: 16px;
flex-wrap: wrap;
align-items: center;
}
.pagination-bar {
margin-top: 16px;
display: flex;
justify-content: flex-end;
}
</style>

View File

@@ -5,6 +5,9 @@ import Panel from "./components/panel.vue";
import Customer from "./components/customer.vue"; import Customer from "./components/customer.vue";
import Contract from "./components/contract.vue"; import Contract from "./components/contract.vue";
import Service from "./components/service.vue"; import Service from "./components/service.vue";
import Employee from "./components/employee.vue";
import Product from "./components/product.vue";
import User from "./components/user.vue";
const routes = [ const routes = [
{ path: "/", redirect: "/login" }, { path: "/", redirect: "/login" },
@@ -13,11 +16,15 @@ const routes = [
path: "/panel", path: "/panel",
component: Panel, component: Panel,
redirect: "/panel/home", redirect: "/panel/home",
meta: { requiresAuth: true },
children: [ children: [
{ path: "home", component: Home }, { path: "home", component: Home },
{ path: "customer", component: Customer }, { path: "customer", component: Customer },
{ path: "contract", component: Contract }, { path: "contract", component: Contract },
{ path: "service", component: Service }, { path: "service", component: Service },
{ path: "employee", component: Employee },
{ path: "product", component: Product },
{ path: "user", component: User },
], ],
}, },
] ]
@@ -26,4 +33,17 @@ const router = createRouter({
history: createWebHistory(), history: createWebHistory(),
routes, routes,
}) })
// 路由守卫:未登录时跳转到登录页
router.beforeEach((to, from, next) => {
const token = localStorage.getItem('token')
if (to.path !== '/login' && !token) {
next('/login')
} else if (to.path === '/login' && token) {
next('/panel')
} else {
next()
}
})
export default router; export default router;

54
src/utils/request.js Normal file
View File

@@ -0,0 +1,54 @@
// utils/request.js —— axios 实例:自动带 token、统一错误处理
import axios from 'axios'
import {ElMessage} from 'element-plus'
import router from '../router'
const request = axios.create({
baseURL: '/api',
timeout: 10000,
})
// 请求拦截器:自动携带 token
request.interceptors.request.use(
(config) => {
const token = localStorage.getItem('token')
if (token) {
config.headers.Authorization = `Bearer ${token}`
}
return config
},
(error) => Promise.reject(error)
)
// 响应拦截器:统一错误提示
request.interceptors.response.use(
(res) => {
const {code, message} = res.data
// 后端 code === 0 或 code === 200 表示成功
if (code === 0 || code === 200) {
return res.data
}
ElMessage.error(message || '请求失败')
return Promise.reject(new Error(message))
},
(error) => {
if (error.response) {
const {status, data} = error.response
if (status === 401) {
ElMessage.error('登录已过期,请重新登录')
localStorage.removeItem('token')
localStorage.removeItem('userInfo')
router.push('/login')
} else if (status === 403) {
ElMessage.error(data?.message || '没有权限')
} else {
ElMessage.error(data?.message || '服务器错误')
}
} else {
ElMessage.error('网络异常,请检查连接')
}
return Promise.reject(error)
}
)
export default request

View File

@@ -4,4 +4,12 @@ import vue from '@vitejs/plugin-vue'
// https://vite.dev/config/ // https://vite.dev/config/
export default defineConfig({ export default defineConfig({
plugins: [vue()], plugins: [vue()],
server: {
proxy: {
'/api': {
target: 'http://127.0.0.1:3000',
changeOrigin: true,
},
},
},
}) })

363
yarn.lock
View File

@@ -34,31 +34,9 @@
"@element-plus/icons-vue@^2.3.2": "@element-plus/icons-vue@^2.3.2":
version "2.3.2" version "2.3.2"
resolved "https://registry.npmmirror.com/@element-plus/icons-vue/-/icons-vue-2.3.2.tgz#7e9cb231fb738b2056f33e22c3a29e214b538dcf" resolved "https://registry.npmjs.org/@element-plus/icons-vue/-/icons-vue-2.3.2.tgz"
integrity sha512-OzIuTaIfC8QXEPmJvB4Y4kw34rSXdCJzxcD1kFStBvr8bK6X1zQAYDo0CNMjojnfTqRQCJ0I7prlErcoRiET2A== integrity sha512-OzIuTaIfC8QXEPmJvB4Y4kw34rSXdCJzxcD1kFStBvr8bK6X1zQAYDo0CNMjojnfTqRQCJ0I7prlErcoRiET2A==
"@emnapi/core@1.10.0":
version "1.10.0"
resolved "https://registry.yarnpkg.com/@emnapi/core/-/core-1.10.0.tgz#380ccc8f2412ea22d1d972df7f8ee23a3b9c7467"
integrity sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==
dependencies:
"@emnapi/wasi-threads" "1.2.1"
tslib "^2.4.0"
"@emnapi/runtime@1.10.0":
version "1.10.0"
resolved "https://registry.yarnpkg.com/@emnapi/runtime/-/runtime-1.10.0.tgz#4b260c0d3534204e98c6110b8db1a987d26ec87c"
integrity sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==
dependencies:
tslib "^2.4.0"
"@emnapi/wasi-threads@1.2.1":
version "1.2.1"
resolved "https://registry.yarnpkg.com/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz#28fed21a1ba1ce797c44a070abc94d42f3ae8548"
integrity sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==
dependencies:
tslib "^2.4.0"
"@floating-ui/core@^1.7.5": "@floating-ui/core@^1.7.5":
version "1.7.5" version "1.7.5"
resolved "https://registry.npmjs.org/@floating-ui/core/-/core-1.7.5.tgz" resolved "https://registry.npmjs.org/@floating-ui/core/-/core-1.7.5.tgz"
@@ -84,13 +62,6 @@
resolved "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz" resolved "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz"
integrity sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og== integrity sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==
"@napi-rs/wasm-runtime@^1.1.4":
version "1.1.4"
resolved "https://registry.yarnpkg.com/@napi-rs/wasm-runtime/-/wasm-runtime-1.1.4.tgz#a46bbfedc29751b7170c5d23bc1d8ee8c7e3c1e1"
integrity sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow==
dependencies:
"@tybys/wasm-util" "^0.10.1"
"@oxc-project/types@=0.133.0": "@oxc-project/types@=0.133.0":
version "0.133.0" version "0.133.0"
resolved "https://registry.npmjs.org/@oxc-project/types/-/types-0.133.0.tgz" resolved "https://registry.npmjs.org/@oxc-project/types/-/types-0.133.0.tgz"
@@ -101,98 +72,22 @@
resolved "https://registry.npmjs.org/@sxzz/popperjs-es/-/popperjs-es-2.11.8.tgz" resolved "https://registry.npmjs.org/@sxzz/popperjs-es/-/popperjs-es-2.11.8.tgz"
integrity sha512-wOwESXvvED3S8xBmcPWHs2dUuzrE4XiZeFu7e1hROIJkm02a49N120pmOXxY33sBb6hArItm5W5tcg1cBtV+HQ== integrity sha512-wOwESXvvED3S8xBmcPWHs2dUuzrE4XiZeFu7e1hROIJkm02a49N120pmOXxY33sBb6hArItm5W5tcg1cBtV+HQ==
"@rolldown/binding-android-arm64@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-android-arm64/-/binding-android-arm64-1.0.3.tgz#54ce8f8382213f4a314a0c2f7ba83f81ffeae592"
integrity sha512-454rs7jHngixp/NMxd5srYD57OnzSlZ/eFTETjORQHLwJG1lRtmNOJcBerZlfu4GjKqeq8aCCIQrMdHyhI51Hw==
"@rolldown/binding-darwin-arm64@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-darwin-arm64/-/binding-darwin-arm64-1.0.3.tgz#388fca1566c14c00c4b446fc3928630e7f0d95fc"
integrity sha512-PcAhP+ynjURNyy8SKGl5DQP94aGuB/7JrXJb/t7P+hanXvQVMWzUvRRhBAcg/lNRadBhoUPqSoP4xw5tR/KBEA==
"@rolldown/binding-darwin-x64@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-darwin-x64/-/binding-darwin-x64-1.0.3.tgz#53f57de1f599ecf1db13823cfc88c18fb80954ad"
integrity sha512-9YpfeUvSE2RS7wysJ81uOZkXJz7f7Q55H2Gvp3VEw/EsahqDtrphrZ0EwDLK5vvKOzaCrBsjF8JmnMLcUt78Gg==
"@rolldown/binding-freebsd-x64@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-freebsd-x64/-/binding-freebsd-x64-1.0.3.tgz#6f3fdda1b7aeaac9d268a526804b4fb96e4e35f1"
integrity sha512-yB1IlAsSNHncV6SCTL27/MVGR5htvQsoGxIv5KMGXALp+Ll1wYsn+x98M9MW7qa+NdSbvrrY7ANI4wLJ0n1e6g==
"@rolldown/binding-linux-arm-gnueabihf@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-linux-arm-gnueabihf/-/binding-linux-arm-gnueabihf-1.0.3.tgz#d87a454bf585cc9676849377e91d6e375297326f"
integrity sha512-Yi30IVAAfLUCy2MseFjbB1jAMDl1VMCAas5StnYp8da9+CKvMd2H2cbEjWcw5NPaPqzvYkVIaF1nNUG+b7u/sw==
"@rolldown/binding-linux-arm64-gnu@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-linux-arm64-gnu/-/binding-linux-arm64-gnu-1.0.3.tgz#419fd6bf612cf348f10528cbcd94ebab9607d8d1"
integrity sha512-jsO7R8To+AdlYgUmN5sHSCZbfhtMBkO0WUx8iORQnPcMMdgr7qM2DQmMwgabs3GhNztdmoKkMKQFHD6DTMCIQw==
"@rolldown/binding-linux-arm64-musl@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-linux-arm64-musl/-/binding-linux-arm64-musl-1.0.3.tgz#fcc6918696bb76844877e1e4930a18fd0d374069"
integrity sha512-VWkUHwWriDciit80wleYwKILoR/KMvxh/IdwS/paX+ZgpuRpCrKLUdadJbc0NpBEiyhpYawsJ73j9aCvOH+f7Q==
"@rolldown/binding-linux-ppc64-gnu@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-linux-ppc64-gnu/-/binding-linux-ppc64-gnu-1.0.3.tgz#32aecb7c8dae5d4f2a8cde57a058ec86991542f8"
integrity sha512-5f1laC0SlIR0yDbFCd8acUhvJIag6N3zC5P7oUPN6wX0aOma+uKJ0wBDH5aq7I1PVI2ttTlhJwzwRIBnLiSGEg==
"@rolldown/binding-linux-s390x-gnu@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-linux-s390x-gnu/-/binding-linux-s390x-gnu-1.0.3.tgz#bed9346ea81e6bb8b93cf11f5d88b77db890b763"
integrity sha512-Iq4ko0r4XsgbrF/LunNgHtAGLRRVE2kXonAXQ/MV0mC6jQpMOhW1SvtZja2EhC/kd05++bP78dsqBeIQyYJ6Yg==
"@rolldown/binding-linux-x64-gnu@1.0.3": "@rolldown/binding-linux-x64-gnu@1.0.3":
version "1.0.3" version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-linux-x64-gnu/-/binding-linux-x64-gnu-1.0.3.tgz#64c2d26f75dffd9b5a1f97557a00ae77250c8cb7" resolved "https://registry.npmjs.org/@rolldown/binding-linux-x64-gnu/-/binding-linux-x64-gnu-1.0.3.tgz"
integrity sha512-B8m6tD5+/N5FeNQFbKlLA/2yVq9ycQP1SeedyEYYKWBNR3ZQbkvIUcNnDNM03lO1l5F2roiiFJGgvoLLyZXtSg== integrity sha512-B8m6tD5+/N5FeNQFbKlLA/2yVq9ycQP1SeedyEYYKWBNR3ZQbkvIUcNnDNM03lO1l5F2roiiFJGgvoLLyZXtSg==
"@rolldown/binding-linux-x64-musl@1.0.3": "@rolldown/binding-linux-x64-musl@1.0.3":
version "1.0.3" version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-linux-x64-musl/-/binding-linux-x64-musl-1.0.3.tgz#5a45132e8a47659eeaaf3b540c2954a97c860ff3" resolved "https://registry.npmjs.org/@rolldown/binding-linux-x64-musl/-/binding-linux-x64-musl-1.0.3.tgz"
integrity sha512-pSdpdUJHkuCxun9LE7jvgUB9qsRgaiyNNCX7m/AvHTcq67AiT/Yhoxvw5zPfhrM8k/BfP8ce/hMOpthKDpEUow== integrity sha512-pSdpdUJHkuCxun9LE7jvgUB9qsRgaiyNNCX7m/AvHTcq67AiT/Yhoxvw5zPfhrM8k/BfP8ce/hMOpthKDpEUow==
"@rolldown/binding-openharmony-arm64@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-openharmony-arm64/-/binding-openharmony-arm64-1.0.3.tgz#290513068c55e849dc8457a32afee1d7b0acb309"
integrity sha512-OXXS3RKJgX2uLwM+gYyuH5omcH8fL1LJs96pZGgtetVCahON57+d4SJHzTgZiOjxgGkSnpXpOsWuPDGAKAigEg==
"@rolldown/binding-wasm32-wasi@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-wasm32-wasi/-/binding-wasm32-wasi-1.0.3.tgz#3d9972dbf1a953d3c7afaa4a0f20ef2b2e39f31b"
integrity sha512-JTtb8BWFynicNSoPrehsCzBtOKjZ6jhMiPFEmOiuXg1Fl8dn2KHQob+GuPSGR0dryQa1PQJbzjF3dqO/whhjLg==
dependencies:
"@emnapi/core" "1.10.0"
"@emnapi/runtime" "1.10.0"
"@napi-rs/wasm-runtime" "^1.1.4"
"@rolldown/binding-win32-arm64-msvc@1.0.3":
version "1.0.3"
resolved "https://registry.yarnpkg.com/@rolldown/binding-win32-arm64-msvc/-/binding-win32-arm64-msvc-1.0.3.tgz#a004ab607a16d6f03bcb555728ff888af75773ad"
integrity sha512-gEdFFEN70A/jxb2svrWsN3aDL7OUtmvlOy+6fa2jxG8K0wQ1ZbdeLGnidov6Yu5/733dI5ySfzFlQ/cb0bSz1g==
"@rolldown/binding-win32-x64-msvc@1.0.3":
version "1.0.3"
resolved "https://registry.npmjs.org/@rolldown/binding-win32-x64-msvc/-/binding-win32-x64-msvc-1.0.3.tgz"
integrity sha512-eXB7CHuaQdqmJcc3koCNtNPmT/bj2gc999kUFgBxG8Ac0NdgXc4rkCHhqrgrhN3zddvvvrgzj1e90SuSfmyIXA==
"@rolldown/pluginutils@^1.0.0", "@rolldown/pluginutils@^1.0.1": "@rolldown/pluginutils@^1.0.0", "@rolldown/pluginutils@^1.0.1":
version "1.0.1" version "1.0.1"
resolved "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.1.tgz" resolved "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.1.tgz"
integrity sha512-2j9bGt5Jh8hj+vPtgzPtl72j0yRxHAyumoo6TNfAjsLB04UtpSvPbPcDcBMxz7n+9CYB0c1GxQFxYRg2jimqGw== integrity sha512-2j9bGt5Jh8hj+vPtgzPtl72j0yRxHAyumoo6TNfAjsLB04UtpSvPbPcDcBMxz7n+9CYB0c1GxQFxYRg2jimqGw==
"@tybys/wasm-util@^0.10.1": "@types/lodash-es@*", "@types/lodash-es@^4.17.12":
version "0.10.2"
resolved "https://registry.yarnpkg.com/@tybys/wasm-util/-/wasm-util-0.10.2.tgz#12b3a1b33db1f9cad4ddff1f604ab7dd00bf464e"
integrity sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg==
dependencies:
tslib "^2.4.0"
"@types/lodash-es@^4.17.12":
version "4.17.12" version "4.17.12"
resolved "https://registry.npmjs.org/@types/lodash-es/-/lodash-es-4.17.12.tgz" resolved "https://registry.npmjs.org/@types/lodash-es/-/lodash-es-4.17.12.tgz"
integrity sha512-0NgftHUcV4v34VhXm8QBSftKVXtbkBG3ViCjs6+eJ5a6y6Mi/jiFGPc1sC7QK+9BFhWrURE3EOggmWaSxL9OzQ== integrity sha512-0NgftHUcV4v34VhXm8QBSftKVXtbkBG3ViCjs6+eJ5a6y6Mi/jiFGPc1sC7QK+9BFhWrURE3EOggmWaSxL9OzQ==
@@ -260,7 +155,7 @@
"@vue/devtools-api@^6.6.4": "@vue/devtools-api@^6.6.4":
version "6.6.4" version "6.6.4"
resolved "https://registry.yarnpkg.com/@vue/devtools-api/-/devtools-api-6.6.4.tgz#cbe97fe0162b365edc1dba80e173f90492535343" resolved "https://registry.npmjs.org/@vue/devtools-api/-/devtools-api-6.6.4.tgz"
integrity sha512-sGhTPMuXqZ1rVOk32RylztWkfXTRhuS7vgAKv0zjqk8gbsHkJ7xfFf+jbySxt7tWObEJwyKaHMikV/WGDiQm8g== integrity sha512-sGhTPMuXqZ1rVOk32RylztWkfXTRhuS7vgAKv0zjqk8gbsHkJ7xfFf+jbySxt7tWObEJwyKaHMikV/WGDiQm8g==
"@vue/reactivity@3.5.35": "@vue/reactivity@3.5.35":
@@ -320,16 +215,53 @@
resolved "https://registry.npmjs.org/@vueuse/shared/-/shared-14.3.0.tgz" resolved "https://registry.npmjs.org/@vueuse/shared/-/shared-14.3.0.tgz"
integrity sha512-bZpge9eSXwa4ToSiqJ7j6KRwhAsneMFoSz3LMWKQDkqimm3D/tbFlrklrs/IOqC8tEcYmXQZJ6N0UrjhBirVCg== integrity sha512-bZpge9eSXwa4ToSiqJ7j6KRwhAsneMFoSz3LMWKQDkqimm3D/tbFlrklrs/IOqC8tEcYmXQZJ6N0UrjhBirVCg==
agent-base@6:
version "6.0.2"
resolved "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz"
integrity sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==
dependencies:
debug "4"
async-validator@^4.2.5: async-validator@^4.2.5:
version "4.2.5" version "4.2.5"
resolved "https://registry.npmjs.org/async-validator/-/async-validator-4.2.5.tgz" resolved "https://registry.npmjs.org/async-validator/-/async-validator-4.2.5.tgz"
integrity sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg== integrity sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg==
asynckit@^0.4.0:
version "0.4.0"
resolved "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz"
integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==
axios@^1.18.0:
version "1.18.0"
resolved "https://registry.npmjs.org/axios/-/axios-1.18.0.tgz"
integrity sha512-E32NzpYKp++W7XRe52rHiXV2ehxmh3wbdgO7MHeFM+vqxLBYHzt0ElkiImtOBxtOmyp0yoC8C6uESVV84Y2/hw==
dependencies:
follow-redirects "^1.16.0"
form-data "^4.0.5"
https-proxy-agent "^5.0.1"
proxy-from-env "^2.1.0"
call-bind-apply-helpers@^1.0.1, call-bind-apply-helpers@^1.0.2:
version "1.0.2"
resolved "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz"
integrity sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==
dependencies:
es-errors "^1.3.0"
function-bind "^1.1.2"
china-division@^2.7.0: china-division@^2.7.0:
version "2.7.0" version "2.7.0"
resolved "https://registry.npmmirror.com/china-division/-/china-division-2.7.0.tgz#4060a4d243be66c7833dea64a48a4038f3e53e74" resolved "https://registry.npmjs.org/china-division/-/china-division-2.7.0.tgz"
integrity sha512-4uUPAT+1WfqDh5jytq7omdCmHNk3j+k76zEG/2IqaGcYB90c2SwcixttcypdsZ3T/9tN1TTpBDoeZn+Yw/qBEA== integrity sha512-4uUPAT+1WfqDh5jytq7omdCmHNk3j+k76zEG/2IqaGcYB90c2SwcixttcypdsZ3T/9tN1TTpBDoeZn+Yw/qBEA==
combined-stream@^1.0.8:
version "1.0.8"
resolved "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz"
integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==
dependencies:
delayed-stream "~1.0.0"
csstype@^3.2.3: csstype@^3.2.3:
version "3.2.3" version "3.2.3"
resolved "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz" resolved "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz"
@@ -340,14 +272,35 @@ dayjs@^1.11.20:
resolved "https://registry.npmjs.org/dayjs/-/dayjs-1.11.21.tgz" resolved "https://registry.npmjs.org/dayjs/-/dayjs-1.11.21.tgz"
integrity sha512-98IT+HOahAisibz/yjKbzuOBwYcjJ7BCLPzARyHiyEBmRz4fatF+KPJszEHXsGYjUG234aH/cOjW1wwTbKUZlA== integrity sha512-98IT+HOahAisibz/yjKbzuOBwYcjJ7BCLPzARyHiyEBmRz4fatF+KPJszEHXsGYjUG234aH/cOjW1wwTbKUZlA==
debug@4:
version "4.4.3"
resolved "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz"
integrity sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==
dependencies:
ms "^2.1.3"
delayed-stream@~1.0.0:
version "1.0.0"
resolved "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz"
integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==
detect-libc@^2.0.3: detect-libc@^2.0.3:
version "2.1.2" version "2.1.2"
resolved "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz" resolved "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz"
integrity sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ== integrity sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==
dunder-proto@^1.0.1:
version "1.0.1"
resolved "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz"
integrity sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==
dependencies:
call-bind-apply-helpers "^1.0.1"
es-errors "^1.3.0"
gopd "^1.2.0"
element-china-area-data@^6.1.0: element-china-area-data@^6.1.0:
version "6.1.0" version "6.1.0"
resolved "https://registry.npmmirror.com/element-china-area-data/-/element-china-area-data-6.1.0.tgz#f14b90c0762b21432e097ed5be8423514a0b57e3" resolved "https://registry.npmjs.org/element-china-area-data/-/element-china-area-data-6.1.0.tgz"
integrity sha512-IkpcjwQv2A/2AxFiSoaISZ+oMw1rZCPUSOg5sOCwT5jKc96TaawmKZeY81xfxXsO0QbKxU5LLc6AirhG52hUmg== integrity sha512-IkpcjwQv2A/2AxFiSoaISZ+oMw1rZCPUSOg5sOCwT5jKc96TaawmKZeY81xfxXsO0QbKxU5LLc6AirhG52hUmg==
dependencies: dependencies:
china-division "^2.7.0" china-division "^2.7.0"
@@ -378,6 +331,33 @@ entities@^7.0.1:
resolved "https://registry.npmjs.org/entities/-/entities-7.0.1.tgz" resolved "https://registry.npmjs.org/entities/-/entities-7.0.1.tgz"
integrity sha512-TWrgLOFUQTH994YUyl1yT4uyavY5nNB5muff+RtWaqNVCAK408b5ZnnbNAUEWLTCpum9w6arT70i1XdQ4UeOPA== integrity sha512-TWrgLOFUQTH994YUyl1yT4uyavY5nNB5muff+RtWaqNVCAK408b5ZnnbNAUEWLTCpum9w6arT70i1XdQ4UeOPA==
es-define-property@^1.0.1:
version "1.0.1"
resolved "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz"
integrity sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==
es-errors@^1.3.0:
version "1.3.0"
resolved "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz"
integrity sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==
es-object-atoms@^1.0.0, es-object-atoms@^1.1.1:
version "1.1.2"
resolved "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.2.tgz"
integrity sha512-HWcBoN6NileqtSydK2FqHbS/LoDd2pqrnQHLyJzBj4kOp/ky2MWMN694xOfkK8/SnUsW2DH7EfyVlydKCsm1Zw==
dependencies:
es-errors "^1.3.0"
es-set-tostringtag@^2.1.0:
version "2.1.0"
resolved "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz"
integrity sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==
dependencies:
es-errors "^1.3.0"
get-intrinsic "^1.2.6"
has-tostringtag "^1.0.2"
hasown "^2.0.2"
estree-walker@^2.0.2: estree-walker@^2.0.2:
version "2.0.2" version "2.0.2"
resolved "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz" resolved "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz"
@@ -388,66 +368,93 @@ fdir@^6.5.0:
resolved "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz" resolved "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz"
integrity sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg== integrity sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==
fsevents@~2.3.3: follow-redirects@^1.16.0:
version "2.3.3" version "1.16.0"
resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" resolved "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.16.0.tgz"
integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw== integrity sha512-y5rN/uOsadFT/JfYwhxRS5R7Qce+g3zG97+JrtFZlC9klX/W5hD7iiLzScI4nZqUS7DNUdhPgw4xI8W2LuXlUw==
lightningcss-android-arm64@1.32.0: form-data@^4.0.5:
version "1.32.0" version "4.0.6"
resolved "https://registry.yarnpkg.com/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz#f033885116dfefd9c6f54787523e3514b61e1968" resolved "https://registry.npmjs.org/form-data/-/form-data-4.0.6.tgz"
integrity sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg== integrity sha512-vKatAh4SlVfgbv+YtmhiRjhEMJsYpsG1Y2rMQtR+SVSbytsSD1YGzDIcrAJmdFec88u/+VoGmxnl+80gL1tRCQ==
dependencies:
asynckit "^0.4.0"
combined-stream "^1.0.8"
es-set-tostringtag "^2.1.0"
hasown "^2.0.4"
mime-types "^2.1.35"
lightningcss-darwin-arm64@1.32.0: function-bind@^1.1.2:
version "1.32.0" version "1.1.2"
resolved "https://registry.yarnpkg.com/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz#50b71871b01c8199584b649e292547faea7af9b5" resolved "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz"
integrity sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ== integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==
lightningcss-darwin-x64@1.32.0: get-intrinsic@^1.2.6:
version "1.32.0" version "1.3.0"
resolved "https://registry.yarnpkg.com/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz#35f3e97332d130b9ca181e11b568ded6aebc6d5e" resolved "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz"
integrity sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w== integrity sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==
dependencies:
call-bind-apply-helpers "^1.0.2"
es-define-property "^1.0.1"
es-errors "^1.3.0"
es-object-atoms "^1.1.1"
function-bind "^1.1.2"
get-proto "^1.0.1"
gopd "^1.2.0"
has-symbols "^1.1.0"
hasown "^2.0.2"
math-intrinsics "^1.1.0"
lightningcss-freebsd-x64@1.32.0: get-proto@^1.0.1:
version "1.32.0" version "1.0.1"
resolved "https://registry.yarnpkg.com/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz#9777a76472b64ed6ff94342ad64c7bafd794a575" resolved "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz"
integrity sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig== integrity sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==
dependencies:
dunder-proto "^1.0.1"
es-object-atoms "^1.0.0"
lightningcss-linux-arm-gnueabihf@1.32.0: gopd@^1.2.0:
version "1.32.0" version "1.2.0"
resolved "https://registry.yarnpkg.com/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz#13ae652e1ab73b9135d7b7da172f666c410ad53d" resolved "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz"
integrity sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw== integrity sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==
lightningcss-linux-arm64-gnu@1.32.0: has-symbols@^1.0.3, has-symbols@^1.1.0:
version "1.32.0" version "1.1.0"
resolved "https://registry.yarnpkg.com/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz#417858795a94592f680123a1b1f9da8a0e1ef335" resolved "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz"
integrity sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ== integrity sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==
lightningcss-linux-arm64-musl@1.32.0: has-tostringtag@^1.0.2:
version "1.32.0" version "1.0.2"
resolved "https://registry.yarnpkg.com/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz#6be36692e810b718040802fd809623cffe732133" resolved "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz"
integrity sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg== integrity sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==
dependencies:
has-symbols "^1.0.3"
hasown@^2.0.2, hasown@^2.0.4:
version "2.0.4"
resolved "https://registry.npmjs.org/hasown/-/hasown-2.0.4.tgz"
integrity sha512-T2UbfbBEF32wiepXIsMlTW9+dDYC6wMh/t/vYA4tuOMKqWz/n3vr1NFSxQiyP+zk2mXsoMA/i/7qV6LKut1t1A==
dependencies:
function-bind "^1.1.2"
https-proxy-agent@^5.0.1:
version "5.0.1"
resolved "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz"
integrity sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==
dependencies:
agent-base "6"
debug "4"
lightningcss-linux-x64-gnu@1.32.0: lightningcss-linux-x64-gnu@1.32.0:
version "1.32.0" version "1.32.0"
resolved "https://registry.yarnpkg.com/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz#0b7803af4eb21cfd38dd39fe2abbb53c7dd091f6" resolved "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz"
integrity sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA== integrity sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==
lightningcss-linux-x64-musl@1.32.0: lightningcss-linux-x64-musl@1.32.0:
version "1.32.0" version "1.32.0"
resolved "https://registry.yarnpkg.com/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz#88dc8ba865ddddb1ac5ef04b0f161804418c163b" resolved "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz"
integrity sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg== integrity sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==
lightningcss-win32-arm64-msvc@1.32.0:
version "1.32.0"
resolved "https://registry.yarnpkg.com/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz#4f30ba3fa5e925f5b79f945e8cc0d176c3b1ab38"
integrity sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==
lightningcss-win32-x64-msvc@1.32.0:
version "1.32.0"
resolved "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz"
integrity sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==
lightningcss@^1.32.0: lightningcss@^1.32.0:
version "1.32.0" version "1.32.0"
resolved "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz" resolved "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz"
@@ -467,7 +474,7 @@ lightningcss@^1.32.0:
lightningcss-win32-arm64-msvc "1.32.0" lightningcss-win32-arm64-msvc "1.32.0"
lightningcss-win32-x64-msvc "1.32.0" lightningcss-win32-x64-msvc "1.32.0"
lodash-es@^4.18.1: lodash-es@*, lodash-es@^4.18.1:
version "4.18.1" version "4.18.1"
resolved "https://registry.npmjs.org/lodash-es/-/lodash-es-4.18.1.tgz" resolved "https://registry.npmjs.org/lodash-es/-/lodash-es-4.18.1.tgz"
integrity sha512-J8xewKD/Gk22OZbhpOVSwcs60zhd95ESDwezOFuA3/099925PdHJ7OFHNTGtajL3AlZkykD32HykiMo+BIBI8A== integrity sha512-J8xewKD/Gk22OZbhpOVSwcs60zhd95ESDwezOFuA3/099925PdHJ7OFHNTGtajL3AlZkykD32HykiMo+BIBI8A==
@@ -477,7 +484,7 @@ lodash-unified@^1.0.3:
resolved "https://registry.npmjs.org/lodash-unified/-/lodash-unified-1.0.3.tgz" resolved "https://registry.npmjs.org/lodash-unified/-/lodash-unified-1.0.3.tgz"
integrity sha512-WK9qSozxXOD7ZJQlpSqOT+om2ZfcT4yO+03FuzAHD0wF6S0l0090LRPDx3vhTTLZ8cFKpBn+IOcVXK6qOcIlfQ== integrity sha512-WK9qSozxXOD7ZJQlpSqOT+om2ZfcT4yO+03FuzAHD0wF6S0l0090LRPDx3vhTTLZ8cFKpBn+IOcVXK6qOcIlfQ==
lodash@^4.18.1: lodash@*, lodash@^4.18.1:
version "4.18.1" version "4.18.1"
resolved "https://registry.npmjs.org/lodash/-/lodash-4.18.1.tgz" resolved "https://registry.npmjs.org/lodash/-/lodash-4.18.1.tgz"
integrity sha512-dMInicTPVE8d1e5otfwmmjlxkZoUpiVLwyeTdUsi/Caj/gfzzblBcCE5sRHV/AsjuCmxWrte2TNGSYuCeCq+0Q== integrity sha512-dMInicTPVE8d1e5otfwmmjlxkZoUpiVLwyeTdUsi/Caj/gfzzblBcCE5sRHV/AsjuCmxWrte2TNGSYuCeCq+0Q==
@@ -489,11 +496,33 @@ magic-string@^0.30.21:
dependencies: dependencies:
"@jridgewell/sourcemap-codec" "^1.5.5" "@jridgewell/sourcemap-codec" "^1.5.5"
math-intrinsics@^1.1.0:
version "1.1.0"
resolved "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz"
integrity sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==
memoize-one@^6.0.0: memoize-one@^6.0.0:
version "6.0.0" version "6.0.0"
resolved "https://registry.npmjs.org/memoize-one/-/memoize-one-6.0.0.tgz" resolved "https://registry.npmjs.org/memoize-one/-/memoize-one-6.0.0.tgz"
integrity sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw== integrity sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw==
mime-db@1.52.0:
version "1.52.0"
resolved "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz"
integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==
mime-types@^2.1.35:
version "2.1.35"
resolved "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz"
integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==
dependencies:
mime-db "1.52.0"
ms@^2.1.3:
version "2.1.3"
resolved "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz"
integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==
nanoid@^3.3.12: nanoid@^3.3.12:
version "3.3.12" version "3.3.12"
resolved "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz" resolved "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz"
@@ -509,7 +538,7 @@ picocolors@^1.1.1:
resolved "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz" resolved "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz"
integrity sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA== integrity sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==
picomatch@^4.0.4: "picomatch@^3 || ^4", picomatch@^4.0.4:
version "4.0.4" version "4.0.4"
resolved "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz" resolved "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz"
integrity sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A== integrity sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==
@@ -523,6 +552,11 @@ postcss@^8.5.15:
picocolors "^1.1.1" picocolors "^1.1.1"
source-map-js "^1.2.1" source-map-js "^1.2.1"
proxy-from-env@^2.1.0:
version "2.1.0"
resolved "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-2.1.0.tgz"
integrity sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA==
rolldown@1.0.3: rolldown@1.0.3:
version "1.0.3" version "1.0.3"
resolved "https://registry.npmjs.org/rolldown/-/rolldown-1.0.3.tgz" resolved "https://registry.npmjs.org/rolldown/-/rolldown-1.0.3.tgz"
@@ -560,12 +594,7 @@ tinyglobby@^0.2.17:
fdir "^6.5.0" fdir "^6.5.0"
picomatch "^4.0.4" picomatch "^4.0.4"
tslib@^2.4.0: "vite@^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0", vite@^8.0.12:
version "2.8.1"
resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.8.1.tgz#612efe4ed235d567e8aba5f2a5fab70280ade83f"
integrity sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==
vite@^8.0.12:
version "8.0.16" version "8.0.16"
resolved "https://registry.npmjs.org/vite/-/vite-8.0.16.tgz" resolved "https://registry.npmjs.org/vite/-/vite-8.0.16.tgz"
integrity sha512-h9bXPmJichP5fLmVQo3PyaGSDE2n3aPuomeAlVRm0JLmt4rY6zmPKd59HYI4LNW8oTK7tlTsuC7l/m7awx9Jcw== integrity sha512-h9bXPmJichP5fLmVQo3PyaGSDE2n3aPuomeAlVRm0JLmt4rY6zmPKd59HYI4LNW8oTK7tlTsuC7l/m7awx9Jcw==
@@ -585,12 +614,12 @@ vue-component-type-helpers@^3.3.1:
vue-router@4: vue-router@4:
version "4.6.4" version "4.6.4"
resolved "https://registry.yarnpkg.com/vue-router/-/vue-router-4.6.4.tgz#a0a9cb9ef811a106d249e4bb9313d286718020d8" resolved "https://registry.npmjs.org/vue-router/-/vue-router-4.6.4.tgz"
integrity sha512-Hz9q5sa33Yhduglwz6g9skT8OBPii+4bFn88w6J+J4MfEo4KRRpmiNG/hHHkdbRFlLBOqxN8y8gf2Fb0MTUgVg== integrity sha512-Hz9q5sa33Yhduglwz6g9skT8OBPii+4bFn88w6J+J4MfEo4KRRpmiNG/hHHkdbRFlLBOqxN8y8gf2Fb0MTUgVg==
dependencies: dependencies:
"@vue/devtools-api" "^6.6.4" "@vue/devtools-api" "^6.6.4"
vue@^3.5.34: vue@^3.2.0, vue@^3.2.25, vue@^3.3.7, vue@^3.5.0, vue@^3.5.34, vue@3.5.35:
version "3.5.35" version "3.5.35"
resolved "https://registry.npmjs.org/vue/-/vue-3.5.35.tgz" resolved "https://registry.npmjs.org/vue/-/vue-3.5.35.tgz"
integrity sha512-cx89fnr+0kVGHiNFG6y6s0bdjypJRFNZn6x3WPstNdQR1bi1mbB7h4v5IBGTsPJU3nK1+0Iqj3Zf+hZWMieR4Q== integrity sha512-cx89fnr+0kVGHiNFG6y6s0bdjypJRFNZn6x3WPstNdQR1bi1mbB7h4v5IBGTsPJU3nK1+0Iqj3Zf+hZWMieR4Q==