From 4dbcc7b837c59b4c3c7cbe48e12a07f05565adac Mon Sep 17 00:00:00 2001 From: lixin Date: Thu, 9 Jan 2025 16:13:14 +0800 Subject: [PATCH] init --- .dockerignore | 5 + .editorconfig | 11 + .env | 5 + .eslintignore | 3 + .eslintrc.js | 47 + .gitattributes | 4 + .gitignore | 18 + .hintrc | 16 + .prettierrc | 8 + .vscode/config.code-snippets | 15 + .vscode/crud.code-snippets | 102 + .vscode/settings.json | 4 + Dockerfile | 16 + LICENSE | 21 + README.md | 57 + build/cool/eps.d.ts | 3475 +++++++++++ build/cool/eps.json | 1 + index.html | 172 + nginx.conf | 123 + package.json | 64 + public/favicon.ico | Bin 0 -> 67646 bytes public/logo.png | Bin 0 -> 1403 bytes public/用户导入模版.xlsx | Bin 0 -> 9198 bytes src/App.vue | 10 + src/config/dev.ts | 20 + src/config/index.ts | 60 + src/config/prod.ts | 9 + src/config/proxy.ts | 13 + src/cool/bootstrap/eps.ts | 129 + src/cool/bootstrap/index.ts | 24 + src/cool/bootstrap/module.ts | 116 + src/cool/hooks/browser.ts | 30 + src/cool/hooks/hmr.ts | 23 + src/cool/hooks/index.ts | 54 + src/cool/hooks/mitt.ts | 8 + src/cool/index.ts | 7 + src/cool/module/index.ts | 27 + src/cool/router/index.ts | 217 + src/cool/service/base.ts | 117 + src/cool/service/index.ts | 9 + src/cool/service/request.ts | 158 + src/cool/types/index.ts | 60 + src/cool/utils/index.ts | 301 + src/cool/utils/loading.ts | 37 + src/cool/utils/storage.ts | 81 + src/main.ts | 17 + src/modules/app/views/complain.vue | 148 + src/modules/app/views/feedback.vue | 148 + src/modules/app/views/goods.vue | 188 + src/modules/app/views/version.vue | 177 + src/modules/base/common/index.ts | 2 + src/modules/base/common/permission.ts | 30 + src/modules/base/common/theme.ts | 12 + src/modules/base/components/avatar/index.tsx | 53 + src/modules/base/components/code/json.vue | 134 + src/modules/base/components/dept/check.vue | 104 + src/modules/base/components/dept/select.vue | 83 + src/modules/base/components/editor/index.tsx | 42 + src/modules/base/components/icon/svg.vue | 48 + src/modules/base/components/image/index.vue | 98 + src/modules/base/components/link/index.vue | 86 + src/modules/base/components/menu/check.vue | 92 + src/modules/base/components/menu/file.vue | 128 + src/modules/base/components/menu/icon.vue | 51 + src/modules/base/components/menu/perms.vue | 91 + src/modules/base/components/menu/select.vue | 83 + src/modules/base/config.ts | 90 + src/modules/base/directives/permission.ts | 13 + src/modules/base/hooks/dept.tsx | 43 + src/modules/base/hooks/index.ts | 1 + src/modules/base/index.ts | 11 + src/modules/base/pages/error/401.vue | 7 + src/modules/base/pages/error/403.vue | 7 + src/modules/base/pages/error/404.vue | 7 + src/modules/base/pages/error/500.vue | 7 + src/modules/base/pages/error/502.vue | 7 + .../pages/error/components/error-page.vue | 136 + .../pages/login/components/pic-captcha.vue | 110 + src/modules/base/pages/login/index.vue | 297 + src/modules/base/pages/login/static/bg.svg | 39 + .../base/pages/main/components/amenu.vue | 154 + .../base/pages/main/components/bmenu.tsx | 97 + .../base/pages/main/components/process.vue | 278 + .../base/pages/main/components/route-nav.vue | 72 + .../base/pages/main/components/slider.vue | 95 + .../base/pages/main/components/topbar.vue | 186 + .../base/pages/main/components/views.vue | 95 + src/modules/base/pages/main/index.vue | 105 + src/modules/base/static/css/animation.scss | 29 + src/modules/base/static/css/index.scss | 26 + src/modules/base/static/svg/icon-activity.svg | 1 + src/modules/base/static/svg/icon-amount.svg | 9 + src/modules/base/static/svg/icon-app.svg | 1 + src/modules/base/static/svg/icon-approve.svg | 1 + src/modules/base/static/svg/icon-auth.svg | 1 + src/modules/base/static/svg/icon-ban.svg | 1 + src/modules/base/static/svg/icon-camera.svg | 1 + src/modules/base/static/svg/icon-card.svg | 1 + src/modules/base/static/svg/icon-cart.svg | 1 + src/modules/base/static/svg/icon-command.svg | 1 + src/modules/base/static/svg/icon-common.svg | 1 + .../base/static/svg/icon-component.svg | 17 + src/modules/base/static/svg/icon-count.svg | 1 + src/modules/base/static/svg/icon-crown.svg | 1 + src/modules/base/static/svg/icon-data.svg | 9 + src/modules/base/static/svg/icon-db.svg | 15 + src/modules/base/static/svg/icon-delete.svg | 9 + src/modules/base/static/svg/icon-dept.svg | 9 + src/modules/base/static/svg/icon-design.svg | 12 + src/modules/base/static/svg/icon-device.svg | 9 + src/modules/base/static/svg/icon-dict.svg | 1 + src/modules/base/static/svg/icon-discover.svg | 1 + src/modules/base/static/svg/icon-emoji.svg | 1 + src/modules/base/static/svg/icon-favor.svg | 1 + src/modules/base/static/svg/icon-fx.svg | 29 + src/modules/base/static/svg/icon-goods.svg | 1 + src/modules/base/static/svg/icon-home.svg | 1 + src/modules/base/static/svg/icon-hot.svg | 1 + src/modules/base/static/svg/icon-info.svg | 1 + src/modules/base/static/svg/icon-iot.svg | 43 + src/modules/base/static/svg/icon-like.svg | 1 + src/modules/base/static/svg/icon-living.svg | 1 + src/modules/base/static/svg/icon-log.svg | 1 + src/modules/base/static/svg/icon-map.svg | 21 + src/modules/base/static/svg/icon-menu.svg | 9 + src/modules/base/static/svg/icon-message.svg | 1 + src/modules/base/static/svg/icon-monitor.svg | 9 + src/modules/base/static/svg/icon-new.svg | 1 + src/modules/base/static/svg/icon-news.svg | 1 + src/modules/base/static/svg/icon-notice.svg | 1 + src/modules/base/static/svg/icon-params.svg | 9 + src/modules/base/static/svg/icon-pending.svg | 1 + src/modules/base/static/svg/icon-phone.svg | 9 + src/modules/base/static/svg/icon-pic.svg | 1 + src/modules/base/static/svg/icon-question.svg | 1 + src/modules/base/static/svg/icon-rank.svg | 1 + src/modules/base/static/svg/icon-scan.svg | 1 + src/modules/base/static/svg/icon-search.svg | 1 + src/modules/base/static/svg/icon-system.svg | 1 + src/modules/base/static/svg/icon-tag.svg | 1 + src/modules/base/static/svg/icon-task.svg | 1 + src/modules/base/static/svg/icon-theme.svg | 12 + src/modules/base/static/svg/icon-time.svg | 1 + src/modules/base/static/svg/icon-track.svg | 12 + src/modules/base/static/svg/icon-upload.svg | 1 + src/modules/base/static/svg/icon-user.svg | 1 + src/modules/base/static/svg/icon-video.svg | 1 + src/modules/base/static/svg/icon-warn.svg | 1 + .../base/static/svg/icon-workbench.svg | 1 + src/modules/base/store/app.ts | 59 + src/modules/base/store/index.ts | 18 + src/modules/base/store/menu.ts | 189 + src/modules/base/store/process.ts | 60 + src/modules/base/store/user.ts | 89 + src/modules/base/types/index.d.ts | 44 + src/modules/base/utils/index.ts | 21 + src/modules/base/views/frame.vue | 45 + src/modules/base/views/info.vue | 102 + src/modules/base/views/log.vue | 147 + .../base/views/menu/components/exp.vue | 99 + .../base/views/menu/components/imp.vue | 112 + src/modules/base/views/menu/index.vue | 450 ++ src/modules/base/views/param.vue | 207 + src/modules/base/views/role.vue | 161 + .../base/views/user/components/dept-list.vue | 466 ++ .../base/views/user/components/user-move.vue | 67 + src/modules/base/views/user/index.vue | 330 + src/modules/cs/components/index.vue | 186 + src/modules/cs/components/message.vue | 526 ++ src/modules/cs/components/session.vue | 223 + src/modules/cs/components/tools/emoji.vue | 162 + src/modules/cs/config.ts | 10 + src/modules/cs/hooks/index.ts | 5 + src/modules/cs/hooks/message.ts | 128 + src/modules/cs/hooks/notice.tsx | 68 + src/modules/cs/hooks/session.ts | 97 + src/modules/cs/hooks/socket.ts | 95 + src/modules/cs/hooks/tools.ts | 28 + src/modules/cs/index.ts | 13 + src/modules/cs/static/audio/ding.wav | Bin 0 -> 3534 bytes src/modules/cs/static/emoji/angry-face.png | Bin 0 -> 3669 bytes .../cs/static/emoji/anguished-face.png | Bin 0 -> 3584 bytes .../cs/static/emoji/astonished-face.png | Bin 0 -> 3584 bytes .../cs/static/emoji/confounded-face.png | Bin 0 -> 3683 bytes src/modules/cs/static/emoji/confused-face.png | Bin 0 -> 3670 bytes src/modules/cs/static/emoji/crying-face.png | Bin 0 -> 3824 bytes .../emoji/disappointed-but-relieved-face.png | Bin 0 -> 3709 bytes .../cs/static/emoji/disappointed-face.png | Bin 0 -> 3584 bytes src/modules/cs/static/emoji/dizzy-face.png | Bin 0 -> 3683 bytes src/modules/cs/static/emoji/drooling-face.png | Bin 0 -> 3552 bytes .../cs/static/emoji/expressionless-face.png | Bin 0 -> 3553 bytes .../emoji/face-savouring-delicious-food.png | Bin 0 -> 3701 bytes .../static/emoji/face-screaming-in-fear.png | Bin 0 -> 3672 bytes .../cs/static/emoji/face-throwing-a-kiss.png | Bin 0 -> 3744 bytes .../cs/static/emoji/face-with-cold-sweat.png | Bin 0 -> 3545 bytes .../cs/static/emoji/face-with-cowboy-hat.png | Bin 0 -> 3683 bytes .../face-with-finger-covering-closed-lips.png | Bin 0 -> 3797 bytes .../static/emoji/face-with-head-bandage.png | Bin 0 -> 3586 bytes .../emoji/face-with-look-of-triumph.png | Bin 0 -> 3809 bytes .../static/emoji/face-with-medical-mask.png | Bin 0 -> 3684 bytes .../cs/static/emoji/face-with-monocle.png | Bin 0 -> 3609 bytes .../emoji/face-with-one-eyebrow-raised.png | Bin 0 -> 3483 bytes .../face-with-open-mouth-and-cold-sweat.png | Bin 0 -> 3377 bytes .../emoji/face-with-open-mouth-vomiting.png | Bin 0 -> 4011 bytes .../cs/static/emoji/face-with-open-mouth.png | Bin 0 -> 3667 bytes .../face-with-party-horn-and-party-hat.png | Bin 0 -> 3971 bytes .../static/emoji/face-with-pleading-eyes.png | Bin 0 -> 3498 bytes .../static/emoji/face-with-rolling-eyes.png | Bin 0 -> 3461 bytes ...uck-out-tongue-and-tightly-closed-eyes.png | Bin 0 -> 3622 bytes ...-with-stuck-out-tongue-and-winking-eye.png | Bin 0 -> 3561 bytes .../emoji/face-with-stuck-out-tongue.png | Bin 0 -> 3535 bytes .../cs/static/emoji/face-with-thermometer.png | Bin 0 -> 3560 bytes .../face-with-uneven-eyes-and-wavy-mouth.png | Bin 0 -> 3640 bytes .../cs/static/emoji/face-without-mouth.png | Bin 0 -> 3664 bytes src/modules/cs/static/emoji/fearful-face.png | Bin 0 -> 3366 bytes src/modules/cs/static/emoji/flushed-face.png | Bin 0 -> 3611 bytes src/modules/cs/static/emoji/freezing-face.png | Bin 0 -> 3615 bytes .../emoji/frowning-face-with-open-mouth.png | Bin 0 -> 3666 bytes .../cs/static/emoji/grimacing-face.png | Bin 0 -> 3695 bytes ...-face-with-one-large-and-one-small-eye.png | Bin 0 -> 3719 bytes .../emoji/grinning-face-with-smiling-eyes.png | Bin 0 -> 3677 bytes .../emoji/grinning-face-with-star-eyes.png | Bin 0 -> 3888 bytes src/modules/cs/static/emoji/grinning-face.png | Bin 0 -> 3675 bytes src/modules/cs/static/emoji/hugging-face.png | Bin 0 -> 4061 bytes src/modules/cs/static/emoji/hushed-face.png | Bin 0 -> 3548 bytes src/modules/cs/static/emoji/imp.png | Bin 0 -> 3972 bytes .../emoji/kissing-face-with-closed-eyes.png | Bin 0 -> 3613 bytes .../emoji/kissing-face-with-smiling-eyes.png | Bin 0 -> 3633 bytes src/modules/cs/static/emoji/kissing-face.png | Bin 0 -> 3589 bytes .../cs/static/emoji/loudly-crying-face.png | Bin 0 -> 3671 bytes src/modules/cs/static/emoji/lying-face.png | Bin 0 -> 3755 bytes .../cs/static/emoji/money-mouth-face.png | Bin 0 -> 3680 bytes .../cs/static/emoji/nauseated-face.png | Bin 0 -> 3759 bytes src/modules/cs/static/emoji/nerd-face.png | Bin 0 -> 3673 bytes src/modules/cs/static/emoji/neutral-face.png | Bin 0 -> 3603 bytes .../cs/static/emoji/overheated-face.png | Bin 0 -> 3440 bytes src/modules/cs/static/emoji/pensive-face.png | Bin 0 -> 3672 bytes .../cs/static/emoji/persevering-face.png | Bin 0 -> 3628 bytes src/modules/cs/static/emoji/pouting-face.png | Bin 0 -> 3643 bytes src/modules/cs/static/emoji/relieved-face.png | Bin 0 -> 3716 bytes .../emoji/rolling-on-the-floor-laughing.png | Bin 0 -> 3698 bytes ...rious-face-with-symbols-covering-mouth.png | Bin 0 -> 3580 bytes .../shocked-face-with-exploding-head.png | Bin 0 -> 4401 bytes src/modules/cs/static/emoji/sleeping-face.png | Bin 0 -> 3525 bytes src/modules/cs/static/emoji/sleepy-face.png | Bin 0 -> 3477 bytes .../static/emoji/slightly-frowning-face.png | Bin 0 -> 3675 bytes .../cs/static/emoji/slightly-smiling-face.png | Bin 0 -> 3737 bytes .../static/emoji/smiling-face-with-halo.png | Bin 0 -> 3981 bytes .../smiling-face-with-heart-shaped-eyes.png | Bin 0 -> 3690 bytes .../static/emoji/smiling-face-with-horns.png | Bin 0 -> 3957 bytes ...-face-with-open-mouth-and-smiling-eyes.png | Bin 0 -> 3710 bytes ...ith-open-mouth-and-tightly-closed-eyes.png | Bin 0 -> 3746 bytes .../emoji/smiling-face-with-open-mouth.png | Bin 0 -> 3581 bytes ...h-smiling-eyes-and-hand-covering-mouth.png | Bin 0 -> 3807 bytes ...ace-with-smiling-eyes-and-three-hearts.png | Bin 0 -> 3986 bytes .../emoji/smiling-face-with-smiling-eyes.png | Bin 0 -> 3684 bytes .../emoji/smiling-face-with-sunglasses.png | Bin 0 -> 3633 bytes src/modules/cs/static/emoji/smirking-face.png | Bin 0 -> 3566 bytes src/modules/cs/static/emoji/sneezing-face.png | Bin 0 -> 3686 bytes src/modules/cs/static/emoji/thinking-face.png | Bin 0 -> 3763 bytes src/modules/cs/static/emoji/tired-face.png | Bin 0 -> 3683 bytes .../cs/static/emoji/upside-down-face.png | Bin 0 -> 3688 bytes src/modules/cs/static/emoji/weary-face.png | Bin 0 -> 3726 bytes .../cs/static/emoji/white-frowning-face.png | Bin 0 -> 3590 bytes .../cs/static/emoji/white-smiling-face.png | Bin 0 -> 3576 bytes src/modules/cs/static/emoji/winking-face.png | Bin 0 -> 3742 bytes src/modules/cs/static/emoji/worried-face.png | Bin 0 -> 3655 bytes .../cs/static/emoji/zipper-mouth-face.png | Bin 0 -> 3747 bytes src/modules/cs/types/index.d.ts | 13 + src/modules/cs/utils/index.ts | 52 + src/modules/demo/config.ts | 7 + src/modules/demo/directives/color.ts | 5 + src/modules/demo/service/test.ts | 154 + src/modules/demo/service/user/follow.ts | 6 + src/modules/demo/service/user/info.ts | 33 + .../views/crud/components/adv-search/base.vue | 138 + .../crud/components/adv-search/custom.vue | 152 + .../demo/views/crud/components/code.vue | 41 + .../demo/views/crud/components/crud/all.vue | 574 ++ .../demo/views/crud/components/crud/base.vue | 145 + .../demo/views/crud/components/crud/dict.vue | 164 + .../demo/views/crud/components/crud/event.vue | 182 + .../views/crud/components/crud/service.vue | 185 + .../views/crud/components/form/children.vue | 118 + .../crud/components/form/component/index.vue | 124 + .../form/component/select-labels.vue | 38 + .../form/component/select-status.vue | 43 + .../components/form/component/select-work.vue | 59 + .../form/component/select-work2.vue | 38 + .../views/crud/components/form/config.vue | 122 + .../demo/views/crud/components/form/crud.vue | 154 + .../views/crud/components/form/disabled.vue | 64 + .../demo/views/crud/components/form/event.vue | 93 + .../demo/views/crud/components/form/group.vue | 105 + .../views/crud/components/form/hidden.vue | 77 + .../views/crud/components/form/layout.vue | 98 + .../demo/views/crud/components/form/open.vue | 83 + .../views/crud/components/form/options.vue | 172 + .../crud/components/form/plugin/index.vue | 109 + .../views/crud/components/form/plugin/role.ts | 20 + .../views/crud/components/form/required.vue | 75 + .../demo/views/crud/components/form/rules.vue | 123 + .../crud/components/other/select-user.vue | 287 + .../demo/views/crud/components/other/tips.vue | 168 + .../crud/components/other/tsx/index.scss | 28 + .../views/crud/components/other/tsx/index.tsx | 107 + .../views/crud/components/search/base.vue | 143 + .../views/crud/components/search/custom.vue | 176 + .../views/crud/components/search/layout.vue | 152 + .../demo/views/crud/components/table/base.vue | 109 + .../views/crud/components/table/children.vue | 98 + .../crud/components/table/column-custom.vue | 108 + .../crud/components/table/component/index.vue | 108 + .../components/table/component/user-info.vue | 30 + .../crud/components/table/context-menu.vue | 191 + .../demo/views/crud/components/table/dict.vue | 156 + .../views/crud/components/table/hidden.vue | 127 + .../demo/views/crud/components/table/op.vue | 163 + .../crud/components/table/plugin/column.tsx | 48 + .../crud/components/table/plugin/index.vue | 86 + .../views/crud/components/table/search.vue | 143 + .../views/crud/components/table/selection.vue | 110 + .../demo/views/crud/components/table/slot.vue | 97 + .../crud/components/table/span-method.vue | 116 + .../views/crud/components/table/summary.vue | 95 + .../views/crud/components/upsert/base.vue | 133 + .../views/crud/components/upsert/event.vue | 210 + .../crud/components/upsert/hook/index.vue | 200 + .../crud/components/upsert/hook/reg-pca2.ts | 14 + .../views/crud/components/upsert/mode.vue | 146 + src/modules/demo/views/crud/index.vue | 292 + .../views/home/components/count-category.vue | 128 + .../views/home/components/count-effect.vue | 100 + .../demo/views/home/components/count-paid.vue | 125 + .../demo/views/home/components/count-user.vue | 84 + .../demo/views/home/components/goods-rank.vue | 93 + .../demo/views/home/components/tab-chart.vue | 200 + src/modules/demo/views/home/index.vue | 114 + src/modules/dict/config.ts | 13 + src/modules/dict/index.ts | 7 + src/modules/dict/store/dict.ts | 67 + src/modules/dict/store/index.ts | 9 + src/modules/dict/types/index.d.ts | 13 + src/modules/dict/utils/index.ts | 17 + src/modules/dict/views/list.vue | 295 + src/modules/goods/components/select.vue | 371 ++ src/modules/goods/components/spec.vue | 322 + src/modules/goods/views/comment.vue | 72 + src/modules/goods/views/info.vue | 276 + src/modules/goods/views/search-keyword.vue | 85 + src/modules/goods/views/type.vue | 162 + .../helper/components/auto-menu/btn.vue | 43 + .../helper/components/auto-menu/index.vue | 8 + .../helper/components/auto-menu/quick.vue | 255 + .../helper/components/auto-perms/index.vue | 206 + src/modules/helper/config.ts | 27 + src/modules/helper/dict/index.ts | 3 + src/modules/helper/hooks/ai.ts | 102 + src/modules/helper/hooks/code.ts | 454 ++ src/modules/helper/hooks/index.ts | 3 + src/modules/helper/hooks/menu.ts | 85 + src/modules/helper/static/index.scss | 109 + src/modules/helper/static/svg/arrow-down.svg | 1 + src/modules/helper/static/svg/code.svg | 1 + src/modules/helper/static/svg/enter.svg | 1 + src/modules/helper/static/svg/icon-vue.svg | 18 + src/modules/helper/types/index.d.ts | 50 + src/modules/helper/utils/index.ts | 52 + src/modules/helper/views/ai-code.vue | 1631 +++++ src/modules/helper/views/plugins/serve.vue | 351 ++ src/modules/helper/views/plugins/vue.vue | 146 + src/modules/info/views/banner.vue | 137 + src/modules/market/views/coupon/info.vue | 282 + src/modules/market/views/coupon/user.vue | 129 + .../order/components/discount-item.vue | 45 + src/modules/order/components/goods.vue | 145 + src/modules/order/components/info-item.vue | 24 + .../order/components/logistics-item.vue | 229 + src/modules/order/components/pay-item.vue | 20 + src/modules/order/components/remark-item.vue | 26 + src/modules/order/components/user-item.vue | 20 + src/modules/order/dict/index.ts | 16 + src/modules/order/views/info.vue | 165 + src/modules/order/views/refund.vue | 272 + src/modules/recycle/views/data.vue | 125 + src/modules/space/components/space-inner.vue | 447 ++ src/modules/space/components/space.vue | 165 + src/modules/space/config.ts | 20 + src/modules/space/hooks/index.ts | 19 + src/modules/space/views/list.vue | 7 + src/modules/task/components/logs.vue | 113 + src/modules/task/views/list.vue | 502 ++ src/modules/user/components/select.vue | 373 ++ src/modules/user/config.ts | 5 + src/modules/user/views/list.vue | 216 + .../crud/components/column-custom/index.vue | 196 + src/plugins/crud/components/date/picker.vue | 175 + src/plugins/crud/components/date/text.vue | 30 + src/plugins/crud/components/select/index.tsx | 143 + src/plugins/crud/components/switch/index.tsx | 111 + src/plugins/crud/config.ts | 51 + src/plugins/distpicker/components/index.tsx | 40 + src/plugins/distpicker/config.ts | 36 + src/plugins/distpicker/data/pca.json | 5300 +++++++++++++++++ src/plugins/distpicker/demo/base.vue | 10 + src/plugins/echarts/config.ts | 16 + .../editor-monaco/components/monaco.vue | 216 + src/plugins/editor-monaco/config.ts | 23 + src/plugins/editor-monaco/demo/base.vue | 28 + src/plugins/editor-monaco/index.ts | 2 + src/plugins/editor-monaco/utils/config.ts | 28 + src/plugins/editor-monaco/utils/declare.ts | 32 + src/plugins/editor-monaco/utils/format.ts | 65 + src/plugins/editor-monaco/utils/theme.ts | 364 ++ src/plugins/editor-monaco/utils/worker.ts | 23 + .../editor-preview/components/preview.vue | 215 + src/plugins/editor-preview/config.ts | 19 + src/plugins/editor-preview/demo/base.vue | 29 + src/plugins/editor-wang/components/wang.vue | 255 + src/plugins/editor-wang/config.ts | 19 + src/plugins/editor-wang/demo/base.vue | 11 + src/plugins/element-ui/config.ts | 17 + src/plugins/element-ui/css/index.scss | 38 + src/plugins/excel/components/export-btn.tsx | 206 + src/plugins/excel/components/import-btn.vue | 385 ++ src/plugins/excel/config.ts | 22 + src/plugins/excel/demo/base.vue | 80 + src/plugins/excel/utils/index.ts | 227 + src/plugins/theme/components/theme.vue | 204 + src/plugins/theme/config.ts | 40 + src/plugins/theme/static/css/index.scss | 197 + src/plugins/theme/types/index.d.ts | 6 + src/plugins/theme/utils/index.ts | 117 + .../upload/components/upload-item/index.vue | 442 ++ .../upload/components/upload-item/viewer.vue | 103 + src/plugins/upload/components/upload.vue | 629 ++ src/plugins/upload/config.ts | 133 + src/plugins/upload/demo/base.vue | 9 + src/plugins/upload/demo/check.vue | 17 + src/plugins/upload/demo/custom.vue | 27 + src/plugins/upload/demo/drag.vue | 11 + src/plugins/upload/demo/file.vue | 9 + src/plugins/upload/demo/multiple.vue | 9 + src/plugins/upload/demo/small.vue | 9 + src/plugins/upload/demo/space.vue | 66 + src/plugins/upload/hooks/index.ts | 180 + src/plugins/upload/static/svg/audio.svg | 30 + src/plugins/upload/static/svg/excel.svg | 30 + src/plugins/upload/static/svg/file.svg | 48 + src/plugins/upload/static/svg/image.svg | 30 + src/plugins/upload/static/svg/pdf.svg | 30 + src/plugins/upload/static/svg/ppt.svg | 30 + src/plugins/upload/static/svg/rar.svg | 71 + src/plugins/upload/static/svg/video.svg | 26 + src/plugins/upload/static/svg/word.svg | 30 + src/plugins/upload/types/index.d.ts | 24 + src/plugins/upload/utils/index.ts | 76 + src/plugins/view/components/group.vue | 643 ++ src/plugins/view/components/head.vue | 52 + src/plugins/view/config.ts | 23 + src/plugins/view/demo/group.vue | 19 + src/plugins/view/demo/head.vue | 3 + src/plugins/view/hooks/group.ts | 14 + src/plugins/view/hooks/index.ts | 1 + src/plugins/view/index.ts | 2 + src/plugins/view/types/index.d.ts | 69 + src/shims-vue.d.ts | 8 + src/vite-env.d.ts | 10 + stats.html | 4842 +++++++++++++++ tsconfig.json | 24 + vite.config.mts | 99 + 471 files changed, 47757 insertions(+) create mode 100644 .dockerignore create mode 100644 .editorconfig create mode 100644 .env create mode 100644 .eslintignore create mode 100644 .eslintrc.js create mode 100644 .gitattributes create mode 100644 .gitignore create mode 100644 .hintrc create mode 100644 .prettierrc create mode 100644 .vscode/config.code-snippets create mode 100644 .vscode/crud.code-snippets create mode 100644 .vscode/settings.json create mode 100644 Dockerfile create mode 100644 LICENSE create mode 100644 README.md create mode 100644 build/cool/eps.d.ts create mode 100644 build/cool/eps.json create mode 100644 index.html create mode 100644 nginx.conf create mode 100644 package.json create mode 100644 public/favicon.ico create mode 100644 public/logo.png create mode 100644 public/用户导入模版.xlsx create mode 100644 src/App.vue create mode 100644 src/config/dev.ts create mode 100644 src/config/index.ts create mode 100644 src/config/prod.ts create mode 100644 src/config/proxy.ts create mode 100644 src/cool/bootstrap/eps.ts create mode 100644 src/cool/bootstrap/index.ts create mode 100644 src/cool/bootstrap/module.ts create mode 100644 src/cool/hooks/browser.ts create mode 100644 src/cool/hooks/hmr.ts create mode 100644 src/cool/hooks/index.ts create mode 100644 src/cool/hooks/mitt.ts create mode 100644 src/cool/index.ts create mode 100644 src/cool/module/index.ts create mode 100644 src/cool/router/index.ts create mode 100644 src/cool/service/base.ts create mode 100644 src/cool/service/index.ts create mode 100644 src/cool/service/request.ts create mode 100644 src/cool/types/index.ts create mode 100644 src/cool/utils/index.ts create mode 100644 src/cool/utils/loading.ts create mode 100644 src/cool/utils/storage.ts create mode 100644 src/main.ts create mode 100644 src/modules/app/views/complain.vue create mode 100644 src/modules/app/views/feedback.vue create mode 100644 src/modules/app/views/goods.vue create mode 100644 src/modules/app/views/version.vue create mode 100644 src/modules/base/common/index.ts create mode 100644 src/modules/base/common/permission.ts create mode 100644 src/modules/base/common/theme.ts create mode 100644 src/modules/base/components/avatar/index.tsx create mode 100644 src/modules/base/components/code/json.vue create mode 100644 src/modules/base/components/dept/check.vue create mode 100644 src/modules/base/components/dept/select.vue create mode 100644 src/modules/base/components/editor/index.tsx create mode 100644 src/modules/base/components/icon/svg.vue create mode 100644 src/modules/base/components/image/index.vue create mode 100644 src/modules/base/components/link/index.vue create mode 100644 src/modules/base/components/menu/check.vue create mode 100644 src/modules/base/components/menu/file.vue create mode 100644 src/modules/base/components/menu/icon.vue create mode 100644 src/modules/base/components/menu/perms.vue create mode 100644 src/modules/base/components/menu/select.vue create mode 100644 src/modules/base/config.ts create mode 100644 src/modules/base/directives/permission.ts create mode 100644 src/modules/base/hooks/dept.tsx create mode 100644 src/modules/base/hooks/index.ts create mode 100644 src/modules/base/index.ts create mode 100644 src/modules/base/pages/error/401.vue create mode 100644 src/modules/base/pages/error/403.vue create mode 100644 src/modules/base/pages/error/404.vue create mode 100644 src/modules/base/pages/error/500.vue create mode 100644 src/modules/base/pages/error/502.vue create mode 100644 src/modules/base/pages/error/components/error-page.vue create mode 100644 src/modules/base/pages/login/components/pic-captcha.vue create mode 100644 src/modules/base/pages/login/index.vue create mode 100644 src/modules/base/pages/login/static/bg.svg create mode 100644 src/modules/base/pages/main/components/amenu.vue create mode 100644 src/modules/base/pages/main/components/bmenu.tsx create mode 100644 src/modules/base/pages/main/components/process.vue create mode 100644 src/modules/base/pages/main/components/route-nav.vue create mode 100644 src/modules/base/pages/main/components/slider.vue create mode 100644 src/modules/base/pages/main/components/topbar.vue create mode 100644 src/modules/base/pages/main/components/views.vue create mode 100644 src/modules/base/pages/main/index.vue create mode 100644 src/modules/base/static/css/animation.scss create mode 100644 src/modules/base/static/css/index.scss create mode 100644 src/modules/base/static/svg/icon-activity.svg create mode 100644 src/modules/base/static/svg/icon-amount.svg create mode 100644 src/modules/base/static/svg/icon-app.svg create mode 100644 src/modules/base/static/svg/icon-approve.svg create mode 100644 src/modules/base/static/svg/icon-auth.svg create mode 100644 src/modules/base/static/svg/icon-ban.svg create mode 100644 src/modules/base/static/svg/icon-camera.svg create mode 100644 src/modules/base/static/svg/icon-card.svg create mode 100644 src/modules/base/static/svg/icon-cart.svg create mode 100644 src/modules/base/static/svg/icon-command.svg create mode 100644 src/modules/base/static/svg/icon-common.svg create mode 100644 src/modules/base/static/svg/icon-component.svg create mode 100644 src/modules/base/static/svg/icon-count.svg create mode 100644 src/modules/base/static/svg/icon-crown.svg create mode 100644 src/modules/base/static/svg/icon-data.svg create mode 100644 src/modules/base/static/svg/icon-db.svg create mode 100644 src/modules/base/static/svg/icon-delete.svg create mode 100644 src/modules/base/static/svg/icon-dept.svg create mode 100644 src/modules/base/static/svg/icon-design.svg create mode 100644 src/modules/base/static/svg/icon-device.svg create mode 100644 src/modules/base/static/svg/icon-dict.svg create mode 100644 src/modules/base/static/svg/icon-discover.svg create mode 100644 src/modules/base/static/svg/icon-emoji.svg create mode 100644 src/modules/base/static/svg/icon-favor.svg create mode 100644 src/modules/base/static/svg/icon-fx.svg create mode 100644 src/modules/base/static/svg/icon-goods.svg create mode 100644 src/modules/base/static/svg/icon-home.svg create mode 100644 src/modules/base/static/svg/icon-hot.svg create mode 100644 src/modules/base/static/svg/icon-info.svg create mode 100644 src/modules/base/static/svg/icon-iot.svg create mode 100644 src/modules/base/static/svg/icon-like.svg create mode 100644 src/modules/base/static/svg/icon-living.svg create mode 100644 src/modules/base/static/svg/icon-log.svg create mode 100644 src/modules/base/static/svg/icon-map.svg create mode 100644 src/modules/base/static/svg/icon-menu.svg create mode 100644 src/modules/base/static/svg/icon-message.svg create mode 100644 src/modules/base/static/svg/icon-monitor.svg create mode 100644 src/modules/base/static/svg/icon-new.svg create mode 100644 src/modules/base/static/svg/icon-news.svg create mode 100644 src/modules/base/static/svg/icon-notice.svg create mode 100644 src/modules/base/static/svg/icon-params.svg create mode 100644 src/modules/base/static/svg/icon-pending.svg create mode 100644 src/modules/base/static/svg/icon-phone.svg create mode 100644 src/modules/base/static/svg/icon-pic.svg create mode 100644 src/modules/base/static/svg/icon-question.svg create mode 100644 src/modules/base/static/svg/icon-rank.svg create mode 100644 src/modules/base/static/svg/icon-scan.svg create mode 100644 src/modules/base/static/svg/icon-search.svg create mode 100644 src/modules/base/static/svg/icon-system.svg create mode 100644 src/modules/base/static/svg/icon-tag.svg create mode 100644 src/modules/base/static/svg/icon-task.svg create mode 100644 src/modules/base/static/svg/icon-theme.svg create mode 100644 src/modules/base/static/svg/icon-time.svg create mode 100644 src/modules/base/static/svg/icon-track.svg create mode 100644 src/modules/base/static/svg/icon-upload.svg create mode 100644 src/modules/base/static/svg/icon-user.svg create mode 100644 src/modules/base/static/svg/icon-video.svg create mode 100644 src/modules/base/static/svg/icon-warn.svg create mode 100644 src/modules/base/static/svg/icon-workbench.svg create mode 100644 src/modules/base/store/app.ts create mode 100644 src/modules/base/store/index.ts create mode 100644 src/modules/base/store/menu.ts create mode 100644 src/modules/base/store/process.ts create mode 100644 src/modules/base/store/user.ts create mode 100644 src/modules/base/types/index.d.ts create mode 100644 src/modules/base/utils/index.ts create mode 100644 src/modules/base/views/frame.vue create mode 100644 src/modules/base/views/info.vue create mode 100644 src/modules/base/views/log.vue create mode 100644 src/modules/base/views/menu/components/exp.vue create mode 100644 src/modules/base/views/menu/components/imp.vue create mode 100644 src/modules/base/views/menu/index.vue create mode 100644 src/modules/base/views/param.vue create mode 100644 src/modules/base/views/role.vue create mode 100644 src/modules/base/views/user/components/dept-list.vue create mode 100644 src/modules/base/views/user/components/user-move.vue create mode 100644 src/modules/base/views/user/index.vue create mode 100644 src/modules/cs/components/index.vue create mode 100644 src/modules/cs/components/message.vue create mode 100644 src/modules/cs/components/session.vue create mode 100644 src/modules/cs/components/tools/emoji.vue create mode 100644 src/modules/cs/config.ts create mode 100644 src/modules/cs/hooks/index.ts create mode 100644 src/modules/cs/hooks/message.ts create mode 100644 src/modules/cs/hooks/notice.tsx create mode 100644 src/modules/cs/hooks/session.ts create mode 100644 src/modules/cs/hooks/socket.ts create mode 100644 src/modules/cs/hooks/tools.ts create mode 100644 src/modules/cs/index.ts create mode 100644 src/modules/cs/static/audio/ding.wav create mode 100644 src/modules/cs/static/emoji/angry-face.png create mode 100644 src/modules/cs/static/emoji/anguished-face.png create mode 100644 src/modules/cs/static/emoji/astonished-face.png create mode 100644 src/modules/cs/static/emoji/confounded-face.png create mode 100644 src/modules/cs/static/emoji/confused-face.png create mode 100644 src/modules/cs/static/emoji/crying-face.png create mode 100644 src/modules/cs/static/emoji/disappointed-but-relieved-face.png create mode 100644 src/modules/cs/static/emoji/disappointed-face.png create mode 100644 src/modules/cs/static/emoji/dizzy-face.png create mode 100644 src/modules/cs/static/emoji/drooling-face.png create mode 100644 src/modules/cs/static/emoji/expressionless-face.png create mode 100644 src/modules/cs/static/emoji/face-savouring-delicious-food.png create mode 100644 src/modules/cs/static/emoji/face-screaming-in-fear.png create mode 100644 src/modules/cs/static/emoji/face-throwing-a-kiss.png create mode 100644 src/modules/cs/static/emoji/face-with-cold-sweat.png create mode 100644 src/modules/cs/static/emoji/face-with-cowboy-hat.png create mode 100644 src/modules/cs/static/emoji/face-with-finger-covering-closed-lips.png create mode 100644 src/modules/cs/static/emoji/face-with-head-bandage.png create mode 100644 src/modules/cs/static/emoji/face-with-look-of-triumph.png create mode 100644 src/modules/cs/static/emoji/face-with-medical-mask.png create mode 100644 src/modules/cs/static/emoji/face-with-monocle.png create mode 100644 src/modules/cs/static/emoji/face-with-one-eyebrow-raised.png create mode 100644 src/modules/cs/static/emoji/face-with-open-mouth-and-cold-sweat.png create mode 100644 src/modules/cs/static/emoji/face-with-open-mouth-vomiting.png create mode 100644 src/modules/cs/static/emoji/face-with-open-mouth.png create mode 100644 src/modules/cs/static/emoji/face-with-party-horn-and-party-hat.png create mode 100644 src/modules/cs/static/emoji/face-with-pleading-eyes.png create mode 100644 src/modules/cs/static/emoji/face-with-rolling-eyes.png create mode 100644 src/modules/cs/static/emoji/face-with-stuck-out-tongue-and-tightly-closed-eyes.png create mode 100644 src/modules/cs/static/emoji/face-with-stuck-out-tongue-and-winking-eye.png create mode 100644 src/modules/cs/static/emoji/face-with-stuck-out-tongue.png create mode 100644 src/modules/cs/static/emoji/face-with-thermometer.png create mode 100644 src/modules/cs/static/emoji/face-with-uneven-eyes-and-wavy-mouth.png create mode 100644 src/modules/cs/static/emoji/face-without-mouth.png create mode 100644 src/modules/cs/static/emoji/fearful-face.png create mode 100644 src/modules/cs/static/emoji/flushed-face.png create mode 100644 src/modules/cs/static/emoji/freezing-face.png create mode 100644 src/modules/cs/static/emoji/frowning-face-with-open-mouth.png create mode 100644 src/modules/cs/static/emoji/grimacing-face.png create mode 100644 src/modules/cs/static/emoji/grinning-face-with-one-large-and-one-small-eye.png create mode 100644 src/modules/cs/static/emoji/grinning-face-with-smiling-eyes.png create mode 100644 src/modules/cs/static/emoji/grinning-face-with-star-eyes.png create mode 100644 src/modules/cs/static/emoji/grinning-face.png create mode 100644 src/modules/cs/static/emoji/hugging-face.png create mode 100644 src/modules/cs/static/emoji/hushed-face.png create mode 100644 src/modules/cs/static/emoji/imp.png create mode 100644 src/modules/cs/static/emoji/kissing-face-with-closed-eyes.png create mode 100644 src/modules/cs/static/emoji/kissing-face-with-smiling-eyes.png create mode 100644 src/modules/cs/static/emoji/kissing-face.png create mode 100644 src/modules/cs/static/emoji/loudly-crying-face.png create mode 100644 src/modules/cs/static/emoji/lying-face.png create mode 100644 src/modules/cs/static/emoji/money-mouth-face.png create mode 100644 src/modules/cs/static/emoji/nauseated-face.png create mode 100644 src/modules/cs/static/emoji/nerd-face.png create mode 100644 src/modules/cs/static/emoji/neutral-face.png create mode 100644 src/modules/cs/static/emoji/overheated-face.png create mode 100644 src/modules/cs/static/emoji/pensive-face.png create mode 100644 src/modules/cs/static/emoji/persevering-face.png create mode 100644 src/modules/cs/static/emoji/pouting-face.png create mode 100644 src/modules/cs/static/emoji/relieved-face.png create mode 100644 src/modules/cs/static/emoji/rolling-on-the-floor-laughing.png create mode 100644 src/modules/cs/static/emoji/serious-face-with-symbols-covering-mouth.png create mode 100644 src/modules/cs/static/emoji/shocked-face-with-exploding-head.png create mode 100644 src/modules/cs/static/emoji/sleeping-face.png create mode 100644 src/modules/cs/static/emoji/sleepy-face.png create mode 100644 src/modules/cs/static/emoji/slightly-frowning-face.png create mode 100644 src/modules/cs/static/emoji/slightly-smiling-face.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-halo.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-heart-shaped-eyes.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-horns.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-open-mouth-and-smiling-eyes.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-open-mouth-and-tightly-closed-eyes.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-open-mouth.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-smiling-eyes-and-hand-covering-mouth.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-smiling-eyes-and-three-hearts.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-smiling-eyes.png create mode 100644 src/modules/cs/static/emoji/smiling-face-with-sunglasses.png create mode 100644 src/modules/cs/static/emoji/smirking-face.png create mode 100644 src/modules/cs/static/emoji/sneezing-face.png create mode 100644 src/modules/cs/static/emoji/thinking-face.png create mode 100644 src/modules/cs/static/emoji/tired-face.png create mode 100644 src/modules/cs/static/emoji/upside-down-face.png create mode 100644 src/modules/cs/static/emoji/weary-face.png create mode 100644 src/modules/cs/static/emoji/white-frowning-face.png create mode 100644 src/modules/cs/static/emoji/white-smiling-face.png create mode 100644 src/modules/cs/static/emoji/winking-face.png create mode 100644 src/modules/cs/static/emoji/worried-face.png create mode 100644 src/modules/cs/static/emoji/zipper-mouth-face.png create mode 100644 src/modules/cs/types/index.d.ts create mode 100644 src/modules/cs/utils/index.ts create mode 100644 src/modules/demo/config.ts create mode 100644 src/modules/demo/directives/color.ts create mode 100644 src/modules/demo/service/test.ts create mode 100644 src/modules/demo/service/user/follow.ts create mode 100644 src/modules/demo/service/user/info.ts create mode 100644 src/modules/demo/views/crud/components/adv-search/base.vue create mode 100644 src/modules/demo/views/crud/components/adv-search/custom.vue create mode 100644 src/modules/demo/views/crud/components/code.vue create mode 100644 src/modules/demo/views/crud/components/crud/all.vue create mode 100644 src/modules/demo/views/crud/components/crud/base.vue create mode 100644 src/modules/demo/views/crud/components/crud/dict.vue create mode 100644 src/modules/demo/views/crud/components/crud/event.vue create mode 100644 src/modules/demo/views/crud/components/crud/service.vue create mode 100644 src/modules/demo/views/crud/components/form/children.vue create mode 100644 src/modules/demo/views/crud/components/form/component/index.vue create mode 100644 src/modules/demo/views/crud/components/form/component/select-labels.vue create mode 100644 src/modules/demo/views/crud/components/form/component/select-status.vue create mode 100644 src/modules/demo/views/crud/components/form/component/select-work.vue create mode 100644 src/modules/demo/views/crud/components/form/component/select-work2.vue create mode 100644 src/modules/demo/views/crud/components/form/config.vue create mode 100644 src/modules/demo/views/crud/components/form/crud.vue create mode 100644 src/modules/demo/views/crud/components/form/disabled.vue create mode 100644 src/modules/demo/views/crud/components/form/event.vue create mode 100644 src/modules/demo/views/crud/components/form/group.vue create mode 100644 src/modules/demo/views/crud/components/form/hidden.vue create mode 100644 src/modules/demo/views/crud/components/form/layout.vue create mode 100644 src/modules/demo/views/crud/components/form/open.vue create mode 100644 src/modules/demo/views/crud/components/form/options.vue create mode 100644 src/modules/demo/views/crud/components/form/plugin/index.vue create mode 100644 src/modules/demo/views/crud/components/form/plugin/role.ts create mode 100644 src/modules/demo/views/crud/components/form/required.vue create mode 100644 src/modules/demo/views/crud/components/form/rules.vue create mode 100644 src/modules/demo/views/crud/components/other/select-user.vue create mode 100644 src/modules/demo/views/crud/components/other/tips.vue create mode 100644 src/modules/demo/views/crud/components/other/tsx/index.scss create mode 100644 src/modules/demo/views/crud/components/other/tsx/index.tsx create mode 100644 src/modules/demo/views/crud/components/search/base.vue create mode 100644 src/modules/demo/views/crud/components/search/custom.vue create mode 100644 src/modules/demo/views/crud/components/search/layout.vue create mode 100644 src/modules/demo/views/crud/components/table/base.vue create mode 100644 src/modules/demo/views/crud/components/table/children.vue create mode 100644 src/modules/demo/views/crud/components/table/column-custom.vue create mode 100644 src/modules/demo/views/crud/components/table/component/index.vue create mode 100644 src/modules/demo/views/crud/components/table/component/user-info.vue create mode 100644 src/modules/demo/views/crud/components/table/context-menu.vue create mode 100644 src/modules/demo/views/crud/components/table/dict.vue create mode 100644 src/modules/demo/views/crud/components/table/hidden.vue create mode 100644 src/modules/demo/views/crud/components/table/op.vue create mode 100644 src/modules/demo/views/crud/components/table/plugin/column.tsx create mode 100644 src/modules/demo/views/crud/components/table/plugin/index.vue create mode 100644 src/modules/demo/views/crud/components/table/search.vue create mode 100644 src/modules/demo/views/crud/components/table/selection.vue create mode 100644 src/modules/demo/views/crud/components/table/slot.vue create mode 100644 src/modules/demo/views/crud/components/table/span-method.vue create mode 100644 src/modules/demo/views/crud/components/table/summary.vue create mode 100644 src/modules/demo/views/crud/components/upsert/base.vue create mode 100644 src/modules/demo/views/crud/components/upsert/event.vue create mode 100644 src/modules/demo/views/crud/components/upsert/hook/index.vue create mode 100644 src/modules/demo/views/crud/components/upsert/hook/reg-pca2.ts create mode 100644 src/modules/demo/views/crud/components/upsert/mode.vue create mode 100644 src/modules/demo/views/crud/index.vue create mode 100644 src/modules/demo/views/home/components/count-category.vue create mode 100644 src/modules/demo/views/home/components/count-effect.vue create mode 100644 src/modules/demo/views/home/components/count-paid.vue create mode 100644 src/modules/demo/views/home/components/count-user.vue create mode 100644 src/modules/demo/views/home/components/goods-rank.vue create mode 100644 src/modules/demo/views/home/components/tab-chart.vue create mode 100644 src/modules/demo/views/home/index.vue create mode 100644 src/modules/dict/config.ts create mode 100644 src/modules/dict/index.ts create mode 100644 src/modules/dict/store/dict.ts create mode 100644 src/modules/dict/store/index.ts create mode 100644 src/modules/dict/types/index.d.ts create mode 100644 src/modules/dict/utils/index.ts create mode 100644 src/modules/dict/views/list.vue create mode 100644 src/modules/goods/components/select.vue create mode 100644 src/modules/goods/components/spec.vue create mode 100644 src/modules/goods/views/comment.vue create mode 100644 src/modules/goods/views/info.vue create mode 100644 src/modules/goods/views/search-keyword.vue create mode 100644 src/modules/goods/views/type.vue create mode 100644 src/modules/helper/components/auto-menu/btn.vue create mode 100644 src/modules/helper/components/auto-menu/index.vue create mode 100644 src/modules/helper/components/auto-menu/quick.vue create mode 100644 src/modules/helper/components/auto-perms/index.vue create mode 100644 src/modules/helper/config.ts create mode 100644 src/modules/helper/dict/index.ts create mode 100644 src/modules/helper/hooks/ai.ts create mode 100644 src/modules/helper/hooks/code.ts create mode 100644 src/modules/helper/hooks/index.ts create mode 100644 src/modules/helper/hooks/menu.ts create mode 100644 src/modules/helper/static/index.scss create mode 100644 src/modules/helper/static/svg/arrow-down.svg create mode 100644 src/modules/helper/static/svg/code.svg create mode 100644 src/modules/helper/static/svg/enter.svg create mode 100644 src/modules/helper/static/svg/icon-vue.svg create mode 100644 src/modules/helper/types/index.d.ts create mode 100644 src/modules/helper/utils/index.ts create mode 100644 src/modules/helper/views/ai-code.vue create mode 100644 src/modules/helper/views/plugins/serve.vue create mode 100644 src/modules/helper/views/plugins/vue.vue create mode 100644 src/modules/info/views/banner.vue create mode 100644 src/modules/market/views/coupon/info.vue create mode 100644 src/modules/market/views/coupon/user.vue create mode 100644 src/modules/order/components/discount-item.vue create mode 100644 src/modules/order/components/goods.vue create mode 100644 src/modules/order/components/info-item.vue create mode 100644 src/modules/order/components/logistics-item.vue create mode 100644 src/modules/order/components/pay-item.vue create mode 100644 src/modules/order/components/remark-item.vue create mode 100644 src/modules/order/components/user-item.vue create mode 100644 src/modules/order/dict/index.ts create mode 100644 src/modules/order/views/info.vue create mode 100644 src/modules/order/views/refund.vue create mode 100644 src/modules/recycle/views/data.vue create mode 100644 src/modules/space/components/space-inner.vue create mode 100644 src/modules/space/components/space.vue create mode 100644 src/modules/space/config.ts create mode 100644 src/modules/space/hooks/index.ts create mode 100644 src/modules/space/views/list.vue create mode 100644 src/modules/task/components/logs.vue create mode 100644 src/modules/task/views/list.vue create mode 100644 src/modules/user/components/select.vue create mode 100644 src/modules/user/config.ts create mode 100644 src/modules/user/views/list.vue create mode 100644 src/plugins/crud/components/column-custom/index.vue create mode 100644 src/plugins/crud/components/date/picker.vue create mode 100644 src/plugins/crud/components/date/text.vue create mode 100644 src/plugins/crud/components/select/index.tsx create mode 100644 src/plugins/crud/components/switch/index.tsx create mode 100644 src/plugins/crud/config.ts create mode 100644 src/plugins/distpicker/components/index.tsx create mode 100644 src/plugins/distpicker/config.ts create mode 100644 src/plugins/distpicker/data/pca.json create mode 100644 src/plugins/distpicker/demo/base.vue create mode 100644 src/plugins/echarts/config.ts create mode 100644 src/plugins/editor-monaco/components/monaco.vue create mode 100644 src/plugins/editor-monaco/config.ts create mode 100644 src/plugins/editor-monaco/demo/base.vue create mode 100644 src/plugins/editor-monaco/index.ts create mode 100644 src/plugins/editor-monaco/utils/config.ts create mode 100644 src/plugins/editor-monaco/utils/declare.ts create mode 100644 src/plugins/editor-monaco/utils/format.ts create mode 100644 src/plugins/editor-monaco/utils/theme.ts create mode 100644 src/plugins/editor-monaco/utils/worker.ts create mode 100644 src/plugins/editor-preview/components/preview.vue create mode 100644 src/plugins/editor-preview/config.ts create mode 100644 src/plugins/editor-preview/demo/base.vue create mode 100644 src/plugins/editor-wang/components/wang.vue create mode 100644 src/plugins/editor-wang/config.ts create mode 100644 src/plugins/editor-wang/demo/base.vue create mode 100644 src/plugins/element-ui/config.ts create mode 100644 src/plugins/element-ui/css/index.scss create mode 100644 src/plugins/excel/components/export-btn.tsx create mode 100644 src/plugins/excel/components/import-btn.vue create mode 100644 src/plugins/excel/config.ts create mode 100644 src/plugins/excel/demo/base.vue create mode 100644 src/plugins/excel/utils/index.ts create mode 100644 src/plugins/theme/components/theme.vue create mode 100644 src/plugins/theme/config.ts create mode 100644 src/plugins/theme/static/css/index.scss create mode 100644 src/plugins/theme/types/index.d.ts create mode 100644 src/plugins/theme/utils/index.ts create mode 100644 src/plugins/upload/components/upload-item/index.vue create mode 100644 src/plugins/upload/components/upload-item/viewer.vue create mode 100644 src/plugins/upload/components/upload.vue create mode 100644 src/plugins/upload/config.ts create mode 100644 src/plugins/upload/demo/base.vue create mode 100644 src/plugins/upload/demo/check.vue create mode 100644 src/plugins/upload/demo/custom.vue create mode 100644 src/plugins/upload/demo/drag.vue create mode 100644 src/plugins/upload/demo/file.vue create mode 100644 src/plugins/upload/demo/multiple.vue create mode 100644 src/plugins/upload/demo/small.vue create mode 100644 src/plugins/upload/demo/space.vue create mode 100644 src/plugins/upload/hooks/index.ts create mode 100644 src/plugins/upload/static/svg/audio.svg create mode 100644 src/plugins/upload/static/svg/excel.svg create mode 100644 src/plugins/upload/static/svg/file.svg create mode 100644 src/plugins/upload/static/svg/image.svg create mode 100644 src/plugins/upload/static/svg/pdf.svg create mode 100644 src/plugins/upload/static/svg/ppt.svg create mode 100644 src/plugins/upload/static/svg/rar.svg create mode 100644 src/plugins/upload/static/svg/video.svg create mode 100644 src/plugins/upload/static/svg/word.svg create mode 100644 src/plugins/upload/types/index.d.ts create mode 100644 src/plugins/upload/utils/index.ts create mode 100644 src/plugins/view/components/group.vue create mode 100644 src/plugins/view/components/head.vue create mode 100644 src/plugins/view/config.ts create mode 100644 src/plugins/view/demo/group.vue create mode 100644 src/plugins/view/demo/head.vue create mode 100644 src/plugins/view/hooks/group.ts create mode 100644 src/plugins/view/hooks/index.ts create mode 100644 src/plugins/view/index.ts create mode 100644 src/plugins/view/types/index.d.ts create mode 100644 src/shims-vue.d.ts create mode 100644 src/vite-env.d.ts create mode 100644 stats.html create mode 100644 tsconfig.json create mode 100644 vite.config.mts diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 0000000..d451ff1 --- /dev/null +++ b/.dockerignore @@ -0,0 +1,5 @@ +node_modules +.DS_Store +dist +dist-ssr +*.local diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 0000000..db1d3f8 --- /dev/null +++ b/.editorconfig @@ -0,0 +1,11 @@ +# 🎨 editorconfig.org + +root = true + +[*] +charset = utf-8 +end_of_line = lf +indent_style = tab +indent_size = 4 +trim_trailing_whitespace = true +insert_final_newline = true diff --git a/.env b/.env new file mode 100644 index 0000000..ea3effc --- /dev/null +++ b/.env @@ -0,0 +1,5 @@ +# 应用名称 +VITE_NAME = "COOL MALL" + +# 网络超时请求时间 +VITE_TIMEOUT = 30000 \ No newline at end of file diff --git a/.eslintignore b/.eslintignore new file mode 100644 index 0000000..e35264d --- /dev/null +++ b/.eslintignore @@ -0,0 +1,3 @@ +packages/ +dist/ +node_modules/ diff --git a/.eslintrc.js b/.eslintrc.js new file mode 100644 index 0000000..5729661 --- /dev/null +++ b/.eslintrc.js @@ -0,0 +1,47 @@ +/* eslint-env node */ +require("@rushstack/eslint-patch/modern-module-resolution"); + +module.exports = { + root: true, + extends: [ + "plugin:vue/vue3-essential", + "eslint:recommended", + "@vue/eslint-config-typescript", + "@vue/eslint-config-prettier/skip-formatting" + ], + parserOptions: { + ecmaVersion: "latest" + }, + rules: { + "@typescript-eslint/ban-ts-ignore": "off", + "@typescript-eslint/explicit-function-return-type": "off", + "@typescript-eslint/no-explicit-any": "off", + "@typescript-eslint/no-var-requires": "off", + "@typescript-eslint/no-empty-function": "off", + "vue/no-mutating-props": "off", + "vue/component-name-in-template-casing": ["error", "kebab-case"], + "vue/component-definition-name-casing": ["error", "kebab-case"], + "no-use-before-define": "off", + "no-unused-vars": "off", + "@typescript-eslint/no-use-before-define": "off", + "@typescript-eslint/ban-ts-comment": "off", + "@typescript-eslint/ban-types": "off", + "@typescript-eslint/no-non-null-assertion": "off", + "@typescript-eslint/explicit-module-boundary-types": "off", + "@typescript-eslint/no-namespace": "off", + "@typescript-eslint/no-unused-vars": "off", + "space-before-function-paren": "off", + "vue/attributes-order": "off", + "vue/one-component-per-file": "off", + "vue/html-closing-bracket-newline": "off", + "vue/max-attributes-per-line": "off", + "vue/multiline-html-element-content-newline": "off", + "vue/multi-word-component-names": "off", + "vue/singleline-html-element-content-newline": "off", + "vue/attribute-hyphenation": "off", + "vue/html-self-closing": "off", + "vue/require-default-prop": "off", + "vue/v-on-event-hyphenation": "off", + "no-self-assign": "off" + } +}; diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..e41187f --- /dev/null +++ b/.gitattributes @@ -0,0 +1,4 @@ +*.js text eol=lf +*.json text eol=lf +*.ts text eol=lf +*.vue text eol=lf \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..4816e5a --- /dev/null +++ b/.gitignore @@ -0,0 +1,18 @@ +.DS_Store +node_modules/ +/dist/ +dist-ssr/ + +# Log files +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# Editor directories and files +.project +.idea +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw* diff --git a/.hintrc b/.hintrc new file mode 100644 index 0000000..de024c3 --- /dev/null +++ b/.hintrc @@ -0,0 +1,16 @@ +{ + "extends": [ + "development" + ], + "hints": { + "meta-viewport": "off", + "axe/text-alternatives": [ + "default", + { + "document-title": "off" + } + ], + "disown-opener": "off", + "css-prefix-order": "off" + } +} \ No newline at end of file diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..ee44e11 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,8 @@ +{ + "tabWidth": 4, + "useTabs": true, + "semi": true, + "singleQuote": false, + "printWidth": 100, + "trailingComma": "none" +} diff --git a/.vscode/config.code-snippets b/.vscode/config.code-snippets new file mode 100644 index 0000000..4796b30 --- /dev/null +++ b/.vscode/config.code-snippets @@ -0,0 +1,15 @@ +{ + "module-config": { + "prefix": "module-config", + "scope": "typescript", + "body": [ + "import type { ModuleConfig } from \"/@/cool\";", + "", + "export default (): ModuleConfig => {", + " return {};", + "};", + "" + ], + "description": "module config snippets" + } +} diff --git a/.vscode/crud.code-snippets b/.vscode/crud.code-snippets new file mode 100644 index 0000000..01b3dde --- /dev/null +++ b/.vscode/crud.code-snippets @@ -0,0 +1,102 @@ +{ + "cl-crud": { + "prefix": "cl-crud", + "scope": "vue", + "body": [ + "", + "", + "", + "" + ], + "description": "cl-crud snippets" + }, + "cl-filter": { + "prefix": "cl-filter", + "scope": "html", + "body": [ + "", + " ", + "" + ], + "description": "cl-filter snippets" + }, + "item": { + "prefix": "item", + "scope": "typescript", + "body": [ + "{", + " label: \"$1\",", + " prop: \"\",", + " component: {", + " name: \"\"", + " }", + "},", + "" + ], + "description": "item snippets" + }, + "column": { + "prefix": "column", + "scope": "typescript", + "body": ["{", " label: \"$1\",", " prop: \"\",", "},", ""], + "description": "column snippets" + } +} diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..f24801c --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,4 @@ +{ + "editor.cursorSmoothCaretAnimation": "on", + "editor.formatOnSave": true, +} diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..b25069d --- /dev/null +++ b/Dockerfile @@ -0,0 +1,16 @@ +FROM node:lts-alpine +WORKDIR /build +# 设置Node-Sass的镜像地址 +RUN npm config set sass_binary_site=https://npm.taobao.org/mirrors/node-sass/ +# 设置npm镜像 +RUN npm config set registry https://registry.npm.taobao.org +COPY package.json /build/package.json +RUN yarn +COPY ./ /build +RUN npm run build + +FROM nginx +RUN mkdir /app +COPY --from=0 /build/dist /app +COPY --from=0 /build/nginx.conf /etc/nginx/nginx.conf +EXPOSE 80 diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..83140ce --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2021 cool-team-official + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..68e76ff --- /dev/null +++ b/README.md @@ -0,0 +1,57 @@ +# cool-admin [vue3 - ts - vite] + +

+ cool-admin Logo +

+ +

cool-admin 一个很酷的后台权限管理系统,开源免费,模块化、插件化、极速开发 CRUD,方便快速构建迭代后台管理系统, 到文档 进一步了解

+ +

+ GitHub license + GitHub tag + GitHub tag +

+ +## 地址 + +- [📌 v6 vue3 + element-plus + ts + vite](https://github.com/cool-team-official/cool-admin-vue/tree/6.x) + +- [⚡️ v5 vue3 + element-plus + ts + vite](https://github.com/cool-team-official/cool-admin-vue/tree/5.x) + +- [🌐 码云仓库地址](https://gitee.com/cool-team-official/cool-admin-vue) + +## 演示 + +[https://show.cool-admin.com](https://show.cool-admin.com) + +账户:admin,密码:123456 + +Admin Home + +## 项目后端 + +[https://github.com/cool-team-official/cool-admin-midway](https://github.com/cool-team-official/cool-admin-midway) + +## 微信群 + +Admin Wechat + +## 安装项目依赖 + +推荐使用 `yarn`: + +```shell +yarn +``` + +## 运行应用程序 + +安装过程完成后,运行以下命令启动服务。您可以在浏览器中预览网站 [http://localhost:9000](http://localhost:9000) + +```shell +yarn dev +``` + +### 低价服务器 + +[阿里云、腾讯云、华为云低价云服务器,不限新老](https://cool-js.com/ad/server.html) diff --git a/build/cool/eps.d.ts b/build/cool/eps.d.ts new file mode 100644 index 0000000..0e4a8b5 --- /dev/null +++ b/build/cool/eps.d.ts @@ -0,0 +1,3475 @@ +declare namespace Eps { + interface AppComplainEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 类型 + */ + type?: number; + /** + * 联系方式 + */ + contact?: string; + /** + * 内容 + */ + content?: string; + /** + * 图片 + */ + images?: json; + /** + * 状态 0-未处理 1-已处理 + */ + status?: number; + /** + * 处理人ID + */ + handlerId?: number; + /** + * 备注 + */ + remark?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface AppFeedbackEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 联系方式 + */ + contact?: string; + /** + * 类型 + */ + type?: number; + /** + * 内容 + */ + content?: string; + /** + * 图片 + */ + images?: json; + /** + * 状态 0-未处理 1-已处理 + */ + status?: number; + /** + * 处理人ID + */ + handlerId?: number; + /** + * 备注 + */ + remark?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface AppGoodsEntity { + /** + * ID + */ + id?: number; + /** + * 标题 + */ + title?: string; + /** + * 价格 + */ + price?: number; + /** + * 原价 + */ + originalPrice?: number; + /** + * 描述 + */ + description?: string; + /** + * 状态 0-禁用 1-启用 + */ + status?: number; + /** + * 排序 + */ + sort?: number; + /** + * 类型 0-天 1-月 2-年 3-永久 + */ + type?: number; + /** + * 时长 + */ + duration?: number; + /** + * 标签 + */ + tag?: string; + /** + * 标签颜色 + */ + tagColor?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface AppVersionEntity { + /** + * ID + */ + id?: number; + /** + * 名称 + */ + name?: string; + /** + * 版本号 + */ + version?: string; + /** + * 类型 + */ + type?: number; + /** + * 下载地址 + */ + url?: string; + /** + * 强制更新 0-否 1-是 + */ + forceUpdate?: number; + /** + * 状态 0-禁用 1-启用 + */ + status?: number; + /** + * 热更新 0-否 1-是 + */ + hotUpdate?: number; + /** + * 描述 + */ + description?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface BaseSysDepartmentEntity { + /** + * ID + */ + id?: number; + /** + * 部门名称 + */ + name?: string; + /** + * 上级部门ID + */ + parentId?: number; + /** + * 排序 + */ + orderNum?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface BaseSysLogEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 行为 + */ + action?: string; + /** + * ip + */ + ip?: string; + /** + * ip地址 + */ + ipAddr?: string; + /** + * 参数 + */ + params?: json; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface BaseSysMenuEntity { + /** + * ID + */ + id?: number; + /** + * 父菜单ID + */ + parentId?: number; + /** + * 菜单名称 + */ + name?: string; + /** + * 菜单地址 + */ + router?: string; + /** + * 权限标识 + */ + perms?: string; + /** + * 类型 0-目录 1-菜单 2-按钮 + */ + type?: number; + /** + * 图标 + */ + icon?: string; + /** + * 排序 + */ + orderNum?: number; + /** + * 视图地址 + */ + viewPath?: string; + /** + * 路由缓存 + */ + keepAlive?: boolean; + /** + * 是否显示 + */ + isShow?: boolean; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface BaseSysParamEntity { + /** + * ID + */ + id?: number; + /** + * 键 + */ + keyName?: string; + /** + * 名称 + */ + name?: string; + /** + * 数据 + */ + data?: string; + /** + * 数据类型 0-字符串 1-富文本 2-文件 + */ + dataType?: number; + /** + * 备注 + */ + remark?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface BaseSysRoleEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: string; + /** + * 名称 + */ + name?: string; + /** + * 角色标签 + */ + label?: string; + /** + * 备注 + */ + remark?: string; + /** + * 数据权限是否关联上下级 + */ + relevance?: boolean; + /** + * 菜单权限 + */ + menuIdList?: json; + /** + * 部门权限 + */ + departmentIdList?: json; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface BaseSysUserEntity { + /** + * ID + */ + id?: number; + /** + * 部门ID + */ + departmentId?: number; + /** + * 姓名 + */ + name?: string; + /** + * 用户名 + */ + username?: string; + /** + * 密码 + */ + password?: string; + /** + * 密码版本, 作用是改完密码,让原来的token失效 + */ + passwordV?: number; + /** + * 昵称 + */ + nickName?: string; + /** + * 头像 + */ + headImg?: string; + /** + * 手机 + */ + phone?: string; + /** + * 邮箱 + */ + email?: string; + /** + * 备注 + */ + remark?: string; + /** + * 状态 0-禁用 1-启用 + */ + status?: number; + /** + * socketId + */ + socketId?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface CsMsgEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 会话ID + */ + sessionId?: number; + /** + * 消息内容 + */ + content?: json; + /** + * 类型 0-反馈 1-回复 + */ + type?: number; + /** + * 状态 0-未读 1-已读 + */ + status?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface CsSessionEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 最后一条消息 + */ + lastMsg?: json; + /** + * 客服未读消息数 + */ + adminUnreadCount?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface DictInfoEntity { + /** + * ID + */ + id?: number; + /** + * 类型ID + */ + typeId?: number; + /** + * 名称 + */ + name?: string; + /** + * 值 + */ + value?: string; + /** + * 排序 + */ + orderNum?: number; + /** + * 备注 + */ + remark?: string; + /** + * 父ID + */ + parentId?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface DictTypeEntity { + /** + * ID + */ + id?: number; + /** + * 名称 + */ + name?: string; + /** + * 标识 + */ + key?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface GoodsCommentEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 商品ID + */ + goodsId?: number; + /** + * 订单ID + */ + orderId?: number; + /** + * 内容 + */ + content?: string; + /** + * 星数 + */ + starCount?: number; + /** + * 图片 + */ + pics?: json; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface GoodsInfoEntity { + /** + * ID + */ + id?: number; + /** + * 类型ID + */ + typeId?: number; + /** + * 标题 + */ + title?: string; + /** + * 副标题 + */ + subTitle?: string; + /** + * 主图 + */ + mainPic?: string; + /** + * 图片 + */ + pics?: json; + /** + * 价格 + */ + price?: number; + /** + * 已售 + */ + sold?: number; + /** + * 详情 + */ + content?: string; + /** + * 状态 0-下架 1-上架 + */ + status?: number; + /** + * 排序 + */ + sortNum?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface GoodsSearchKeywordEntity { + /** + * ID + */ + id?: number; + /** + * 名称 + */ + name?: string; + /** + * 排序 + */ + sortNum?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface GoodsSpecEntity { + /** + * ID + */ + id?: number; + /** + * 商品ID + */ + goodsId?: number; + /** + * 名称 + */ + name?: string; + /** + * 价格 + */ + price?: number; + /** + * 库存 + */ + stock?: number; + /** + * 排序 + */ + sortNum?: number; + /** + * 图片 + */ + images?: json; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface GoodsTypeEntity { + /** + * ID + */ + id?: number; + /** + * 名称 + */ + name?: string; + /** + * 父ID + */ + parentId?: number; + /** + * 排序 + */ + sortNum?: number; + /** + * 图片 + */ + pic?: string; + /** + * 状态 0-禁用 1-启用 + */ + status?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface InfoBannerEntity { + /** + * ID + */ + id?: number; + /** + * 描述 + */ + description?: string; + /** + * 跳转路径 + */ + path?: string; + /** + * 图片 + */ + pic?: string; + /** + * 排序 + */ + sortNum?: number; + /** + * 状态 1:启用 2:禁用 + */ + status?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface MarketCouponInfoEntity { + /** + * ID + */ + id?: number; + /** + * 标题 + */ + title?: string; + /** + * 描述 + */ + description?: string; + /** + * 类型 0-满减 + */ + type?: number; + /** + * 金额 + */ + amount?: number; + /** + * 数量 + */ + num?: number; + /** + * 已领取 + */ + receivedNum?: number; + /** + * 开始时间 + */ + startTime?: Date; + /** + * 结束时间 + */ + endTime?: Date; + /** + * 状态 0-禁用 1-启用 + */ + status?: number; + /** + * 条件 + */ + condition?: json; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface MarketCouponUserEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 优惠券ID + */ + couponId?: number; + /** + * 状态 0-未使用 1-已使用 + */ + status?: number; + /** + * 使用时间 + */ + useTime?: Date; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface OrderInfoEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 标题 + */ + title?: string; + /** + * 支付方式 0-待支付 1-微信 2-支付宝 + */ + payType?: number; + /** + * 支付时间 + */ + payTime?: Date; + /** + * 订单号 + */ + orderNum?: string; + /** + * 状态 0-待付款 1-待发货 2-待收货 3-待评价 4-交易完成 5-退款中 6-已退款 7-已关闭 + */ + status?: number; + /** + * 价格 + */ + price?: number; + /** + * 优惠金额 + */ + discountPrice?: number; + /** + * 优惠来源 + */ + discountSource?: json; + /** + * 地址 + */ + address?: json; + /** + * 物流信息 + */ + logistics?: json; + /** + * 退款 + */ + refund?: json; + /** + * 退款状态 + */ + refundStatus?: number; + /** + * 退款申请时间 + */ + refundApplyTime?: Date; + /** + * 备注 + */ + remark?: string; + /** + * 关闭备注 + */ + closeRemark?: string; + /** + * 已开票: 0-未开票 1-已开票 + */ + invoice?: number; + /** + * 微信类型 0-小程序 1-公众号 2-App + */ + wxType?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface PluginInfoEntity { + /** + * ID + */ + id?: number; + /** + * 名称 + */ + name?: string; + /** + * 简介 + */ + description?: string; + /** + * Key名 + */ + keyName?: string; + /** + * Hook + */ + hook?: string; + /** + * 描述 + */ + readme?: string; + /** + * 版本 + */ + version?: string; + /** + * Logo(base64) + */ + logo?: string; + /** + * 作者 + */ + author?: string; + /** + * 状态 0-禁用 1-启用 + */ + status?: number; + /** + * 内容 + */ + content?: json; + /** + * 插件的plugin.json + */ + pluginJson?: json; + /** + * 配置 + */ + config?: json; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface RecycleDataEntity { + /** + * ID + */ + id?: number; + /** + * 表 + */ + entityInfo?: json; + /** + * 操作人 + */ + userId?: number; + /** + * 被删除的数据 + */ + data?: json; + /** + * 请求的接口 + */ + url?: string; + /** + * 请求参数 + */ + params?: json; + /** + * 删除数据条数 + */ + count?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface SpaceInfoEntity { + /** + * ID + */ + id?: number; + /** + * 地址 + */ + url?: string; + /** + * 类型 + */ + type?: string; + /** + * 分类ID + */ + classifyId?: number; + /** + * 文件id + */ + fileId?: string; + /** + * 文件名 + */ + name?: string; + /** + * 文件大小 + */ + size?: number; + /** + * 文档版本 + */ + version?: number; + /** + * 文件位置 + */ + key?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface SpaceTypeEntity { + /** + * ID + */ + id?: number; + /** + * 类别名称 + */ + name?: string; + /** + * 父分类ID + */ + parentId?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface TaskInfoEntity { + /** + * ID + */ + id?: number; + /** + * 任务ID + */ + jobId?: string; + /** + * 任务配置 + */ + repeatConf?: string; + /** + * 名称 + */ + name?: string; + /** + * cron + */ + cron?: string; + /** + * 最大执行次数 不传为无限次 + */ + limit?: number; + /** + * 每间隔多少毫秒执行一次 如果cron设置了 这项设置就无效 + */ + every?: number; + /** + * 备注 + */ + remark?: string; + /** + * 状态 0-停止 1-运行 + */ + status?: number; + /** + * 开始时间 + */ + startDate?: Date; + /** + * 结束时间 + */ + endDate?: Date; + /** + * 数据 + */ + data?: string; + /** + * 执行的service实例ID + */ + service?: string; + /** + * 状态 0-系统 1-用户 + */ + type?: number; + /** + * 下一次执行时间 + */ + nextRunTime?: Date; + /** + * 状态 0-cron 1-时间间隔 + */ + taskType?: number; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface UserAddressEntity { + /** + * ID + */ + id?: number; + /** + * 用户ID + */ + userId?: number; + /** + * 联系人 + */ + contact?: string; + /** + * 手机号 + */ + phone?: string; + /** + * 省 + */ + province?: string; + /** + * 市 + */ + city?: string; + /** + * 区 + */ + district?: string; + /** + * 地址 + */ + address?: string; + /** + * 是否默认 + */ + isDefault?: boolean; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface UserInfoEntity { + /** + * ID + */ + id?: number; + /** + * 登录唯一ID + */ + unionid?: string; + /** + * 头像 + */ + avatarUrl?: string; + /** + * 昵称 + */ + nickName?: string; + /** + * 手机号 + */ + phone?: string; + /** + * 性别 0-未知 1-男 2-女 + */ + gender?: number; + /** + * 状态 0-禁用 1-正常 2-已注销 + */ + status?: number; + /** + * 登录方式 0-小程序 1-公众号 2-H5 + */ + loginType?: number; + /** + * 密码 + */ + password?: string; + /** + * 创建时间 + */ + createTime?: Date; + /** + * 更新时间 + */ + updateTime?: Date; + /** + * 任意键值 + */ + [key: string]: any; + } + + interface TestEntity { + /** + * 任意键值 + */ + [key: string]: any; + } + + interface DemoUserFollowEntity { + /** + * 任意键值 + */ + [key: string]: any; + } + + interface DemoUserInfoEntity { + /** + * 任意键值 + */ + [key: string]: any; + } + interface AppComplain { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: AppComplainEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface AppFeedback { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: AppFeedbackEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface AppGoods { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: AppGoodsEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface AppVersion { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: AppVersionEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseComm { + /** + * 修改个人信息 + */ + personUpdate(data?: any): Promise; + /** + * 文件上传模式 + */ + uploadMode(data?: any): Promise; + /** + * 权限与菜单 + */ + permmenu(data?: any): Promise; + /** + * 个人信息 + */ + person(data?: any): Promise; + /** + * 文件上传 + */ + upload(data?: any): Promise; + /** + * 退出 + */ + logout(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + personUpdate: string; + uploadMode: string; + permmenu: string; + person: string; + upload: string; + logout: string; + }; + /** + * 权限状态 + */ + _permission: { + personUpdate: boolean; + uploadMode: boolean; + permmenu: boolean; + person: boolean; + upload: boolean; + logout: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseOpen { + /** + * 刷新token + */ + refreshToken(data?: any): Promise; + /** + * 验证码 + */ + captcha(data?: any): Promise; + /** + * 登录 + */ + login(data?: any): Promise; + /** + * 获得网页内容的参数值 + */ + html(data?: any): Promise; + /** + * 实体信息与路径 + */ + eps(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + refreshToken: string; + captcha: string; + login: string; + html: string; + eps: string; + }; + /** + * 权限状态 + */ + _permission: { + refreshToken: boolean; + captcha: boolean; + login: boolean; + html: boolean; + eps: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseSysDepartment { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 排序 + */ + order(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { delete: string; update: string; order: string; list: string; add: string }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + order: boolean; + list: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseSysLog { + /** + * 日志保存时间 + */ + setKeep(data?: any): Promise; + /** + * 获得日志保存时间 + */ + getKeep(data?: any): Promise; + /** + * 清理 + */ + clear(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: BaseSysLogEntity[]; + [key: string]: any; + }>; + /** + * 权限标识 + */ + permission: { setKeep: string; getKeep: string; clear: string; page: string }; + /** + * 权限状态 + */ + _permission: { setKeep: boolean; getKeep: boolean; clear: boolean; page: boolean }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseSysMenu { + /** + * 创建代码 + */ + create(data?: any): Promise; + /** + * 导出 + */ + export(data?: any): Promise; + /** + * 导入 + */ + import(data?: any): Promise; + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 解析 + */ + parse(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: BaseSysMenuEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + create: string; + export: string; + import: string; + delete: string; + update: string; + parse: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + create: boolean; + export: boolean; + import: boolean; + delete: boolean; + update: boolean; + parse: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseSysParam { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 获得网页内容的参数值 + */ + html(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: BaseSysParamEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + html: string; + info: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + html: boolean; + info: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseSysRole { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: BaseSysRoleEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface BaseSysUser { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 移动部门 + */ + move(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: BaseSysUserEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + move: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + move: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface CountHome { + /** + * 商品分类 + */ + goodsCategory(data?: any): Promise; + /** + * 订单概况 + */ + orderSummary(data?: any): Promise; + /** + * 用户概况 + */ + userSummary(data?: any): Promise; + /** + * 订单图表 + */ + orderChart(data?: any): Promise; + /** + * 用户图表 + */ + userChart(data?: any): Promise; + /** + * 商品排行 + */ + goodsRank(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + goodsCategory: string; + orderSummary: string; + userSummary: string; + orderChart: string; + userChart: string; + goodsRank: string; + }; + /** + * 权限状态 + */ + _permission: { + goodsCategory: boolean; + orderSummary: boolean; + userSummary: boolean; + orderChart: boolean; + userChart: boolean; + goodsRank: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface CsMsg { + /** + * 未读消息数 + */ + unreadCount(data?: any): Promise; + /** + * 标记已读 + */ + read(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: CsMsgEntity[]; + [key: string]: any; + }>; + /** + * 权限标识 + */ + permission: { unreadCount: string; read: string; page: string }; + /** + * 权限状态 + */ + _permission: { unreadCount: boolean; read: boolean; page: boolean }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface CsSession { + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: CsSessionEntity[]; + [key: string]: any; + }>; + /** + * 权限标识 + */ + permission: { page: string }; + /** + * 权限状态 + */ + _permission: { page: boolean }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface DictInfo { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 获得字典数据 + */ + data(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: DictInfoEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + data: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + data: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface DictType { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: DictTypeEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface GoodsComment { + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: GoodsCommentEntity[]; + [key: string]: any; + }>; + /** + * 权限标识 + */ + permission: { page: string }; + /** + * 权限状态 + */ + _permission: { page: boolean }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface GoodsInfo { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: GoodsInfoEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface GoodsSearchKeyword { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: GoodsSearchKeywordEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface GoodsSpec { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: GoodsSpecEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface GoodsType { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: GoodsTypeEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface InfoBanner { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: InfoBannerEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface MarketCouponInfo { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: MarketCouponInfoEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface MarketCouponUser { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: MarketCouponUserEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface OrderInfo { + /** + * 退款处理 + */ + refundHandle(data?: any): Promise; + /** + * 物流信息 + */ + logistics(data?: any): Promise; + /** + * 发货 + */ + deliver(data?: any): Promise; + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: OrderInfoEntity[]; + [key: string]: any; + }>; + /** + * 权限标识 + */ + permission: { + refundHandle: string; + logistics: string; + deliver: string; + delete: string; + update: string; + info: string; + list: string; + page: string; + }; + /** + * 权限状态 + */ + _permission: { + refundHandle: boolean; + logistics: boolean; + deliver: boolean; + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface PluginInfo { + /** + * 安装插件 + */ + install(data?: any): Promise; + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: PluginInfoEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + install: string; + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + install: boolean; + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface RecycleData { + /** + * 恢复数据 + */ + restore(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: RecycleDataEntity[]; + [key: string]: any; + }>; + /** + * 权限标识 + */ + permission: { restore: string; info: string; page: string }; + /** + * 权限状态 + */ + _permission: { restore: boolean; info: boolean; page: boolean }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface SpaceInfo { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: SpaceInfoEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface SpaceType { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: SpaceTypeEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface TaskInfo { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 开始 + */ + start(data?: any): Promise; + /** + * 执行一次 + */ + once(data?: any): Promise; + /** + * 停止 + */ + stop(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: TaskInfoEntity[]; + [key: string]: any; + }>; + /** + * 日志 + */ + log(data?: any): Promise; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + start: string; + once: string; + stop: string; + info: string; + page: string; + log: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + start: boolean; + once: boolean; + stop: boolean; + info: boolean; + page: boolean; + log: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface UserAddress { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: UserAddressEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface UserInfo { + /** + * 删除 + */ + delete(data?: any): Promise; + /** + * 修改 + */ + update(data?: any): Promise; + /** + * 单个信息 + */ + info(data?: any): Promise; + /** + * 列表查询 + */ + list(data?: any): Promise; + /** + * 分页查询 + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: UserInfoEntity[]; + [key: string]: any; + }>; + /** + * 新增 + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + delete: string; + update: string; + info: string; + list: string; + page: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + delete: boolean; + update: boolean; + info: boolean; + list: boolean; + page: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface Test { + /** + * page + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: TestEntity[]; + [key: string]: any; + }>; + /** + * update + */ + update(data?: any): Promise; + /** + * add + */ + add(data?: any): Promise; + /** + * info + */ + info(data?: any): Promise; + /** + * delete + */ + delete(data?: any): Promise; + /** + * list + */ + list(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + page: string; + update: string; + add: string; + info: string; + delete: string; + list: string; + }; + /** + * 权限状态 + */ + _permission: { + page: boolean; + update: boolean; + add: boolean; + info: boolean; + delete: boolean; + list: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface DemoUserFollow { + /** + * page + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: DemoUserFollowEntity[]; + [key: string]: any; + }>; + /** + * list + */ + list(data?: any): Promise; + /** + * info + */ + info(data?: any): Promise; + /** + * delete + */ + delete(data?: any): Promise; + /** + * update + */ + update(data?: any): Promise; + /** + * add + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + page: string; + list: string; + info: string; + delete: string; + update: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + page: boolean; + list: boolean; + info: boolean; + delete: boolean; + update: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + interface DemoUserInfo { + /** + * t1 + */ + t1(data?: any): Promise; + /** + * t2 + */ + t2(data?: any): Promise; + /** + * t3 + */ + t3(data?: any): Promise; + /** + * page + */ + page(data?: any): Promise<{ + pagination: { size: number; page: number; total: number; [key: string]: any }; + list: DemoUserInfoEntity[]; + [key: string]: any; + }>; + /** + * list + */ + list(data?: any): Promise; + /** + * info + */ + info(data?: any): Promise; + /** + * delete + */ + delete(data?: any): Promise; + /** + * update + */ + update(data?: any): Promise; + /** + * add + */ + add(data?: any): Promise; + /** + * 权限标识 + */ + permission: { + t1: string; + t2: string; + t3: string; + page: string; + list: string; + info: string; + delete: string; + update: string; + add: string; + }; + /** + * 权限状态 + */ + _permission: { + t1: boolean; + t2: boolean; + t3: boolean; + page: boolean; + list: boolean; + info: boolean; + delete: boolean; + update: boolean; + add: boolean; + }; + /** + * 请求 + */ + request: Service["request"]; + } + + type json = any; + + type Service = { + request(options?: { + url: string; + method?: "POST" | "GET" | "PUT" | "DELETE" | "PATCH" | "HEAD" | "OPTIONS"; + data?: any; + params?: any; + headers?: { + [key: string]: any; + }; + timeout?: number; + proxy?: boolean; + [key: string]: any; + }): Promise; + app: { complain: AppComplain; feedback: AppFeedback; goods: AppGoods; version: AppVersion }; + base: { + comm: BaseComm; + open: BaseOpen; + sys: { + department: BaseSysDepartment; + log: BaseSysLog; + menu: BaseSysMenu; + param: BaseSysParam; + role: BaseSysRole; + user: BaseSysUser; + }; + }; + count: { home: CountHome }; + cs: { msg: CsMsg; session: CsSession }; + dict: { info: DictInfo; type: DictType }; + goods: { + comment: GoodsComment; + info: GoodsInfo; + searchKeyword: GoodsSearchKeyword; + spec: GoodsSpec; + type: GoodsType; + }; + info: { banner: InfoBanner }; + market: { coupon: { info: MarketCouponInfo; user: MarketCouponUser } }; + order: { info: OrderInfo }; + plugin: { info: PluginInfo }; + recycle: { data: RecycleData }; + space: { info: SpaceInfo; type: SpaceType }; + task: { info: TaskInfo }; + user: { address: UserAddress; info: UserInfo }; + test: Test; + demo: { user: { follow: DemoUserFollow; info: DemoUserInfo } }; + }; +} diff --git a/build/cool/eps.json b/build/cool/eps.json new file mode 100644 index 0000000..6299858 --- /dev/null +++ b/build/cool/eps.json @@ -0,0 +1 @@ +[{"prefix":"/admin/app/complain","name":"AppComplainEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/app/feedback","name":"AppFeedbackEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/app/goods","name":"AppGoodsEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/app/version","name":"AppVersionEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/base/comm","name":"","api":[{"method":"post","path":"/personUpdate"},{"method":"get","path":"/uploadMode"},{"method":"get","path":"/permmenu"},{"method":"get","path":"/person"},{"method":"post","path":"/upload"},{"method":"post","path":"/logout"}]},{"prefix":"/admin/base/open","name":"","api":[{"method":"get","path":"/refreshToken"},{"method":"get","path":"/captcha"},{"method":"post","path":"/login"},{"method":"get","path":"/html"},{"method":"get","path":"/eps"}]},{"prefix":"/admin/base/sys/department","name":"BaseSysDepartmentEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"post","path":"/order"},{"method":"post","path":"/list"},{"method":"post","path":"/add"}]},{"prefix":"/admin/base/sys/log","name":"BaseSysLogEntity","api":[{"method":"post","path":"/setKeep"},{"method":"get","path":"/getKeep"},{"method":"post","path":"/clear"},{"method":"post","path":"/page"}]},{"prefix":"/admin/base/sys/menu","name":"BaseSysMenuEntity","api":[{"method":"post","path":"/create"},{"method":"post","path":"/export"},{"method":"post","path":"/import"},{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"post","path":"/parse"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/base/sys/param","name":"BaseSysParamEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/html"},{"method":"get","path":"/info"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/base/sys/role","name":"BaseSysRoleEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/base/sys/user","name":"BaseSysUserEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"post","path":"/move"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/count/home","name":"","api":[{"method":"post","path":"/goodsCategory"},{"method":"post","path":"/orderSummary"},{"method":"post","path":"/userSummary"},{"method":"post","path":"/orderChart"},{"method":"post","path":"/userChart"},{"method":"post","path":"/goodsRank"}]},{"prefix":"/admin/cs/msg","name":"CsMsgEntity","api":[{"method":"get","path":"/unreadCount"},{"method":"post","path":"/read"},{"method":"post","path":"/page"}]},{"prefix":"/admin/cs/session","name":"CsSessionEntity","api":[{"method":"post","path":"/page"}]},{"prefix":"/admin/dict/info","name":"DictInfoEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"post","path":"/data"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/dict/type","name":"DictTypeEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/goods/comment","name":"GoodsCommentEntity","api":[{"method":"post","path":"/page"}]},{"prefix":"/admin/goods/info","name":"GoodsInfoEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/goods/searchKeyword","name":"GoodsSearchKeywordEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/goods/spec","name":"GoodsSpecEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/goods/type","name":"GoodsTypeEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/info/banner","name":"InfoBannerEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/market/coupon/info","name":"MarketCouponInfoEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/market/coupon/user","name":"MarketCouponUserEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/order/info","name":"OrderInfoEntity","api":[{"method":"post","path":"/refundHandle"},{"method":"get","path":"/logistics"},{"method":"post","path":"/deliver"},{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"}]},{"prefix":"/admin/plugin/info","name":"PluginInfoEntity","api":[{"method":"post","path":"/install"},{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/recycle/data","name":"RecycleDataEntity","api":[{"method":"post","path":"/restore"},{"method":"get","path":"/info"},{"method":"post","path":"/page"}]},{"prefix":"/admin/space/info","name":"SpaceInfoEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/space/type","name":"SpaceTypeEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/task/info","name":"TaskInfoEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"post","path":"/start"},{"method":"post","path":"/once"},{"method":"post","path":"/stop"},{"method":"get","path":"/info"},{"method":"post","path":"/page"},{"method":"get","path":"/log"},{"method":"post","path":"/add"}]},{"prefix":"/admin/user/address","name":"UserAddressEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/user/info","name":"UserInfoEntity","api":[{"method":"post","path":"/delete"},{"method":"post","path":"/update"},{"method":"get","path":"/info"},{"method":"post","path":"/list"},{"method":"post","path":"/page"},{"method":"post","path":"/add"}]},{"prefix":"/admin/test","name":"TestEntity","api":[{"path":"/page"},{"path":"/update"},{"path":"/add"},{"path":"/info"},{"path":"/delete"},{"path":"/list"}]},{"prefix":"/admin/demo/user/follow","name":"DemoUserFollowEntity","api":[{"path":"/page"},{"path":"/list"},{"path":"/info"},{"path":"/delete"},{"path":"/update"},{"path":"/add"}]},{"prefix":"/admin/demo/user/info","name":"DemoUserInfoEntity","api":[{"path":"/t1"},{"path":"/t2"},{"path":"/t3"},{"path":"/page"},{"path":"/list"},{"path":"/info"},{"path":"/delete"},{"path":"/update"},{"path":"/add"}]}] \ No newline at end of file diff --git a/index.html b/index.html new file mode 100644 index 0000000..392df9c --- /dev/null +++ b/index.html @@ -0,0 +1,172 @@ + + + + + + + + + + + + + + +
+
+

%VITE_NAME%

+
+

正在加载资源...

+

初次加载资源可能需要较多时间 请耐心等待

+
+ + +
+ +
+ + + diff --git a/nginx.conf b/nginx.conf new file mode 100644 index 0000000..09dbdcf --- /dev/null +++ b/nginx.conf @@ -0,0 +1,123 @@ +user nginx; +worker_processes 1; +error_log /var/log/nginx/error.log warn; +pid /var/run/nginx.pid; +events { + worker_connections 1024; +} +http { + include /etc/nginx/mime.types; + default_type application/octet-stream; + log_format main '$remote_addr - $remote_user [$time_local] "$request" ' + '$status $body_bytes_sent "$http_referer" ' + '"$http_user_agent" "$http_x_forwarded_for"'; + access_log /var/log/nginx/access.log main; + sendfile on; + keepalive_timeout 65; + upstream backend { + server midway:8001; + } + + server { + listen 80; + server_name localhost; + location / { + root /app; + index index.html; + try_files $uri $uri/ /index.html; + } + location /api/ + { + proxy_pass http://backend/; + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header REMOTE-HOST $remote_addr; + + #缓存相关配置 + #proxy_cache cache_one; + #proxy_cache_key $host$request_uri$is_args$args; + #proxy_cache_valid 200 304 301 302 1h; + + #持久化连接相关配置 + proxy_connect_timeout 3000s; + proxy_read_timeout 86400s; + proxy_send_timeout 3000s; + #proxy_http_version 1.1; + #proxy_set_header Upgrade $http_upgrade; + #proxy_set_header Connection "upgrade"; + + add_header X-Cache $upstream_cache_status; + + #expires 12h; + } +# location /im { +# proxy_pass http://backend/im; +# proxy_connect_timeout 3600s; #配置点1 +# proxy_read_timeout 3600s; #配置点2,如果没效,可以考虑这个时间配置长一点 +# proxy_send_timeout 3600s; #配置点3 +# proxy_set_header Host $host; +# proxy_set_header X-Real-IP $remote_addr; +# proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; +# proxy_set_header REMOTE-HOST $remote_addr; +# #proxy_bind $remote_addr transparent; +# proxy_http_version 1.1; +# proxy_set_header Upgrade $http_upgrade; +# proxy_set_header Connection "upgrade"; +# # rewrite /socket/(.*) /$1 break; +# proxy_redirect off; + +# } + +# location /socket { +# proxy_pass http://backend/socket; +# proxy_connect_timeout 3600s; #配置点1 +# proxy_read_timeout 3600s; #配置点2,如果没效,可以考虑这个时间配置长一点 +# proxy_send_timeout 3600s; #配置点3 +# proxy_set_header Host $host; +# proxy_set_header X-Real-IP $remote_addr; +# proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; +# proxy_set_header REMOTE-HOST $remote_addr; +# #proxy_bind $remote_addr transparent; +# proxy_http_version 1.1; +# proxy_set_header Upgrade $http_upgrade; +# proxy_set_header Connection "upgrade"; +# rewrite /socket/(.*) /$1 break; +# proxy_redirect off; + +# } + + + location /adminer/ + { + proxy_pass http://adminer:8080/; + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header REMOTE-HOST $remote_addr; + + #缓存相关配置 + #proxy_cache cache_one; + #proxy_cache_key $host$request_uri$is_args$args; + #proxy_cache_valid 200 304 301 302 1h; + + #持久化连接相关配置 + proxy_connect_timeout 3000s; + proxy_read_timeout 86400s; + proxy_send_timeout 3000s; + #proxy_http_version 1.1; + #proxy_set_header Upgrade $http_upgrade; + #proxy_set_header Connection "upgrade"; + + add_header X-Cache $upstream_cache_status; + + #expires 12h; + } + + error_page 500 502 503 504 /50x.html; + location = /50x.html { + root /usr/share/nginx/html; + } + + } +} \ No newline at end of file diff --git a/package.json b/package.json new file mode 100644 index 0000000..7dab5be --- /dev/null +++ b/package.json @@ -0,0 +1,64 @@ +{ + "name": "cool-admin", + "version": "7.2.0", + "scripts": { + "dev": "vite --host", + "build": "vite build", + "preview": "vite preview", + "format": "prettier --write src/", + "lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs,.ts,.tsx,.cts,.mts --fix --ignore-path .eslintignore" + }, + "dependencies": { + "@cool-vue/crud": "^7.2.0", + "@element-plus/icons-vue": "^2.3.1", + "@vueuse/core": "^10.4.0", + "@wangeditor/editor": "^5.1.23", + "@wangeditor/editor-for-vue": "^5.1.12", + "axios": "^1.7.2", + "chardet": "^2.0.0", + "core-js": "^3.32.1", + "dayjs": "^1.11.10", + "echarts": "^5.4.3", + "element-plus": "^2.7.7", + "file-saver": "^2.0.5", + "lodash-es": "^4.17.21", + "marked": "^11.1.1", + "mitt": "^3.0.1", + "mockjs": "^1.1.0", + "monaco-editor": "0.49.0", + "nprogress": "^0.2.0", + "pinia": "^2.1.7", + "socket.io-client": "^4.7.2", + "store": "^2.0.12", + "vue": "^3.4.15", + "vue-echarts": "^6.6.1", + "vue-router": "^4.4.0", + "vuedraggable": "^4.1.0", + "xlsx": "^0.18.5" + }, + "devDependencies": { + "@cool-vue/vite-plugin": "^7.2.1", + "@rushstack/eslint-patch": "^1.8.0", + "@types/file-saver": "^2.0.7", + "@types/lodash-es": "^4.17.8", + "@types/mockjs": "^1.0.7", + "@types/node": "^20.14.5", + "@types/nprogress": "^0.2.0", + "@types/store": "^2.0.2", + "@vitejs/plugin-vue": "^5.0.3", + "@vitejs/plugin-vue-jsx": "^3.1.0", + "@vue/compiler-sfc": "^3.4.15", + "@vue/eslint-config-prettier": "^9.0.0", + "@vue/eslint-config-typescript": "^13.0.0", + "eslint": "^8.57.0", + "eslint-plugin-vue": "^9.23.0", + "prettier": "^3.3.3", + "rollup-plugin-visualizer": "^5.9.2", + "sass": "^1.53.0", + "terser": "^5.27.0", + "typescript": "^5.4.0", + "vite": "^5.3.4", + "vite-plugin-compression": "^0.5.1", + "vite-plugin-vue-devtools": "^7.3.1" + } +} diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..0b823a3934267be14403e19338226b5c33cc5a56 GIT binary patch literal 67646 zcmeI5f2dVe7RT=^DGA945ebtq63?dOCypo8BkO+~Gi1|Y#L}G{t3F$ov2@#P9 zkwGLPBqBomLn1$~7o7yYoWUhgXqR`MtLzCdmLfTaU=G?B=wH z{yTQgA@UxgT^|w>3pe*AGzF~&$6DkTQY6<|V`NU9KsE;rsshg7{|54q<~afT z4*f}^8lOoAr}#ie2!zY5$glIpUTO-Tx;3pV;*4V z9wh19GfwV&j;-zb0Qd5K>o=qbUzZrS3n~1Wim@LS^j!q6k1@4cAK?Cw2mWtj<1fgH zJm&=ZpY-6{M^1~{sPy^(_p~tM`*p0>jV7nRmy?z zaTeKdpez+m|x97ve*9SP?9Dw)3X}Ry^e>FIYe)hCEt!jB9 z>jRu?f996`II=Y`^N)U~k*em;7w%x|16)6Xh(BN0huJxunfP7@aBNh@G7Mjt>r~eV zxIP+yZ`l$&@*jBQJLfXzWK!6?t`Bg$0+DMpn(T^m~+?91n>cacBkInL-W-?QCBGM2b0|wTyk%RCbl}-z9zicJSLUt zC*RX)bMR7<|EzkjEkk6li8jW%**}!Ut+d1IC1q|UcZI(P?qPgg8DrgYFY%1>UE8(* zyuLolJWT5Y#)pt!0${DnJoqAw&$taf&na^%IrFkV5IZN7vDPK`5=Z9RPoGBv@OdJh zYvaCxbsgf~=jl6VH}UaZ0A76Kw(ZI1_|)%LBp3A2-eV)VS~qUq?eWB2-@+RWTUy_~2Vjg|Ig8Lg~?D8CmCHJ;aw+4~>9m>2uB<`z;b0D!H5@S-{eAIsv z`}6$@?&p=UOZ?c4oObN*SB^o>0Z-?_NdHN%51(h@m6!tXwm%EUxb4Is_XPf^9Lq@8 zP=>_Pm(aN`L zmssR{oA&G{AMay7aQzn5dw8`zKm0Wd7yds$d~lLvLvpbu3y05k{GCybN%mJ{zNg9? z6E=OX52>HEB?d%dNy?p%`ZIZX7`H$1RQj9-ejn?@cUe5*C&WEnu|G@0pxH)jPa#RU^HqO9GX~c-@r=4Y$Y=i4 zHJA1S0r+{!jUl8?e6rukSFVTI`8)PiuW1b93!ho*L+WuSm=VD~EGA!dBV8*S7N4&= z;(J*+M&arOUwDhy!T+%JVRxD>!5n~>m(wtWXd^~BPPF$3s(q)Cmm^}8^+EQ-Ecu?w zwne-4cV)tPDh3;mW*Hm@5nt!DBs-65#^BR7Ts$6tkAprjk7J8iWsjito`z4x!5DHZ zTzsOOf8pB8-acQvfJdG~mvrpwMwWvLhG~!RjOCBpdH(;V){psd2k}!9!z@ z17eqF5GKu8AN8d>@7eh$_7Lmfr?$^Bm5IMm?|WSCiNEphgW;etpL^{56L+IN2ivjR zWq4@Bbq_mcDHC_?uY-O#D807gL#a_>%8ND*N06ms?`d{*FtyD#hMDH};}B@q~Ep$=0Iy zMjb~SL7#&&Dz38kgSSdP_sHdyID$L}%k^6a9qvvdzCITZv!l945bPoL_MNOlu7||{ zf9sX=FLT1%a}V7Z#C8Du>s`Nf(&6u%GKarl-}~NZv5AQ7VdouRb?54o>tV6L-!^3q zB^P5q_t?cQHUr?!e14#dtBN{2mj1ctHm~3f@f`e1w$D13;d57jV`+~s4#B!70N!1` z@Rr>UT*~#+o64MC^~D1?uPS2|E`s0V?u(xyc8IsMzk6iI1I``FxtI4Vui8GFP$uRT zh&($@;@6VgzkB9#54H=6_#|%OV1?_q2|95fRpwUs&Goy7Ht#Yv0KTKP&nA`OHpCvm z%xCf^aSp!I%K5MC-o7Nx^47y|X@2+E#tyF4i1yyT4{e`KEW>dre)rtv9$beJt@kCd z-OaAwChG8fQJHJ`ea_6^<%zkMzKD#8^Q7^5G4|K=S9pF_nQO`4zEOFWz5(z&J1U+g z^n>r~%ADW8e&#h6>4QjLjblExdA$ia%@ZSh?^WhpxPLASlk^FIV{aCYrnSSl>=FDy znftG@c*Xx$$`~ZZTJ7!2;&jw@csZoZy_^FQ_XtW`0RJQXHuR{R)u|ucuR?xP&cR1X zoZG7jTuU7yZSi|Fjn`AI-&P-F0RA^9=Rop$e9)_z zcgF+x-QfCd^`#7uhx5w0kbQg)#-62&Xz%SiANy-x(-|Na+YycPuWhrxM-X2j_V#U0 zkGgHO86YQ@mFK`eurK@iM5Jub4d3O7GsHdh*Wr4zN-3(y>{zL5DPntjddkBU=YpVhkDAyE82R1NRUG2(@*QzyNa~#Jzj& z99VJ>VFJ9iTA2azAL8C!JO4xPAyy8vHJ`!&b3psveLMe&&k#s~*fuFLK>oKPc8)a5 ziO&#G46!wx!2tRHI=~$GIs-=AJRt_i{muaS-#MZ1TDKVtkbCWC_l@)n!3>yfvmyh` zfmamI0k0@RZ%un(fZWR-!5fO$uey1}%P;t96IN;~b;DCkB;(!@O&UY3EtVExKo}R(G_fkNO2i)z7Pe*ztX(bt%&p1uH7qf01&M2|?C7J9luXcY8}l#FhW8EVnZCZypHh*hd@ zp<~$SGrasBIWh!@E{Vv$|A?g@+iwTP9i3y6`#stN%VuE7WhJa}kh6yVsB&KJn9<{`@ z#Y8UZ5kcr!mA}Z-d|!`{pwF1*9zk<`J&FWAxJRA?SggA(uE-O*|=R| z<*%;`A^gS7hLBW!PP&_eD;neK^U~g@#}G82FGzcz4nxp@4mlPbhM)l*0xUYrj7ilY z#G=C-QUkgnWRVVYNLlnmge=mtRG%B5!^{|qo~8Qy03C)Pi!MWmMbA>*o;`xg)gfrw I#+$7FA7}!`OaK4? literal 0 HcmV?d00001 diff --git a/public/logo.png b/public/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..a75c6187a4c8ebd40815ca264db6518574605f21 GIT binary patch literal 1403 zcmV->1%&#EP)bj)RKO&~I*-$z3=m5uWmUjk#0#4K0Jg3WuML(1*!q{U zD_{p-+^FL;~80cYPKP5N~vF zp9hTkyPe-KNGj5{%Eq#!{wf>&?z)AoMdD5fQ33mm@*fg!aME0H*UR*qO@T~~7)MQ# zhB9mo<*+r9H}-_06<}SSeonlxsmD3@eVP92lQWACN?I|F*ybiSHyiD(xH$fk3fPP7 zyzdNfqKV82Y2=@?k8W@f;=EYxL18`xF~*#T1CJFF@W0t{Zi(J3R~g z$omk+khv|!QMoN?opJP&w$ABW*iv={SWjd=11=bK5-vD>3;9lrJhL`z91Tg@m}BJA zZDS6f!#=hXEEQl~aC`}zGU`-w%Gt4CUl@60Y^|h4{=e#cz^Ukg> zeZ+A$rC9>nfISAs0Pg`ejXKrz0*|Ktzv=8+*n{sBZ}kl*qqR#Km2^nbUy@E$eIN8j z?3;i^qrHkIb%GFS$=&IP}(9y@n&b2;!$qIbd_GP#h6b57L1OL zXV}RqaUU)_y|kU<#<&*h#nUVSAvYq8O5BHMtLz2YkTpxdexsAKlA@HpZ;AVGq$;+^ z`LnQJh6Ln2jL1{sJ}gV>tBP^lXrty(CZXQ%;DxGd z9|M;SHo9r_^g5N9F6E!AWpY}z)AS9ajU~nXYghr>fB{2hTg3)y-#oLfI0uOPw~Fcn zgnQr^jwRr2YWD(p_8j&5=gPwRSgK6^l=P9}nA()Rw#fZ>u3iF;1q!|hCndcksYm`x z=}(M4jwzLZZ<|&;2+yfWUvps@lbemK225vAbvjILwvkkWTEoE<{yyJ+^F}Tg2Dg*002ov JPDHLkV1f{#oVEY} literal 0 HcmV?d00001 diff --git a/public/用户导入模版.xlsx b/public/用户导入模版.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..e73dd9e43adadbb2564cd568f0a31bd3530f967b GIT binary patch literal 9198 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 + + + + + + diff --git a/src/config/dev.ts b/src/config/dev.ts new file mode 100644 index 0000000..d49ea77 --- /dev/null +++ b/src/config/dev.ts @@ -0,0 +1,20 @@ +import { getUrlParam, storage } from "/@/cool/utils"; +import { proxy } from "./proxy"; + +export default { + // 根地址 + host: proxy["/dev/"].target, + + // 请求地址 + get baseUrl() { + let proxy = getUrlParam("proxy"); + + if (proxy) { + storage.set("proxy", proxy); + } else { + proxy = storage.get("proxy") || "dev"; + } + + return `/${proxy}`; + } +}; diff --git a/src/config/index.ts b/src/config/index.ts new file mode 100644 index 0000000..27c3263 --- /dev/null +++ b/src/config/index.ts @@ -0,0 +1,60 @@ +import dev from "./dev"; +import prod from "./prod"; + +// 是否开发模式 +export const isDev = import.meta.env.DEV; + +// 配置 +export const config = { + // 项目信息 + app: { + name: import.meta.env.VITE_NAME, + + // 菜单 + menu: { + // 是否分组显示 + isGroup: false, + // 自定义菜单列表 + list: [] + }, + + // 路由 + router: { + // 模式 + mode: "history", + // 转场动画 + transition: "slide" + }, + + // 字体图标库 + iconfont: [] + }, + + // 忽略规则 + ignore: { + // 不显示请求进度条 + NProgress: [ + "/__cool_eps", + "/base/open/eps", + "/base/comm/person", + "/base/comm/permmenu", + "/base/comm/upload", + "/base/comm/uploadMode", + "/dict/info/data", + "/space/info/add" + ], + // 页面不需要登录验证 + token: ["/login", "/401", "/403", "/404", "/500", "/502"] + }, + + // 调试 + test: { + token: "", + eps: true + }, + + // 当前环境 + ...(isDev ? dev : prod) +}; + +export * from "./proxy"; diff --git a/src/config/prod.ts b/src/config/prod.ts new file mode 100644 index 0000000..838fd82 --- /dev/null +++ b/src/config/prod.ts @@ -0,0 +1,9 @@ +import { proxy } from "./proxy"; + +export default { + // 根地址 + host: proxy["/prod/"].target, + + // 请求地址 + baseUrl: "/api" +}; diff --git a/src/config/proxy.ts b/src/config/proxy.ts new file mode 100644 index 0000000..9d6c263 --- /dev/null +++ b/src/config/proxy.ts @@ -0,0 +1,13 @@ +export const proxy = { + "/dev/": { + target: "http://127.0.0.1:8001", + changeOrigin: true, + rewrite: (path: string) => path.replace(/^\/dev/, "") + }, + + "/prod/": { + target: "https://show.cool-admin.com", + changeOrigin: true, + rewrite: (path: string) => path.replace(/^\/prod/, "/api") + } +}; diff --git a/src/cool/bootstrap/eps.ts b/src/cool/bootstrap/eps.ts new file mode 100644 index 0000000..d891a56 --- /dev/null +++ b/src/cool/bootstrap/eps.ts @@ -0,0 +1,129 @@ +import { cloneDeep, merge } from "lodash-es"; +import { BaseService, service } from "../service"; +import { Module } from "../types"; +import { path2Obj } from "../utils"; +import { config, isDev } from "/@/config"; +import { eps } from "virtual:eps"; +import { hmr } from "../hooks"; +import { module } from "../module"; + +// 更新事件 +function onUpdate() { + // 设置 request 方法 + function set(d: any) { + if (d.namespace) { + const a = new BaseService(d.namespace); + + for (const i in d) { + const { path, method = "get" } = d[i]; + + if (path) { + a.request = a.request; + + a[i] = function (data?: any) { + return this.request({ + url: path, + method, + [method.toLocaleLowerCase() == "post" ? "data" : "params"]: data + }); + }; + } + } + + for (const i in a) { + d[i] = a[i]; + } + } else { + for (const i in d) { + set(d[i]); + } + } + } + + // 遍历每一个方法 + set(eps.service); + + // 合并 eps + merge(service, eps.service); + + // 合并[local] + merge( + service, + cloneDeep( + path2Obj( + module.list.reduce((a, b) => { + return a.concat(...((b.services as any[]) || [])); + }, []) + ) + ) + ); + + // 热更新处理 + hmr.setData("service", service); + + // 提示 + if (isDev) { + console.log("[cool-eps] updated"); + } +} + +export function createEps(modules: Module[]) { + // 更新 eps + onUpdate(); + + // 开发环境下,生成本地 service 的类型描述文件 + if (isDev && config.test.eps) { + const list: any[] = []; + + // 模拟 eps 数据 + modules.forEach((m) => { + m.services?.forEach((s) => { + const api = Array.from( + new Set([ + ...Object.getOwnPropertyNames(s.value.constructor.prototype), + "page", + "list", + "info", + "delete", + "update", + "add" + ]) + ) + .filter((e) => !["constructor", "namespace"].includes(e)) + .map((e) => { + return { + path: `/${e}` + }; + }); + + list.push({ + api, + module: m.name, + name: s.value.constructor.name + "Entity", + prefix: `/admin/${s.path}`, + isLocal: true + }); + }); + }); + + // 生成文件 + service.request({ + url: "/__cool_eps", + method: "POST", + proxy: false, + data: { + list + } + }); + } +} + +// 监听 vite 触发事件 +if (import.meta.hot) { + import.meta.hot.on("eps-update", ({ service }) => { + if (service) { + eps.service = service; + } + onUpdate(); + }); +} diff --git a/src/cool/bootstrap/index.ts b/src/cool/bootstrap/index.ts new file mode 100644 index 0000000..b2ed2d2 --- /dev/null +++ b/src/cool/bootstrap/index.ts @@ -0,0 +1,24 @@ +import { createPinia } from "pinia"; +import { App } from "vue"; +import { createModule } from "./module"; +import { router } from "../router"; +import { Loading } from "../utils"; +import { createEps } from "./eps"; +import "virtual:svg-register"; + +export async function bootstrap(app: App) { + // pinia + app.use(createPinia()); + + // 路由 + app.use(router); + + // 模块 + const { eventLoop, list } = createModule(app); + + // eps + createEps(list); + + // 加载 + Loading.set([eventLoop()]); +} diff --git a/src/cool/bootstrap/module.ts b/src/cool/bootstrap/module.ts new file mode 100644 index 0000000..e5421cd --- /dev/null +++ b/src/cool/bootstrap/module.ts @@ -0,0 +1,116 @@ +import { type App, type Directive } from "vue"; +import { assign, isFunction, orderBy } from "lodash-es"; +import { filename } from "../utils"; +import { module } from "../module"; +import { hmr } from "../hooks"; + +// 扫描文件 +const files = import.meta.glob("/src/{modules,plugins}/*/{config.ts,service/**,directives/**}", { + eager: true, + import: "default" +}); + +// 模块列表 +module.list = hmr.getData("modules", []); + +// 解析 +for (const i in files) { + // 分割 + const [, , type, name, action] = i.split("/"); + + // 文件名 + const n = filename(i); + + // 文件内容 + const v = files[i]; + + // 模块是否存在 + const m = module.get(name); + + // 数据 + const d = m || { + name, + type, + value: null, + services: [], + directives: [] + }; + + // 配置 + if (action == "config.ts") { + d.value = v; + } + // 服务 + else if (action == "service") { + const s = new (v as any)(); + + if (s) { + d.services?.push({ + path: s.namespace, + value: s + }); + } + } + // 指令 + else if (action == "directives") { + d.directives?.push({ name: n, value: v as Directive }); + } + + if (!m) { + module.add(d); + } +} + +// 创建 +export function createModule(app: App) { + // 排序 + module.list.forEach((e) => { + const d = isFunction(e.value) ? e.value(app) : e.value; + + if (d) { + assign(e, d); + } + + if (!d.order) { + e.order = 0; + } + }); + + const list = orderBy(module.list, "order", "desc").map((e) => { + // 初始化 + e.install?.(app, e.options); + + // 注册组件 + e.components?.forEach(async (c) => { + // @ts-ignore + const v = await (isFunction(c) ? c() : c); + const n = v.default || v; + + if (n.name) { + app.component(n.name, n); + } + }); + + // 注册指令 + e.directives?.forEach((v) => { + app.directive(v.name, v.value); + }); + + return e; + }); + + return { + // 模块列表 + list, + // 事件加载 + async eventLoop() { + const events: any = {}; + + for (let i = 0; i < list.length; i++) { + if (list[i].onLoad) { + assign(events, await list[i]?.onLoad?.(events)); + } + } + } + }; +} diff --git a/src/cool/hooks/browser.ts b/src/cool/hooks/browser.ts new file mode 100644 index 0000000..618d256 --- /dev/null +++ b/src/cool/hooks/browser.ts @@ -0,0 +1,30 @@ +import { useEventListener } from "@vueuse/core"; +import { reactive, watch } from "vue"; +import { getBrowser } from "../utils"; + +const browser = reactive(getBrowser()); +const events: (() => void)[] = []; + +watch( + () => browser.screen, + () => { + events.forEach((ev) => ev()); + } +); + +useEventListener(window, "resize", () => { + Object.assign(browser, getBrowser()); +}); + +export function useBrowser() { + return { + browser, + onScreenChange(ev: () => void, immediate = true) { + events.push(ev); + + if (immediate) { + ev(); + } + } + }; +} diff --git a/src/cool/hooks/hmr.ts b/src/cool/hooks/hmr.ts new file mode 100644 index 0000000..e7526ae --- /dev/null +++ b/src/cool/hooks/hmr.ts @@ -0,0 +1,23 @@ +// 解决热更新后失效问题; +const data = import.meta.hot?.data.getData?.() || {}; + +if (import.meta.hot) { + import.meta.hot.data.getData = () => { + return data; + }; +} + +export const hmr = { + data, + + setData(key: string, value: any) { + data[key] = value; + }, + + getData(key: string, defaultValue?: any) { + if (defaultValue !== undefined && !data[key]) { + this.setData(key, defaultValue); + } + return data[key]; + } +}; diff --git a/src/cool/hooks/index.ts b/src/cool/hooks/index.ts new file mode 100644 index 0000000..31a5a00 --- /dev/null +++ b/src/cool/hooks/index.ts @@ -0,0 +1,54 @@ +import { getCurrentInstance, Ref, reactive } from "vue"; +import { useRoute, useRouter } from "vue-router"; +import { service } from "../service"; +import { useBrowser } from "./browser"; +import { useMitt } from "./mitt"; + +export function useRefs() { + const refs = reactive<{ [key: string]: any }>({}); + + function setRefs(name: string) { + return (el: any) => { + refs[name] = el; + return () => refs[name]; + }; + } + + return { refs, setRefs }; +} + +export function useParent(name: string, r: Ref) { + const d = getCurrentInstance(); + + if (d) { + let parent = d.proxy?.$.parent; + + if (parent) { + while (parent && parent.type?.name != name) { + parent = parent?.parent; + } + + if (parent) { + if (parent.type.name == name) { + r.value = parent.exposed; + } + } + } + } + + return r; +} + +export function useCool() { + return { + service, + route: useRoute(), + router: useRouter(), + mitt: useMitt(), + ...useBrowser(), + ...useRefs() + }; +} + +export * from "./browser"; +export * from "./hmr"; diff --git a/src/cool/hooks/mitt.ts b/src/cool/hooks/mitt.ts new file mode 100644 index 0000000..c74482d --- /dev/null +++ b/src/cool/hooks/mitt.ts @@ -0,0 +1,8 @@ +import Mitt, { Emitter } from "mitt"; +import { hmr } from "./hmr"; + +const mitt: Emitter = hmr.getData("mitt", Mitt()); + +export function useMitt() { + return mitt; +} diff --git a/src/cool/index.ts b/src/cool/index.ts new file mode 100644 index 0000000..0942ffc --- /dev/null +++ b/src/cool/index.ts @@ -0,0 +1,7 @@ +export * from "./service"; +export * from "./bootstrap"; +export * from "./hooks"; +export * from "./module"; +export * from "./router"; +export * from "./types"; +export { storage } from "./utils"; diff --git a/src/cool/module/index.ts b/src/cool/module/index.ts new file mode 100644 index 0000000..8012811 --- /dev/null +++ b/src/cool/module/index.ts @@ -0,0 +1,27 @@ +import { Module } from "../types"; +import { hmr } from "../hooks"; +import { ctx } from "virtual:ctx"; + +// 模块列表 +const list: Module[] = hmr.getData("modules", []); + +// 模块对象 +const module = { + list, + dirs: ctx.modules, + req: Promise.resolve(), + get(name: string): Module { + return this.list.find((e) => e.name == name)!; + }, + config(name: string) { + return this.get(name).options || {}; + }, + add(data: Module) { + this.list.push(data); + }, + wait() { + return this.req; + } +}; + +export { module }; diff --git a/src/cool/router/index.ts b/src/cool/router/index.ts new file mode 100644 index 0000000..705cf04 --- /dev/null +++ b/src/cool/router/index.ts @@ -0,0 +1,217 @@ +import { ElMessage } from "element-plus"; +import { createRouter, createWebHashHistory, createWebHistory, RouteRecordRaw } from "vue-router"; +import { Router, storage, module } from "/@/cool"; +import { isArray } from "lodash-es"; +import { useBase } from "/$/base"; +import { Loading } from "../utils"; +import { config } from "/@/config"; + +// 基本路径 +const baseUrl = import.meta.env.BASE_URL; + +// 扫描文件 +const files = import.meta.glob(["/src/modules/*/{views,pages}/**/*", "!**/components"]); + +// 默认路由 +const routes: RouteRecordRaw[] = [ + { + path: "/", + name: "index", + component: () => import("/$/base/pages/main/index.vue"), + children: [] + }, + { + path: "/:catchAll(.*)", + name: "404", + component: () => import("/$/base/pages/error/404.vue") + } +]; + +// 创建路由器 +const router = createRouter({ + history: + config.app.router.mode == "history" + ? createWebHistory(baseUrl) + : createWebHashHistory(baseUrl), + routes +}) as Router; + +// 组件加载后 +router.beforeResolve(() => { + Loading.close(); +}); + +let lock = false; + +// 错误监听 +router.onError((err: Error) => { + if (!lock) { + lock = true; + + ElMessage.error(`页面存在错误:${err.message}`); + console.error(err); + + // 动态加载组件错误,刷新页面 + if (err.message?.includes("Failed to fetch dynamically imported module")) { + window.location.reload(); + } + + setTimeout(() => { + lock = false; + }, 0); + } +}); + +// 添加试图,页面路由 +router.append = function (data) { + const list = isArray(data) ? data : [data]; + + list.forEach((d) => { + if (!d.meta) { + d.meta = {}; + } + + // 组件路径 + if (!d.component) { + const url = d.viewPath; + + if (url) { + if (url.indexOf("http") == 0) { + if (d.meta) { + d.meta.iframeUrl = url; + } + + d.component = () => import("/$/base/views/frame.vue"); + } else { + d.component = files["/src/" + url.replace("cool/", "")]; + } + } else { + d.redirect = "/404"; + } + } + + // 是否动态添加 + d.meta.dynamic = true; + + if (d.isPage) { + router.addRoute(d as any); + } else { + router.addRoute("index", d as any); + } + }); +}; + +// 清空路由 +router.clear = function () { + const rs = router.getRoutes(); + + rs.forEach((e) => { + if (e.name && e.meta?.dynamic) { + router.removeRoute(e.name); + } + }); +}; + +// 找路由 +router.find = function (path: string) { + return router.getRoutes().find((e) => { + if (path == "/") { + return e.path == path && e.name != "index"; + } else { + return e.path == path; + } + }); +}; + +// 注册 +router.register = async function (path: string) { + // 当前路由是否注册 + const isReg = Boolean(router.find(path)); + + if (!isReg) { + const { menu } = useBase(); + + // 等待应用配置加载完 + await Loading.wait(); + + // 待注册列表 + const list: any[] = []; + + // 动态菜单数据 + menu.routes.find((e) => { + list.push({ + ...e, + isPage: e.viewPath?.includes("/pages") + }); + }); + + // 本地模块数据 + module.list.forEach((e) => { + if (e.views) { + list.push(...e.views); + } + + if (e.pages) { + list.push( + ...e.pages.map((d) => { + return { + ...d, + isPage: true + }; + }) + ); + } + }); + + // 需要注册的路由 + const r = list.find((e) => e.path == path); + + if (r) { + router.append(r); + } + } + + return { route: router.find(path), isReg }; +}; + +// 路由守卫 +router.beforeEach(async (to, from, next) => { + // 数据缓存 + const { user, process } = useBase(); + + // 预先注册路由 + const { isReg, route } = await router.register(to.path); + + // 组件不存在、路由不存在 + if (!route?.components) { + next(user.token ? "/404" : "/login"); + } else { + if (!isReg) { + next(to.fullPath); + } else { + // 登录成功 + if (user.token) { + // 在登录页 + if (to.path.includes("/login")) { + // Token 未过期 + if (!storage.isExpired("token")) { + // 回到首页 + return next("/"); + } + } else { + // 添加路由进程 + process.add(to); + } + } else { + // 忽略部分 Token 验证 + if (!config.ignore.token.find((e) => to.path == e)) { + return next("/login"); + } + } + + next(); + } + } +}); + +export { router }; diff --git a/src/cool/service/base.ts b/src/cool/service/base.ts new file mode 100644 index 0000000..337d473 --- /dev/null +++ b/src/cool/service/base.ts @@ -0,0 +1,117 @@ +// @ts-nocheck +import { isDev, config, proxy } from "../../config"; +import { isObject } from "lodash-es"; +import { request } from "./request"; +import { AxiosRequestConfig } from "axios"; + +export function Service( + value: + | string + | { + proxy?: string; + namespace?: string; + url?: string; + } +) { + return function (target: any) { + // 命名 + if (typeof value == "string") { + target.prototype.namespace = value; + } + + // 复杂项 + if (isObject(value)) { + const { namespace, proxy: proxyName, url } = value; + + target.prototype.namespace = namespace; + + if (proxyName) { + target.prototype.url = proxy[proxyName]?.target || url; + } else { + target.prototype.url = url; + } + } + }; +} + +export class BaseService { + constructor(namespace?: string) { + if (namespace) { + this.namespace = namespace; + } + } + + async request(options: AxiosRequestConfig = {}) { + if (options.url) { + // 过滤 http 开头的地址 + if (options.url.indexOf("http") < 0) { + let ns = ""; + + if (isDev) { + ns = this.proxy || config.baseUrl; + } else { + ns = this.proxy ? this.url : config.baseUrl; + } + + // 拼接前缀 + if (this.namespace) { + ns += "/" + this.namespace; + } + + // 处理地址 + if (options.proxy === undefined || options.proxy) { + options.url = ns + options.url; + } + } + } + + return request(options); + } + + async list(data: any) { + return this.request({ + url: "/list", + method: "POST", + data + }); + } + + async page(data: any) { + return this.request({ + url: "/page", + method: "POST", + data + }); + } + + async info(params: any) { + return this.request({ + url: "/info", + params + }); + } + + async update(data: any) { + return this.request({ + url: "/update", + method: "POST", + data + }); + } + + async delete(data: any) { + return this.request({ + url: "/delete", + method: "POST", + data + }); + } + + async add(data: any) { + return this.request({ + url: "/add", + method: "POST", + data + }); + } +} diff --git a/src/cool/service/index.ts b/src/cool/service/index.ts new file mode 100644 index 0000000..ee01e75 --- /dev/null +++ b/src/cool/service/index.ts @@ -0,0 +1,9 @@ +import { hmr } from "../hooks"; +import { BaseService } from "./base"; + +// service 数据集合 +export const service: Eps.Service = hmr.getData("service", { + request: new BaseService().request +}); + +export * from "./base"; diff --git a/src/cool/service/request.ts b/src/cool/service/request.ts new file mode 100644 index 0000000..88b3e44 --- /dev/null +++ b/src/cool/service/request.ts @@ -0,0 +1,158 @@ +import axios from "axios"; +import NProgress from "nprogress"; +import "nprogress/nprogress.css"; +import { ElMessage } from "element-plus"; +import { endsWith } from "lodash-es"; +import { storage } from "/@/cool/utils"; +import { useBase } from "/$/base"; +import { router } from "../router"; +import { config, isDev } from "/@/config"; + +const request = axios.create({ + timeout: import.meta.env.VITE_TIMEOUT, + withCredentials: false +}); + +NProgress.configure({ + showSpinner: true +}); + +// 请求队列 +let queue: Array<(token: string) => void> = []; + +// 是否刷新中 +let isRefreshing = false; + +// 请求 +request.interceptors.request.use( + (req: any) => { + const { user } = useBase(); + + if (req.url) { + // 请求进度条 + if ( + !config.ignore.NProgress.some((e) => req.url.match(new RegExp(`${e}.*`))) && + (req.NProgress ?? true) + ) { + NProgress.start(); + } + } + + // 请求信息 + if (isDev) { + console.group(req.url); + console.log("method:", req.method); + console.table("data:", req.method == "get" ? req.params : req.data); + console.groupEnd(); + } + + // 验证 token + if (user.token) { + // 请求标识 + if (req.headers && req.headers["Authorization"] !== null) { + req.headers["Authorization"] = user.token; + } + + // 忽略 + if (["eps", "refreshToken"].some((e) => endsWith(req.url, e))) { + return req; + } + + // 判断 token 是否过期 + if (storage.isExpired("token")) { + // 判断 refreshToken 是否过期 + if (storage.isExpired("refreshToken")) { + ElMessage.error("登录状态已失效,请重新登录"); + user.logout(); + } else { + // 是否在刷新中 + if (!isRefreshing) { + isRefreshing = true; + + user.refreshToken() + .then((token) => { + queue.forEach((cb) => cb(token)); + queue = []; + isRefreshing = false; + }) + .catch(() => { + user.logout(); + }); + } + + return new Promise((resolve) => { + // 继续请求 + queue.push((token) => { + // 重新设置 token + if (req.headers) { + req.headers["Authorization"] = token; + } + resolve(req); + }); + }); + } + } + } + + return req; + }, + (error) => { + return Promise.reject(error); + } +); + +// 响应 +request.interceptors.response.use( + (res) => { + NProgress.done(); + + if (!res?.data) { + return res; + } + + const { code, data, message } = res.data; + + if (!code) { + return res.data; + } + + switch (code) { + case 1000: + return data; + default: + return Promise.reject({ code, message }); + } + }, + async (error) => { + NProgress.done(); + + if (error.response) { + const { status } = error.response; + const { user } = useBase(); + + if (status == 401) { + user.logout(); + } else { + if (!isDev) { + switch (status) { + case 403: + router.push("/403"); + break; + + case 500: + router.push("/500"); + break; + + case 502: + router.push("/502"); + break; + } + } + } + } + + return Promise.reject({ message: error.message }); + } +); + +export { request }; diff --git a/src/cool/types/index.ts b/src/cool/types/index.ts new file mode 100644 index 0000000..58eec90 --- /dev/null +++ b/src/cool/types/index.ts @@ -0,0 +1,60 @@ +import { Component, Directive, App } from "vue"; +import { Router as VueRouter, RouteRecordRaw } from "vue-router"; + +export declare type Merge = Omit & B; + +export declare interface ModuleConfig { + name?: string; + label?: string; + description?: string; + order?: number; + version?: string; + logo?: string; + author?: string; + updateTime?: string; + demo?: { name: string; component: Component }[] | string; + options?: { + [key: string]: any; + }; + toolbar?: { + order?: number; + pc?: boolean; + h5?: boolean; + component: Promise; + }; + components?: Component[]; + views?: RouteRecordRaw[]; + pages?: RouteRecordRaw[]; + install?(app: App, options?: any): any; + onLoad?(events: { + hasToken: (cb: () => Promise | void) => Promise | void; + [key: string]: any; + }): Promise<{ [key: string]: any }> | Promise | void; +} + +export declare interface Module extends ModuleConfig { + name: string; + options: { + [key: string]: any; + }; + value?: any; + services?: { path: string; value: any }[]; + directives?: { name: string; value: Directive }[]; + [key: string]: any; +} + +export declare interface Router extends VueRouter { + find(path: string): RouteRecordRaw | undefined; + append( + data: { + name?: string; + path: string; + component?: any; + viewPath?: string; + isPage?: boolean; + [key: string]: any; + }[] + ): void; + register(path: string): Promise<{ route: RouteRecordRaw | undefined; isReg: boolean }>; + [key: string]: any; +} diff --git a/src/cool/utils/index.ts b/src/cool/utils/index.ts new file mode 100644 index 0000000..e7547ab --- /dev/null +++ b/src/cool/utils/index.ts @@ -0,0 +1,301 @@ +import { isArray, isNumber, isString, orderBy } from "lodash-es"; +import { resolveComponent } from "vue"; +import storage from "./storage"; + +// 首字母大写 +export function firstUpperCase(value: string): string { + return value.replace(/\b(\w)(\w*)/g, function ($0, $1, $2) { + return $1.toUpperCase() + $2; + }); +} + +// 获取方法名 +export function getNames(value: any) { + return Object.getOwnPropertyNames(value.constructor.prototype); +} + +// 获取地址栏参数 +export function getUrlParam(name: string): string | null { + const reg = new RegExp("(^|&)" + name + "=([^&]*)(&|$)"); + const r = window.location.search.substr(1).match(reg); + if (r != null) return decodeURIComponent(r[2]); + return null; +} + +// 文件名 +export function filename(path: string): string { + return basename(path.substring(0, path.lastIndexOf("."))); +} + +// 路径名称 +export function basename(path: string): string { + let index = path.lastIndexOf("/"); + index = index > -1 ? index : path.lastIndexOf("\\"); + if (index < 0) { + return path; + } + return path.substring(index + 1); +} + +// 文件扩展名 +export function extname(path: string): string { + return path.substring(path.lastIndexOf(".") + 1).split(/(\?|&)/)[0]; +} + +// 横杠转驼峰 +export function toCamel(str: string): string { + return str.replace(/([^-])(?:-+([^-]))/g, function ($0, $1, $2) { + return $1 + $2.toUpperCase(); + }); +} + +// uuid +export function uuid(separator = "-"): string { + const s: any[] = []; + const hexDigits = "0123456789abcdef"; + for (let i = 0; i < 36; i++) { + s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1); + } + s[14] = "4"; + s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1); + s[8] = s[13] = s[18] = s[23] = separator; + + return s.join(""); +} + +// 浏览器信息 +export function getBrowser() { + const { clientHeight, clientWidth } = document.documentElement; + + // 浏览器信息 + const ua = navigator.userAgent.toLowerCase(); + + // 浏览器类型 + let type = (ua.match(/firefox|chrome|safari|opera/g) || "other")[0]; + + if ((ua.match(/msie|trident/g) || [])[0]) { + type = "msie"; + } + + // 平台标签 + let tag = ""; + + const isTocuh = + "ontouchstart" in window || ua.indexOf("touch") !== -1 || ua.indexOf("mobile") !== -1; + if (isTocuh) { + if (ua.indexOf("ipad") !== -1) { + tag = "pad"; + } else if (ua.indexOf("mobile") !== -1) { + tag = "mobile"; + } else if (ua.indexOf("android") !== -1) { + tag = "androidPad"; + } else { + tag = "pc"; + } + } else { + tag = "pc"; + } + + // 浏览器内核 + let prefix = ""; + + switch (type) { + case "chrome": + case "safari": + case "mobile": + prefix = "webkit"; + break; + case "msie": + prefix = "ms"; + break; + case "firefox": + prefix = "Moz"; + break; + case "opera": + prefix = "O"; + break; + default: + prefix = "webkit"; + break; + } + + // 操作平台 + const plat = ua.indexOf("android") > 0 ? "android" : navigator.platform.toLowerCase(); + + // 屏幕信息 + let screen = "full"; + + if (clientWidth < 768) { + screen = "xs"; + } else if (clientWidth < 992) { + screen = "sm"; + } else if (clientWidth < 1200) { + screen = "md"; + } else if (clientWidth < 1920) { + screen = "xl"; + } else { + screen = "full"; + } + + // 是否 ios + const isIOS = !!navigator.userAgent.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/); + + // 是否 PC 端 + const isPC = tag === "pc"; + + // 是否移动端 + const isMobile = isPC ? false : true; + + // 是否移动端 + 屏幕宽过小 + const isMini = screen === "xs" || isMobile; + + return { + height: clientHeight, + width: clientWidth, + type, + plat, + tag, + prefix, + isMobile, + isIOS, + isPC, + isMini, + screen + }; +} + +// 路径转数组 +export function deepPaths(paths: string[], splitor?: string) { + const list: any[] = []; + + paths.forEach((e) => { + const arr: string[] = e.split(splitor || "/").filter(Boolean); + + let c = list; + + arr.forEach((a, i) => { + let d = c.find((e) => e.label == a); + + if (!d) { + d = { + label: a, + value: a, + children: arr[i + 1] ? [] : null + }; + + c.push(d); + } + + if (d.children) { + c = d.children; + } + }); + }); + + return list; +} + +// 列表转树形 +export function deepTree(list: any[], sort?: "desc" | "asc"): any[] { + const newList: any[] = []; + const map: any = {}; + + orderBy(list, "orderNum", sort) + .map((e) => { + map[e.id] = e; + return e; + }) + .forEach((e) => { + const parent = map[e.parentId]; + + if (parent) { + (parent.children || (parent.children = [])).push(e); + } else { + newList.push(e); + } + }); + + return newList; +} + +// 树形转列表 +export function revDeepTree(list: any[]) { + const arr: any[] = []; + let id = 0; + + function deep(list: any[], parentId: number) { + list.forEach((e) => { + if (!e.id) { + e.id = ++id; + } + + if (!e.parentId) { + e.parentId = parentId; + } + + arr.push(e); + + if (e.children && isArray(e.children) && e.id) { + deep(e.children, e.id); + } + }); + } + + deep(list || [], 0); + + return arr; +} + +// 路径转对象 +export function path2Obj(list: any[]) { + const data: any = {}; + + list.forEach(({ path, value }) => { + if (path) { + const arr: string[] = path.split("/"); + const parents = arr.slice(0, arr.length - 1); + const name = basename(path).replace(".ts", ""); + + let curr = data; + + parents.forEach((k) => { + if (!curr[k]) { + curr[k] = {}; + } + + curr = curr[k]; + }); + + curr[name] = value; + } + }); + + return data; +} + +// 是否是组件 +export function isComponent(name: string) { + return !isString(resolveComponent(name)); +} + +// 是否Promise +export function isPromise(val: any) { + return val && Object.prototype.toString.call(val) === "[object Promise]"; +} + +// 单位转换 +export function parsePx(val: string | number) { + return isNumber(val) ? `${val}px` : val; +} + +// 延迟 +export function sleep(duration: number) { + return new Promise((resolve) => { + setTimeout(() => { + resolve(true); + }, duration); + }); +} + +export { storage }; +export * from "./loading"; diff --git a/src/cool/utils/loading.ts b/src/cool/utils/loading.ts new file mode 100644 index 0000000..2120339 --- /dev/null +++ b/src/cool/utils/loading.ts @@ -0,0 +1,37 @@ +export const Loading = { + resolve: null as (() => void) | null, + next: null as Promise | null, + + async set(list: Promise[]) { + try { + await Promise.all(list); + } catch (e) { + console.error("[Loading] Error: ", e); + } + + if (this.resolve) { + this.resolve(); + } + }, + + async wait() { + if (this.next) { + return this.next; + } + return Promise.resolve(); + }, + + close() { + const el = document.getElementById("Loading"); + + if (el) { + setTimeout(() => { + el.className += " is-hide"; + }, 0); + } + } +}; + +Loading.next = new Promise((resolve) => { + Loading.resolve = resolve; +}); diff --git a/src/cool/utils/storage.ts b/src/cool/utils/storage.ts new file mode 100644 index 0000000..d200aca --- /dev/null +++ b/src/cool/utils/storage.ts @@ -0,0 +1,81 @@ +import store from "store"; + +export default { + // 后缀标识 + suffix: "_deadtime", + + /** + * 获取 + * @param {string} key 关键字 + */ + get(key: string) { + return store.get(key); + }, + + /** + * 获取全部 + */ + info() { + const d: any = {}; + + store.each(function (value: any, key: any) { + d[key] = value; + }); + + return d; + }, + + /** + * 设置 + * @param {string} key 关键字 + * @param {*} value 值 + * @param {number} expires 过期时间 + */ + set(key: string, value: any, expires?: any) { + store.set(key, value); + + if (expires) { + store.set(`${key}${this.suffix}`, Date.parse(String(new Date())) + expires * 1000); + } + }, + + /** + * 是否过期 + * @param {string} key 关键字 + */ + isExpired(key: string) { + return (this.getExpiration(key) || 0) - Date.parse(String(new Date())) <= 2000; + }, + + /** + * 获取到期时间 + * @param {string} key 关键字 + */ + getExpiration(key: string) { + return this.get(key + this.suffix); + }, + + /** + * 移除 + * @param {string} key 关键字 + */ + remove(key: string) { + store.remove(key); + this.removeExpiration(key); + }, + + /** + * 移除到期时间 + * @param {string} key 关键字 + */ + removeExpiration(key: string) { + store.remove(key + this.suffix); + }, + + /** + * 清理 + */ + clearAll() { + store.clearAll(); + } +}; diff --git a/src/main.ts b/src/main.ts new file mode 100644 index 0000000..77fefe8 --- /dev/null +++ b/src/main.ts @@ -0,0 +1,17 @@ +import { createApp } from "vue"; +import App from "./App.vue"; +import { bootstrap } from "./cool"; + +const app = createApp(App); + +// 启动 +bootstrap(app) + .then(() => { + app.mount("#app"); + + // 加载图表 + import("echarts"); + }) + .catch((err) => { + console.error("COOL-ADMIN 启动失败", err); + }); diff --git a/src/modules/app/views/complain.vue b/src/modules/app/views/complain.vue new file mode 100644 index 0000000..7883b4d --- /dev/null +++ b/src/modules/app/views/complain.vue @@ -0,0 +1,148 @@ + + + diff --git a/src/modules/app/views/feedback.vue b/src/modules/app/views/feedback.vue new file mode 100644 index 0000000..22fe925 --- /dev/null +++ b/src/modules/app/views/feedback.vue @@ -0,0 +1,148 @@ + + + diff --git a/src/modules/app/views/goods.vue b/src/modules/app/views/goods.vue new file mode 100644 index 0000000..af440fe --- /dev/null +++ b/src/modules/app/views/goods.vue @@ -0,0 +1,188 @@ + + + diff --git a/src/modules/app/views/version.vue b/src/modules/app/views/version.vue new file mode 100644 index 0000000..7b83641 --- /dev/null +++ b/src/modules/app/views/version.vue @@ -0,0 +1,177 @@ + + + diff --git a/src/modules/base/common/index.ts b/src/modules/base/common/index.ts new file mode 100644 index 0000000..1729f7c --- /dev/null +++ b/src/modules/base/common/index.ts @@ -0,0 +1,2 @@ +export * from "./theme"; +export * from "./permission"; diff --git a/src/modules/base/common/permission.ts b/src/modules/base/common/permission.ts new file mode 100644 index 0000000..066f2c4 --- /dev/null +++ b/src/modules/base/common/permission.ts @@ -0,0 +1,30 @@ +import { useStore } from "../store"; +import { isObject } from "lodash-es"; + +function parse(value: any) { + const { menu } = useStore(); + + if (typeof value == "string") { + return value ? menu.perms.some((e: any) => e.includes(value.replace(/\s/g, ""))) : false; + } else { + return Boolean(value); + } +} + +export function checkPerm(value: string | { or?: string[]; and?: string[] }) { + if (!value) { + return false; + } + + if (isObject(value)) { + if (value.or) { + return value.or.some(parse); + } + + if (value.and) { + return value.and.some((e: any) => !parse(e)) ? false : true; + } + } + + return parse(value); +} diff --git a/src/modules/base/common/theme.ts b/src/modules/base/common/theme.ts new file mode 100644 index 0000000..25b34b0 --- /dev/null +++ b/src/modules/base/common/theme.ts @@ -0,0 +1,12 @@ +import { config } from "/@/config"; +import { createLink } from "../utils"; + +// 字体图标库加载 +if (config.app.iconfont) { + config.app.iconfont.forEach((e: string) => { + createLink(e); + }); +} + +// 默认 +createLink("//at.alicdn.com/t/c/font_3254019_h02ghb7ckt5.css"); diff --git a/src/modules/base/components/avatar/index.tsx b/src/modules/base/components/avatar/index.tsx new file mode 100644 index 0000000..7547e12 --- /dev/null +++ b/src/modules/base/components/avatar/index.tsx @@ -0,0 +1,53 @@ +import { defineComponent, type PropType } from "vue"; +import { UserFilled } from "@element-plus/icons-vue"; + +export default defineComponent({ + name: "cl-avatar", + + props: { + modelValue: String, + src: String, + icon: { + type: null, + default: UserFilled + }, + size: { + type: [String, Number] as PropType<"large" | "default" | "small" | number>, + default: 40 + }, + shape: { + type: String as PropType<"circle" | "square">, + default: "square" + }, + fit: { + type: String as PropType<"fill" | "contain" | "cover" | "none" | "scale-down">, + default: "cover" + } + }, + + setup(props) { + return () => { + const height = props.size + "px"; + + return ( +
+ +
+ ); + }; + } +}); diff --git a/src/modules/base/components/code/json.vue b/src/modules/base/components/code/json.vue new file mode 100644 index 0000000..5711221 --- /dev/null +++ b/src/modules/base/components/code/json.vue @@ -0,0 +1,134 @@ + + + + + diff --git a/src/modules/base/components/dept/check.vue b/src/modules/base/components/dept/check.vue new file mode 100644 index 0000000..a8cfad1 --- /dev/null +++ b/src/modules/base/components/dept/check.vue @@ -0,0 +1,104 @@ + + + + + diff --git a/src/modules/base/components/dept/select.vue b/src/modules/base/components/dept/select.vue new file mode 100644 index 0000000..ee3e284 --- /dev/null +++ b/src/modules/base/components/dept/select.vue @@ -0,0 +1,83 @@ + + + + + diff --git a/src/modules/base/components/editor/index.tsx b/src/modules/base/components/editor/index.tsx new file mode 100644 index 0000000..e1f1409 --- /dev/null +++ b/src/modules/base/components/editor/index.tsx @@ -0,0 +1,42 @@ +import { defineComponent, h, resolveComponent, ref, reactive, watch } from "vue"; +import { isComponent } from "/@/cool/utils"; + +export default defineComponent({ + name: "cl-editor", + + props: { + name: { + type: String, + required: true + } + }, + + setup(props, { slots, expose }) { + const Editor = ref(); + const ex = reactive({}); + + watch(Editor, (v) => { + if (v) { + Object.assign(ex, v); + } + }); + + expose(ex); + + return () => { + return isComponent(props.name) ? ( + h( + // @ts-ignore + resolveComponent(props.name), + { + ...props, + ref: Editor + }, + slots + ) + ) : ( + + ); + }; + } +}); diff --git a/src/modules/base/components/icon/svg.vue b/src/modules/base/components/icon/svg.vue new file mode 100644 index 0000000..c4903dd --- /dev/null +++ b/src/modules/base/components/icon/svg.vue @@ -0,0 +1,48 @@ + + + + + diff --git a/src/modules/base/components/image/index.vue b/src/modules/base/components/image/index.vue new file mode 100644 index 0000000..be7f51a --- /dev/null +++ b/src/modules/base/components/image/index.vue @@ -0,0 +1,98 @@ + + + + + diff --git a/src/modules/base/components/link/index.vue b/src/modules/base/components/link/index.vue new file mode 100644 index 0000000..ef18156 --- /dev/null +++ b/src/modules/base/components/link/index.vue @@ -0,0 +1,86 @@ + + + + + diff --git a/src/modules/base/components/menu/check.vue b/src/modules/base/components/menu/check.vue new file mode 100644 index 0000000..f2d712a --- /dev/null +++ b/src/modules/base/components/menu/check.vue @@ -0,0 +1,92 @@ + + + + + diff --git a/src/modules/base/components/menu/file.vue b/src/modules/base/components/menu/file.vue new file mode 100644 index 0000000..7ffd56c --- /dev/null +++ b/src/modules/base/components/menu/file.vue @@ -0,0 +1,128 @@ + + + + + diff --git a/src/modules/base/components/menu/icon.vue b/src/modules/base/components/menu/icon.vue new file mode 100644 index 0000000..6c8608e --- /dev/null +++ b/src/modules/base/components/menu/icon.vue @@ -0,0 +1,51 @@ + + + + + diff --git a/src/modules/base/components/menu/perms.vue b/src/modules/base/components/menu/perms.vue new file mode 100644 index 0000000..90107c9 --- /dev/null +++ b/src/modules/base/components/menu/perms.vue @@ -0,0 +1,91 @@ + + + + + diff --git a/src/modules/base/components/menu/select.vue b/src/modules/base/components/menu/select.vue new file mode 100644 index 0000000..1460cd9 --- /dev/null +++ b/src/modules/base/components/menu/select.vue @@ -0,0 +1,83 @@ + + + + + diff --git a/src/modules/base/config.ts b/src/modules/base/config.ts new file mode 100644 index 0000000..de9f3a5 --- /dev/null +++ b/src/modules/base/config.ts @@ -0,0 +1,90 @@ +import type { ModuleConfig } from "/@/cool"; +import { useStore } from "./store"; +import { config } from "/@/config"; +import "./static/css/index.scss"; + +export default (): ModuleConfig => { + return { + order: 99, + components: Object.values(import.meta.glob("./components/**/*.{vue,tsx}")), + views: [ + { + path: "/my/info", + meta: { + label: "个人中心" + }, + component: () => import("./views/info.vue") + } + ], + pages: [ + { + path: "/login", + component: () => import("./pages/login/index.vue") + }, + { + path: "/401", + meta: { + process: false + }, + component: () => import("./pages/error/401.vue") + }, + { + path: "/403", + meta: { + process: false + }, + component: () => import("./pages/error/403.vue") + }, + { + path: "/404", + meta: { + process: false + }, + component: () => import("./pages/error/404.vue") + }, + { + path: "/500", + meta: { + process: false + }, + component: () => import("./pages/error/500.vue") + }, + { + path: "/502", + meta: { + process: false + }, + component: () => import("./pages/error/502.vue") + } + ], + install() { + // 设置标题 + document.title = config.app.name; + }, + async onLoad() { + const { user, menu, app } = useStore(); + + // token 事件 + async function hasToken(cb: () => Promise | void) { + if (cb) { + app.addEvent("hasToken", cb); + + if (user.token) { + await cb(); + } + } + } + + await hasToken(async () => { + // 获取用户信息 + user.get(); + // 获取菜单权限 + await menu.get(); + }); + + return { + hasToken + }; + } + }; +}; diff --git a/src/modules/base/directives/permission.ts b/src/modules/base/directives/permission.ts new file mode 100644 index 0000000..902ed5e --- /dev/null +++ b/src/modules/base/directives/permission.ts @@ -0,0 +1,13 @@ +import { checkPerm } from "../common/permission"; + +function change(el: any, binding: any) { + el.style.display = checkPerm(binding.value) ? el.getAttribute("_display") : "none"; +} + +export default { + created(el: any, binding: any) { + el.setAttribute("_display", el.style.display || ""); + change(el, binding); + }, + updated: change +}; diff --git a/src/modules/base/hooks/dept.tsx b/src/modules/base/hooks/dept.tsx new file mode 100644 index 0000000..17df731 --- /dev/null +++ b/src/modules/base/hooks/dept.tsx @@ -0,0 +1,43 @@ +import { TreeData } from "element-plus/es/components/tree/src/tree.type"; +import { service } from "/@/cool"; +import Node from "element-plus/es/components/tree/src/model/node"; +import ClAvatar from "../components/avatar/index"; +import { type ClViewGroup, useViewGroup } from "/@/plugins/view"; + +export function useDeptViewGroup(options: DeepPartial) { + const { ViewGroup } = useViewGroup({ + label: "员工列表", + service: service.base.sys.department, + enableAdd: false, + enableRefresh: false, + enableContextMenu: false, + tree: { + lazy: true, + onLoad(node: Node, resolve: (data: TreeData) => void) { + if (node.data.id) { + service.base.sys.user.list({ departmentId: node.data.id }).then((res) => { + res.forEach((e) => { + e.isLeaf = true; + e.icon = ( + + ); + }); + + res.unshift(...(node.data.children || [])); + + resolve(res); + }); + } + } + }, + ...options + }); + + return { + ViewGroup + }; +} diff --git a/src/modules/base/hooks/index.ts b/src/modules/base/hooks/index.ts new file mode 100644 index 0000000..f9d9b65 --- /dev/null +++ b/src/modules/base/hooks/index.ts @@ -0,0 +1 @@ +export * from "./dept"; diff --git a/src/modules/base/index.ts b/src/modules/base/index.ts new file mode 100644 index 0000000..94e86cc --- /dev/null +++ b/src/modules/base/index.ts @@ -0,0 +1,11 @@ +import { useStore } from "./store"; + +export function useBase() { + return { + ...useStore() + }; +} + +export * from "./common"; +export * from "./hooks"; +export * from "./types/index.d"; diff --git a/src/modules/base/pages/error/401.vue b/src/modules/base/pages/error/401.vue new file mode 100644 index 0000000..080bd0a --- /dev/null +++ b/src/modules/base/pages/error/401.vue @@ -0,0 +1,7 @@ + + + diff --git a/src/modules/base/pages/error/403.vue b/src/modules/base/pages/error/403.vue new file mode 100644 index 0000000..f82678d --- /dev/null +++ b/src/modules/base/pages/error/403.vue @@ -0,0 +1,7 @@ + + + diff --git a/src/modules/base/pages/error/404.vue b/src/modules/base/pages/error/404.vue new file mode 100644 index 0000000..ef39e7b --- /dev/null +++ b/src/modules/base/pages/error/404.vue @@ -0,0 +1,7 @@ + + + diff --git a/src/modules/base/pages/error/500.vue b/src/modules/base/pages/error/500.vue new file mode 100644 index 0000000..d9415b4 --- /dev/null +++ b/src/modules/base/pages/error/500.vue @@ -0,0 +1,7 @@ + + + diff --git a/src/modules/base/pages/error/502.vue b/src/modules/base/pages/error/502.vue new file mode 100644 index 0000000..956b2df --- /dev/null +++ b/src/modules/base/pages/error/502.vue @@ -0,0 +1,7 @@ + + + diff --git a/src/modules/base/pages/error/components/error-page.vue b/src/modules/base/pages/error/components/error-page.vue new file mode 100644 index 0000000..2938c3f --- /dev/null +++ b/src/modules/base/pages/error/components/error-page.vue @@ -0,0 +1,136 @@ + + + + + diff --git a/src/modules/base/pages/login/components/pic-captcha.vue b/src/modules/base/pages/login/components/pic-captcha.vue new file mode 100644 index 0000000..6fa8b29 --- /dev/null +++ b/src/modules/base/pages/login/components/pic-captcha.vue @@ -0,0 +1,110 @@ + + + + + diff --git a/src/modules/base/pages/login/index.vue b/src/modules/base/pages/login/index.vue new file mode 100644 index 0000000..db46d6b --- /dev/null +++ b/src/modules/base/pages/login/index.vue @@ -0,0 +1,297 @@ + + + + + diff --git a/src/modules/base/pages/login/static/bg.svg b/src/modules/base/pages/login/static/bg.svg new file mode 100644 index 0000000..a96530d --- /dev/null +++ b/src/modules/base/pages/login/static/bg.svg @@ -0,0 +1,39 @@ + + + + + + + + + + + + + + + diff --git a/src/modules/base/pages/main/components/amenu.vue b/src/modules/base/pages/main/components/amenu.vue new file mode 100644 index 0000000..5e620d6 --- /dev/null +++ b/src/modules/base/pages/main/components/amenu.vue @@ -0,0 +1,154 @@ + + + + + diff --git a/src/modules/base/pages/main/components/bmenu.tsx b/src/modules/base/pages/main/components/bmenu.tsx new file mode 100644 index 0000000..22428e5 --- /dev/null +++ b/src/modules/base/pages/main/components/bmenu.tsx @@ -0,0 +1,97 @@ +import { defineComponent, h } from "vue"; +import { useBase, Menu } from "/$/base"; +import { useCool } from "/@/cool"; + +export default defineComponent({ + name: "b-menu", + + setup() { + const { router, route, browser } = useCool(); + const { menu, app } = useBase(); + + // 页面跳转 + function toView(url: string) { + if (url != route.path) { + router.push(url); + } + + // 小屏下点击收起左侧菜单 + if (browser.isMini) { + app.fold(true); + } + } + + // 渲染子菜单 + function renderMenu() { + function deep(list: Menu.Item[], index: number) { + return list + .filter((e) => e.isShow) + .map((e) => { + const item = (e: Menu.Item) => { + return [ + + + , + {e.meta?.label} + ]; + }; + + if (e.type == 0) { + return h( + , + { + index: String(e.id), + key: e.id, + popperClass: "app-slider__menu" + }, + { + title() { + return item(e); + }, + default() { + return deep(e.children || [], index + 1); + } + } + ); + } else { + return h( + , + { + index: + route.path == "/" + ? e.meta?.isHome + ? "/" + : e.path + : e.path, + key: e.id + }, + { + default() { + return item(e); + } + } + ); + } + }); + } + + return deep(menu.list, 1); + } + + return () => { + return ( +
+ + {renderMenu()} + +
+ ); + }; + } +}); diff --git a/src/modules/base/pages/main/components/process.vue b/src/modules/base/pages/main/components/process.vue new file mode 100644 index 0000000..f505d01 --- /dev/null +++ b/src/modules/base/pages/main/components/process.vue @@ -0,0 +1,278 @@ + + + + + diff --git a/src/modules/base/pages/main/components/route-nav.vue b/src/modules/base/pages/main/components/route-nav.vue new file mode 100644 index 0000000..3ac4205 --- /dev/null +++ b/src/modules/base/pages/main/components/route-nav.vue @@ -0,0 +1,72 @@ + + + + + diff --git a/src/modules/base/pages/main/components/slider.vue b/src/modules/base/pages/main/components/slider.vue new file mode 100644 index 0000000..2afbc04 --- /dev/null +++ b/src/modules/base/pages/main/components/slider.vue @@ -0,0 +1,95 @@ + + + + + diff --git a/src/modules/base/pages/main/components/topbar.vue b/src/modules/base/pages/main/components/topbar.vue new file mode 100644 index 0000000..7c11719 --- /dev/null +++ b/src/modules/base/pages/main/components/topbar.vue @@ -0,0 +1,186 @@ + + + + + diff --git a/src/modules/base/pages/main/components/views.vue b/src/modules/base/pages/main/components/views.vue new file mode 100644 index 0000000..81a6860 --- /dev/null +++ b/src/modules/base/pages/main/components/views.vue @@ -0,0 +1,95 @@ + + + + + diff --git a/src/modules/base/pages/main/index.vue b/src/modules/base/pages/main/index.vue new file mode 100644 index 0000000..9352bda --- /dev/null +++ b/src/modules/base/pages/main/index.vue @@ -0,0 +1,105 @@ + + + + + diff --git a/src/modules/base/static/css/animation.scss b/src/modules/base/static/css/animation.scss new file mode 100644 index 0000000..65b8453 --- /dev/null +++ b/src/modules/base/static/css/animation.scss @@ -0,0 +1,29 @@ +@keyframes dou { + 0% { + transform: rotate(0); + } + 11% { + transform: rotate(7.61deg); + } + 23% { + transform: rotate(-5.8deg); + } + 36% { + transform: rotate(3.35deg); + } + 49% { + transform: rotate(-1.9deg); + } + 62% { + transform: rotate(1.12deg); + } + 75% { + transform: rotate(-0.64deg); + } + 88% { + transform: rotate(0.37deg); + } + 100% { + transform: rotate(-0.28deg); + } +} diff --git a/src/modules/base/static/css/index.scss b/src/modules/base/static/css/index.scss new file mode 100644 index 0000000..d5831e5 --- /dev/null +++ b/src/modules/base/static/css/index.scss @@ -0,0 +1,26 @@ +#app { + height: 100vh; + width: 100vw; + overflow: hidden; +} + +:root { + --view-bg-color: #f7f7f7; +} + +a { + text-decoration: none; +} + +input, +button { + outline: none; +} + +input { + &:-webkit-autofill { + box-shadow: 0 0 0px 1000px white inset; + } +} + +@import "./animation.scss"; diff --git a/src/modules/base/static/svg/icon-activity.svg b/src/modules/base/static/svg/icon-activity.svg new file mode 100644 index 0000000..be00ee9 --- /dev/null +++ b/src/modules/base/static/svg/icon-activity.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-amount.svg b/src/modules/base/static/svg/icon-amount.svg new file mode 100644 index 0000000..13c68d9 --- /dev/null +++ b/src/modules/base/static/svg/icon-amount.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-app.svg b/src/modules/base/static/svg/icon-app.svg new file mode 100644 index 0000000..1d82a1c --- /dev/null +++ b/src/modules/base/static/svg/icon-app.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-approve.svg b/src/modules/base/static/svg/icon-approve.svg new file mode 100644 index 0000000..e77593c --- /dev/null +++ b/src/modules/base/static/svg/icon-approve.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-auth.svg b/src/modules/base/static/svg/icon-auth.svg new file mode 100644 index 0000000..b299f59 --- /dev/null +++ b/src/modules/base/static/svg/icon-auth.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-ban.svg b/src/modules/base/static/svg/icon-ban.svg new file mode 100644 index 0000000..0376a77 --- /dev/null +++ b/src/modules/base/static/svg/icon-ban.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-camera.svg b/src/modules/base/static/svg/icon-camera.svg new file mode 100644 index 0000000..c277464 --- /dev/null +++ b/src/modules/base/static/svg/icon-camera.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-card.svg b/src/modules/base/static/svg/icon-card.svg new file mode 100644 index 0000000..f6ae0e8 --- /dev/null +++ b/src/modules/base/static/svg/icon-card.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-cart.svg b/src/modules/base/static/svg/icon-cart.svg new file mode 100644 index 0000000..f6833eb --- /dev/null +++ b/src/modules/base/static/svg/icon-cart.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-command.svg b/src/modules/base/static/svg/icon-command.svg new file mode 100644 index 0000000..172c809 --- /dev/null +++ b/src/modules/base/static/svg/icon-command.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-common.svg b/src/modules/base/static/svg/icon-common.svg new file mode 100644 index 0000000..7fe6060 --- /dev/null +++ b/src/modules/base/static/svg/icon-common.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-component.svg b/src/modules/base/static/svg/icon-component.svg new file mode 100644 index 0000000..a44dae7 --- /dev/null +++ b/src/modules/base/static/svg/icon-component.svg @@ -0,0 +1,17 @@ + + + + diff --git a/src/modules/base/static/svg/icon-count.svg b/src/modules/base/static/svg/icon-count.svg new file mode 100644 index 0000000..daf26ba --- /dev/null +++ b/src/modules/base/static/svg/icon-count.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-crown.svg b/src/modules/base/static/svg/icon-crown.svg new file mode 100644 index 0000000..71429de --- /dev/null +++ b/src/modules/base/static/svg/icon-crown.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-data.svg b/src/modules/base/static/svg/icon-data.svg new file mode 100644 index 0000000..1dbfe70 --- /dev/null +++ b/src/modules/base/static/svg/icon-data.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-db.svg b/src/modules/base/static/svg/icon-db.svg new file mode 100644 index 0000000..4a5d345 --- /dev/null +++ b/src/modules/base/static/svg/icon-db.svg @@ -0,0 +1,15 @@ + + + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-delete.svg b/src/modules/base/static/svg/icon-delete.svg new file mode 100644 index 0000000..792bce7 --- /dev/null +++ b/src/modules/base/static/svg/icon-delete.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-dept.svg b/src/modules/base/static/svg/icon-dept.svg new file mode 100644 index 0000000..f0af65d --- /dev/null +++ b/src/modules/base/static/svg/icon-dept.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-design.svg b/src/modules/base/static/svg/icon-design.svg new file mode 100644 index 0000000..3aed42f --- /dev/null +++ b/src/modules/base/static/svg/icon-design.svg @@ -0,0 +1,12 @@ + + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-device.svg b/src/modules/base/static/svg/icon-device.svg new file mode 100644 index 0000000..f82b640 --- /dev/null +++ b/src/modules/base/static/svg/icon-device.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-dict.svg b/src/modules/base/static/svg/icon-dict.svg new file mode 100644 index 0000000..219f205 --- /dev/null +++ b/src/modules/base/static/svg/icon-dict.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-discover.svg b/src/modules/base/static/svg/icon-discover.svg new file mode 100644 index 0000000..3747d7e --- /dev/null +++ b/src/modules/base/static/svg/icon-discover.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-emoji.svg b/src/modules/base/static/svg/icon-emoji.svg new file mode 100644 index 0000000..2809858 --- /dev/null +++ b/src/modules/base/static/svg/icon-emoji.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-favor.svg b/src/modules/base/static/svg/icon-favor.svg new file mode 100644 index 0000000..98ad58e --- /dev/null +++ b/src/modules/base/static/svg/icon-favor.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-fx.svg b/src/modules/base/static/svg/icon-fx.svg new file mode 100644 index 0000000..02e7688 --- /dev/null +++ b/src/modules/base/static/svg/icon-fx.svg @@ -0,0 +1,29 @@ + + + + + + + diff --git a/src/modules/base/static/svg/icon-goods.svg b/src/modules/base/static/svg/icon-goods.svg new file mode 100644 index 0000000..391d0ea --- /dev/null +++ b/src/modules/base/static/svg/icon-goods.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-home.svg b/src/modules/base/static/svg/icon-home.svg new file mode 100644 index 0000000..94cf445 --- /dev/null +++ b/src/modules/base/static/svg/icon-home.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-hot.svg b/src/modules/base/static/svg/icon-hot.svg new file mode 100644 index 0000000..b907eeb --- /dev/null +++ b/src/modules/base/static/svg/icon-hot.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-info.svg b/src/modules/base/static/svg/icon-info.svg new file mode 100644 index 0000000..797ed3a --- /dev/null +++ b/src/modules/base/static/svg/icon-info.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-iot.svg b/src/modules/base/static/svg/icon-iot.svg new file mode 100644 index 0000000..f7ccae0 --- /dev/null +++ b/src/modules/base/static/svg/icon-iot.svg @@ -0,0 +1,43 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-like.svg b/src/modules/base/static/svg/icon-like.svg new file mode 100644 index 0000000..c49af81 --- /dev/null +++ b/src/modules/base/static/svg/icon-like.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-living.svg b/src/modules/base/static/svg/icon-living.svg new file mode 100644 index 0000000..47fd717 --- /dev/null +++ b/src/modules/base/static/svg/icon-living.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-log.svg b/src/modules/base/static/svg/icon-log.svg new file mode 100644 index 0000000..190b185 --- /dev/null +++ b/src/modules/base/static/svg/icon-log.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-map.svg b/src/modules/base/static/svg/icon-map.svg new file mode 100644 index 0000000..714981b --- /dev/null +++ b/src/modules/base/static/svg/icon-map.svg @@ -0,0 +1,21 @@ + + + + + diff --git a/src/modules/base/static/svg/icon-menu.svg b/src/modules/base/static/svg/icon-menu.svg new file mode 100644 index 0000000..a53f8ba --- /dev/null +++ b/src/modules/base/static/svg/icon-menu.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-message.svg b/src/modules/base/static/svg/icon-message.svg new file mode 100644 index 0000000..0110fc8 --- /dev/null +++ b/src/modules/base/static/svg/icon-message.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-monitor.svg b/src/modules/base/static/svg/icon-monitor.svg new file mode 100644 index 0000000..5fa7b77 --- /dev/null +++ b/src/modules/base/static/svg/icon-monitor.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-new.svg b/src/modules/base/static/svg/icon-new.svg new file mode 100644 index 0000000..3ebc4bf --- /dev/null +++ b/src/modules/base/static/svg/icon-new.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-news.svg b/src/modules/base/static/svg/icon-news.svg new file mode 100644 index 0000000..5cec609 --- /dev/null +++ b/src/modules/base/static/svg/icon-news.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-notice.svg b/src/modules/base/static/svg/icon-notice.svg new file mode 100644 index 0000000..138d9b8 --- /dev/null +++ b/src/modules/base/static/svg/icon-notice.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-params.svg b/src/modules/base/static/svg/icon-params.svg new file mode 100644 index 0000000..9683e72 --- /dev/null +++ b/src/modules/base/static/svg/icon-params.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-pending.svg b/src/modules/base/static/svg/icon-pending.svg new file mode 100644 index 0000000..90be6b0 --- /dev/null +++ b/src/modules/base/static/svg/icon-pending.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-phone.svg b/src/modules/base/static/svg/icon-phone.svg new file mode 100644 index 0000000..8910aab --- /dev/null +++ b/src/modules/base/static/svg/icon-phone.svg @@ -0,0 +1,9 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-pic.svg b/src/modules/base/static/svg/icon-pic.svg new file mode 100644 index 0000000..0c68a0d --- /dev/null +++ b/src/modules/base/static/svg/icon-pic.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-question.svg b/src/modules/base/static/svg/icon-question.svg new file mode 100644 index 0000000..8fbe696 --- /dev/null +++ b/src/modules/base/static/svg/icon-question.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-rank.svg b/src/modules/base/static/svg/icon-rank.svg new file mode 100644 index 0000000..60916c5 --- /dev/null +++ b/src/modules/base/static/svg/icon-rank.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-scan.svg b/src/modules/base/static/svg/icon-scan.svg new file mode 100644 index 0000000..6e1e2c7 --- /dev/null +++ b/src/modules/base/static/svg/icon-scan.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-search.svg b/src/modules/base/static/svg/icon-search.svg new file mode 100644 index 0000000..166aebf --- /dev/null +++ b/src/modules/base/static/svg/icon-search.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-system.svg b/src/modules/base/static/svg/icon-system.svg new file mode 100644 index 0000000..a370d13 --- /dev/null +++ b/src/modules/base/static/svg/icon-system.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-tag.svg b/src/modules/base/static/svg/icon-tag.svg new file mode 100644 index 0000000..84adbce --- /dev/null +++ b/src/modules/base/static/svg/icon-tag.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-task.svg b/src/modules/base/static/svg/icon-task.svg new file mode 100644 index 0000000..1a3ff3b --- /dev/null +++ b/src/modules/base/static/svg/icon-task.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-theme.svg b/src/modules/base/static/svg/icon-theme.svg new file mode 100644 index 0000000..7833dde --- /dev/null +++ b/src/modules/base/static/svg/icon-theme.svg @@ -0,0 +1,12 @@ + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-time.svg b/src/modules/base/static/svg/icon-time.svg new file mode 100644 index 0000000..7ac6296 --- /dev/null +++ b/src/modules/base/static/svg/icon-time.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-track.svg b/src/modules/base/static/svg/icon-track.svg new file mode 100644 index 0000000..8ecd23a --- /dev/null +++ b/src/modules/base/static/svg/icon-track.svg @@ -0,0 +1,12 @@ + + + + + + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-upload.svg b/src/modules/base/static/svg/icon-upload.svg new file mode 100644 index 0000000..f68cdf1 --- /dev/null +++ b/src/modules/base/static/svg/icon-upload.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-user.svg b/src/modules/base/static/svg/icon-user.svg new file mode 100644 index 0000000..66d8df7 --- /dev/null +++ b/src/modules/base/static/svg/icon-user.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-video.svg b/src/modules/base/static/svg/icon-video.svg new file mode 100644 index 0000000..1eee5b9 --- /dev/null +++ b/src/modules/base/static/svg/icon-video.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-warn.svg b/src/modules/base/static/svg/icon-warn.svg new file mode 100644 index 0000000..3eab18d --- /dev/null +++ b/src/modules/base/static/svg/icon-warn.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/static/svg/icon-workbench.svg b/src/modules/base/static/svg/icon-workbench.svg new file mode 100644 index 0000000..e67c8ed --- /dev/null +++ b/src/modules/base/static/svg/icon-workbench.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/base/store/app.ts b/src/modules/base/store/app.ts new file mode 100644 index 0000000..5e65b68 --- /dev/null +++ b/src/modules/base/store/app.ts @@ -0,0 +1,59 @@ +import { defineStore } from "pinia"; +import { reactive, ref } from "vue"; +import { merge } from "lodash-es"; +import { useBrowser } from "/@/cool"; +import { storage } from "/@/cool/utils"; +import { config } from "/@/config"; + +export const useAppStore = defineStore("app", function () { + const { browser, onScreenChange } = useBrowser(); + + // 基本信息 + const info = reactive({ + ...config.app + }); + + // 是否折叠 + const isFold = ref(false); + + // 事件 + const events = reactive<{ [key: string]: any[] }>({ + hasToken: [] + }); + + // 折叠 + function fold(v?: boolean) { + if (v === undefined) { + v = !isFold.value; + } + + isFold.value = v; + } + + // 设置基本信息 + function set(data: any) { + merge(info, data); + storage.set("__app__", info); + } + + // 添加事件 + function addEvent(name: string, func: any) { + if (func) { + events[name].push(func); + } + } + + // 监听屏幕变化 + onScreenChange(() => { + isFold.value = browser.isMini; + }); + + return { + info, + isFold, + fold, + events, + set, + addEvent + }; +}); diff --git a/src/modules/base/store/index.ts b/src/modules/base/store/index.ts new file mode 100644 index 0000000..8707b0f --- /dev/null +++ b/src/modules/base/store/index.ts @@ -0,0 +1,18 @@ +import { useAppStore } from "./app"; +import { useMenuStore } from "./menu"; +import { useProcessStore } from "./process"; +import { useUserStore } from "./user"; + +export function useStore() { + const app = useAppStore(); + const menu = useMenuStore(); + const process = useProcessStore(); + const user = useUserStore(); + + return { + app, + menu, + process, + user + }; +} diff --git a/src/modules/base/store/menu.ts b/src/modules/base/store/menu.ts new file mode 100644 index 0000000..b643677 --- /dev/null +++ b/src/modules/base/store/menu.ts @@ -0,0 +1,189 @@ +import { defineStore } from "pinia"; +import { ref } from "vue"; +import { deepTree, revDeepTree, storage } from "/@/cool/utils"; +import { cloneDeep, isArray, isEmpty, orderBy } from "lodash-es"; +import { router, service } from "/@/cool"; +import { revisePath } from "../utils"; +import { Menu } from "../types"; +import { config } from "/@/config"; + +// 本地缓存 +const data = storage.info(); + +export const useMenuStore = defineStore("menu", function () { + // 视图路由 + const routes = ref([]); + + // 菜单组 + const group = ref(data["base.menuGroup"] || []); + + // 左侧菜单列表 + const list = ref([]); + + // 权限列表 + const perms = ref(data["base.menuPerms"] || []); + + // 设置左侧菜单 + function setMenu(i: number = 0) { + // 显示分组显示菜单 + if (config.app.menu.isGroup) { + list.value = group.value.filter((e) => e.isShow)[i]?.children || []; + } else { + list.value = group.value; + } + } + + // 设置权限 + function setPerms(list: Menu.List) { + function deep(d: any) { + if (typeof d == "object") { + if (d.permission) { + if (d.namespace) { + d._permission = {}; + for (const i in d.permission) { + d._permission[i] = + list.findIndex((e) => + e + .replace(/:/g, "/") + .includes(`${d.namespace.replace("admin/", "")}/${i}`) + ) >= 0; + } + } else { + console.error("namespace is required", d); + } + } else { + for (const i in d) { + deep(d[i]); + } + } + } + } + + perms.value = list; + storage.set("base.menuPerms", list); + + deep(service); + } + + // 设置视图 + function setRoutes(list: Menu.List) { + // 获取第一个菜单路径 + const fp = getPath(group.value); + + // 查找符合路由 + const route = list.find((e) => (e.meta!.isHome = e.path == fp)); + + if (route) { + const home = router.getRoutes().find((e) => e.meta.isHome); + + // 判断是否存在 + if (!home) { + router.append([ + { + ...route, + path: "/", + name: "home" + } + ]); + } else { + Object.assign(home.meta, route.meta); + } + } + + routes.value = list.filter((e) => e.type == 1); + } + + // 设置菜单组 + function setGroup(list: Menu.List) { + group.value = orderBy(deepTree(list), "orderNum"); + storage.set("base.menuGroup", group.value); + } + + // 获取菜单,权限信息 + async function get() { + function next(res: { menus: Menu.List; perms?: any[] }) { + const list = res.menus + ?.filter((e) => e.type != 2) + .map((e) => { + const path = revisePath(e.router || String(e.id)); + const isShow = e.isShow === undefined ? true : e.isShow; + + return { + ...e, + path, + isShow, + meta: { + ...e.meta, + label: e.name, // 菜单名称的唯一标识 + keepAlive: e.keepAlive || 0 + }, + name: `${e.name}-${e.id}`, // 避免重复命名之前的冲突 + children: [] + }; + }); + + // 设置权限 + setPerms(res.perms || []); + + // 设置菜单组 + setGroup(list); + + // 设置视图路由 + setRoutes(list); + + // 设置菜单 + setMenu(); + + return list; + } + + // 自定义菜单 + if (!isEmpty(config.app.menu.list)) { + next({ + menus: revDeepTree(config.app.menu.list || []) + }); + } else { + // 动态菜单 + await service.base.comm.permmenu().then(next); + } + } + + // 获取菜单路径 + function getPath(data: Menu.Item | Menu.List) { + const list = isArray(data) ? data : [data]; + + let path = ""; + + function deep(arr: Menu.List) { + arr.forEach((e: Menu.Item) => { + switch (e.type) { + case 0: + deep(e.children || []); + break; + case 1: + if (!path) { + path = e.path; + } + break; + } + }); + } + + deep(list); + + return path; + } + + return { + routes, + group, + list, + perms, + get, + setPerms, + setMenu, + setRoutes, + setGroup, + getPath + }; +}); diff --git a/src/modules/base/store/process.ts b/src/modules/base/store/process.ts new file mode 100644 index 0000000..36b5ed8 --- /dev/null +++ b/src/modules/base/store/process.ts @@ -0,0 +1,60 @@ +import { defineStore } from "pinia"; +import { ref } from "vue"; +import { Process } from "../types"; + +export const useProcessStore = defineStore("process", function () { + const list = ref([]); + + // 添加 + function add(data: any) { + list.value.forEach((e: Process.Item) => { + e.active = false; + }); + + if (!data.meta?.isHome && data.meta?.process !== false) { + const index = list.value.findIndex((e) => e.path === data.path); + + if (index < 0) { + list.value.push({ + ...data, + active: true + }); + } else { + Object.assign(list.value[index], data, { active: true }); + } + } + } + + // 移除 + function remove(index: number) { + list.value.splice(index, 1); + } + + // 设置 + function set(data: Process.Item[]) { + list.value = data; + } + + // 清空 + function clear() { + list.value = []; + } + + // 设置标题 + function setTitle(title: string) { + const item = list.value.find((e) => e.active); + + if (item) { + item.meta.label = title; + } + } + + return { + list, + add, + remove, + set, + clear, + setTitle + }; +}); diff --git a/src/modules/base/store/user.ts b/src/modules/base/store/user.ts new file mode 100644 index 0000000..c3fe4d8 --- /dev/null +++ b/src/modules/base/store/user.ts @@ -0,0 +1,89 @@ +import { defineStore } from "pinia"; +import { ref } from "vue"; +import { storage } from "/@/cool/utils"; +import { service, router } from "/@/cool"; +import { config } from "/@/config"; + +// 本地缓存 +const data = storage.info(); + +export const useUserStore = defineStore("user", function () { + // 标识 + const token = ref(config.test.token || data.token); + + // 设置标识 + function setToken(data: { + token: string; + expire: number; + refreshToken: string; + refreshExpire: number; + }) { + // 请求的唯一标识 + token.value = data.token; + storage.set("token", data.token, data.expire); + + // 刷新 token 的唯一标识 + storage.set("refreshToken", data.refreshToken, data.refreshExpire); + } + + // 刷新标识 + async function refreshToken(): Promise { + return new Promise((resolve, reject) => { + service.base.open + .refreshToken({ + refreshToken: storage.get("refreshToken") + }) + .then((res) => { + setToken(res); + resolve(res.token); + }) + .catch((err) => { + logout(); + reject(err); + }); + }); + } + + // 用户信息 + const info = ref(data.userInfo); + + // 设置用户信息 + function set(value: any) { + info.value = value; + storage.set("userInfo", value); + } + + // 清除用户 + function clear() { + storage.remove("userInfo"); + storage.remove("token"); + token.value = ""; + info.value = null; + } + + // 退出 + async function logout() { + clear(); + router.clear(); + router.push("/login"); + } + + // 获取用户信息 + async function get() { + return service.base.comm.person().then((res) => { + set(res); + return res; + }); + } + + return { + token, + info, + get, + set, + logout, + clear, + setToken, + refreshToken + }; +}); diff --git a/src/modules/base/types/index.d.ts b/src/modules/base/types/index.d.ts new file mode 100644 index 0000000..23e76ab --- /dev/null +++ b/src/modules/base/types/index.d.ts @@ -0,0 +1,44 @@ +import { RouteComponent, RouteLocationNormalized } from "vue-router"; + +export declare namespace Menu { + enum Type { + "目录" = 0, + "菜单" = 1, + "权限" = 2 + } + + interface Item { + id: number; + parentId: number; + path: string; + router?: string; + viewPath?: string; + type: Type; + name: string; + icon: string; + orderNum: number; + isShow: number | boolean; + keepAlive?: number; + meta?: { + label?: string; + keepAlive?: number | boolean; + iframeUrl?: string; + isHome?: boolean; + [key: string]: any; + }; + children?: Item[]; + component?: RouteComponent; + redirect?: string; + [key: string]: any; + } + + type List = Item[]; +} + +export declare namespace Process { + interface Item extends RouteLocationNormalized { + active: boolean; + } + + type List = Item[]; +} diff --git a/src/modules/base/utils/index.ts b/src/modules/base/utils/index.ts new file mode 100644 index 0000000..814ea94 --- /dev/null +++ b/src/modules/base/utils/index.ts @@ -0,0 +1,21 @@ +export function revisePath(path: string) { + if (!path) { + return ""; + } + + return path[0] == "/" ? path : `/${path}`; +} + +export function createLink(url: string, id?: string) { + const link = document.createElement("link"); + link.href = url; + link.type = "text/css"; + link.rel = "stylesheet"; + if (id) { + link.id = id; + } + + setTimeout(() => { + document.getElementsByTagName("head").item(0)?.appendChild(link); + }, 0); +} diff --git a/src/modules/base/views/frame.vue b/src/modules/base/views/frame.vue new file mode 100644 index 0000000..ed8f0f2 --- /dev/null +++ b/src/modules/base/views/frame.vue @@ -0,0 +1,45 @@ + + + + + diff --git a/src/modules/base/views/info.vue b/src/modules/base/views/info.vue new file mode 100644 index 0000000..d965a05 --- /dev/null +++ b/src/modules/base/views/info.vue @@ -0,0 +1,102 @@ + + + + + diff --git a/src/modules/base/views/log.vue b/src/modules/base/views/log.vue new file mode 100644 index 0000000..cd05b13 --- /dev/null +++ b/src/modules/base/views/log.vue @@ -0,0 +1,147 @@ + + + diff --git a/src/modules/base/views/menu/components/exp.vue b/src/modules/base/views/menu/components/exp.vue new file mode 100644 index 0000000..db6307f --- /dev/null +++ b/src/modules/base/views/menu/components/exp.vue @@ -0,0 +1,99 @@ + + + diff --git a/src/modules/base/views/menu/components/imp.vue b/src/modules/base/views/menu/components/imp.vue new file mode 100644 index 0000000..8cc2888 --- /dev/null +++ b/src/modules/base/views/menu/components/imp.vue @@ -0,0 +1,112 @@ + + + diff --git a/src/modules/base/views/menu/index.vue b/src/modules/base/views/menu/index.vue new file mode 100644 index 0000000..b49dda4 --- /dev/null +++ b/src/modules/base/views/menu/index.vue @@ -0,0 +1,450 @@ + + + diff --git a/src/modules/base/views/param.vue b/src/modules/base/views/param.vue new file mode 100644 index 0000000..45c2385 --- /dev/null +++ b/src/modules/base/views/param.vue @@ -0,0 +1,207 @@ + + + diff --git a/src/modules/base/views/role.vue b/src/modules/base/views/role.vue new file mode 100644 index 0000000..679c89a --- /dev/null +++ b/src/modules/base/views/role.vue @@ -0,0 +1,161 @@ + + + diff --git a/src/modules/base/views/user/components/dept-list.vue b/src/modules/base/views/user/components/dept-list.vue new file mode 100644 index 0000000..581e217 --- /dev/null +++ b/src/modules/base/views/user/components/dept-list.vue @@ -0,0 +1,466 @@ + + + + + diff --git a/src/modules/base/views/user/components/user-move.vue b/src/modules/base/views/user/components/user-move.vue new file mode 100644 index 0000000..fea976b --- /dev/null +++ b/src/modules/base/views/user/components/user-move.vue @@ -0,0 +1,67 @@ + + + diff --git a/src/modules/base/views/user/index.vue b/src/modules/base/views/user/index.vue new file mode 100644 index 0000000..f2c7dcc --- /dev/null +++ b/src/modules/base/views/user/index.vue @@ -0,0 +1,330 @@ + + + diff --git a/src/modules/cs/components/index.vue b/src/modules/cs/components/index.vue new file mode 100644 index 0000000..a87a733 --- /dev/null +++ b/src/modules/cs/components/index.vue @@ -0,0 +1,186 @@ + + + + + diff --git a/src/modules/cs/components/message.vue b/src/modules/cs/components/message.vue new file mode 100644 index 0000000..2af1d6a --- /dev/null +++ b/src/modules/cs/components/message.vue @@ -0,0 +1,526 @@ + + + + + diff --git a/src/modules/cs/components/session.vue b/src/modules/cs/components/session.vue new file mode 100644 index 0000000..d3cb68b --- /dev/null +++ b/src/modules/cs/components/session.vue @@ -0,0 +1,223 @@ + + + + + diff --git a/src/modules/cs/components/tools/emoji.vue b/src/modules/cs/components/tools/emoji.vue new file mode 100644 index 0000000..8f46380 --- /dev/null +++ b/src/modules/cs/components/tools/emoji.vue @@ -0,0 +1,162 @@ + + + + + + + diff --git a/src/modules/cs/config.ts b/src/modules/cs/config.ts new file mode 100644 index 0000000..9378325 --- /dev/null +++ b/src/modules/cs/config.ts @@ -0,0 +1,10 @@ +import type { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return { + toolbar: { + order: 2, + component: import("./components/index.vue") + } + }; +}; diff --git a/src/modules/cs/hooks/index.ts b/src/modules/cs/hooks/index.ts new file mode 100644 index 0000000..df17a57 --- /dev/null +++ b/src/modules/cs/hooks/index.ts @@ -0,0 +1,5 @@ +export * from "./socket"; +export * from "./session"; +export * from "./message"; +export * from "./tools"; +export * from "./notice"; diff --git a/src/modules/cs/hooks/message.ts b/src/modules/cs/hooks/message.ts new file mode 100644 index 0000000..2e670fa --- /dev/null +++ b/src/modules/cs/hooks/message.ts @@ -0,0 +1,128 @@ +import { defineStore } from "pinia"; +import { computed, ref } from "vue"; +import type { Cs } from "../types"; +import dayjs from "dayjs"; +import { service } from "/@/cool"; +import { isEmpty, last } from "lodash-es"; +import { useSession } from "./session"; +import { uuid } from "/@/cool/utils"; +import { useTools } from "./tools"; +import { dateFormatter } from "../utils"; +import { useBase } from "/$/base"; + +export const useMessage = defineStore("cs.message", () => { + const session = useSession(); + const tools = useTools(); + const { user } = useBase(); + + // 加载中 + const loading = ref(false); + + // 未读消息数 + const unread = ref(0); + + // 所有页 + const data = ref<{ id: string; data: Cs.Msg[] }[]>([]); + + // 列表 + const list = computed(() => { + data.value.forEach((e) => { + if (e.data[0]) { + let date = e.data[0].createTime; + + if (date) { + e.data.forEach((a, i) => { + const d = dateFormatter(a.createTime); + + if (i == 0) { + a.date = d; + } + + if (dayjs(a.createTime).subtract(10, "minute").isAfter(dayjs(date))) { + a.date = d; + date = a.createTime; + } + }); + } + } + }); + + return data.value; + }); + + // 追加消息 + function append(item: Cs.Msg) { + const list = last(data.value)?.data; + + if (list) { + list.push({ + sessionId: session.info?.id, + type: 1, + nickName: user.info?.name, + avatarUrl: user.info?.headImg, + ...item + }); + + tools.scrollToBottom(); + } + } + + // 获取消息 + async function refresh(params?: any) { + loading.value = true; + + if (params.page == 1) { + data.value = []; + } + + await service.cs.msg + .page({ + sessionId: session.info?.id, + order: "createTime", + sort: "desc", + ...params + }) + .then((res) => { + if (isEmpty(data.value) || !isEmpty(res.list)) { + res.list.forEach((e) => { + if (e.type == 1) { + e.avatarUrl = e.adminUserHeadImg; + e.nickName = e.adminUserName; + } + }); + + data.value.unshift({ + id: uuid(), + data: res.list.reverse() as any[] + }); + } + }); + + loading.value = false; + } + + // 获取未读消息 + function getUnread() { + service.cs.msg.unreadCount().then((res) => { + unread.value = res; + }); + } + + // 读消息 + function read(id: number) { + service.cs.msg.read({ + msgIds: [id] + }); + } + + return { + data, + list, + loading, + unread, + append, + refresh, + read, + getUnread + }; +}); diff --git a/src/modules/cs/hooks/notice.tsx b/src/modules/cs/hooks/notice.tsx new file mode 100644 index 0000000..8b9690d --- /dev/null +++ b/src/modules/cs/hooks/notice.tsx @@ -0,0 +1,68 @@ +import Ding from "/$/cs/static/audio/ding.wav"; +import type { Cs } from "../types"; +import { ElNotification } from "element-plus"; +import { getEmoji } from "../utils"; +import { useSession } from "./session"; + +export function useNotice() { + const session = useSession(); + + // ding + const audio = new Audio(Ding); + + // 消息提示 + function create({ content, user, sessionId }: Cs.Msg, { open }: { open: () => Promise }) { + if (content) { + const icon = ; + + let message = content.data; + + switch (content.type) { + case "image": + message = "[图片]"; + break; + + case "voice": + message = "[语音]"; + break; + + case "video": + message = "[视频]"; + break; + + case "file": + message = "[文件]"; + break; + + case "location": + message = "[位置]"; + break; + + case "emoji": + message = ( + + ); + break; + } + + ElNotification({ + title: user.nickName, + message, + icon, + async onClick() { + await open(); + + // 切换会话 + session.setById(sessionId!); + } + }); + } + + // 播放声音 + audio?.play(); + } + + return { + create + }; +} diff --git a/src/modules/cs/hooks/session.ts b/src/modules/cs/hooks/session.ts new file mode 100644 index 0000000..26b1c33 --- /dev/null +++ b/src/modules/cs/hooks/session.ts @@ -0,0 +1,97 @@ +import { defineStore } from "pinia"; +import { reactive, ref } from "vue"; +import { Cs } from "../types"; +import { service } from "/@/cool"; + +export const useSession = defineStore("cs.session", () => { + // 详情 + const info = ref(); + + // 列表 + const list = ref([]); + + // 是否加载完 + const loaded = ref(false); + + // 加载状态 + const loading = ref(false); + + // 分页 + const pagination = reactive({ + page: 1, + size: 20 + }); + + // 请求参数 + const reqParams = { + order: "updateTime", + sort: "desc" + }; + + // 刷新 + async function refresh(params?: any) { + loading.value = true; + + // 合并参数 + Object.assign(reqParams, params); + + // 请求 + await service.cs.session + .page({ + ...pagination, + ...reqParams + }) + .then((res) => { + // 设置分页 + Object.assign(pagination, res.pagination); + + if (res.pagination.page == 1) { + list.value = res.list; + } else { + list.value.push(...res.list); + } + + // 是否加载完成 + loaded.value = res.pagination.total <= list.value.length; + + // 默认第一个 + if (!info.value) { + set(list.value[0]); + } + }); + + loading.value = false; + } + + // 设置会话 + function set(data?: Cs.Session) { + if (data) { + data.adminUnreadCount = 0; + } + + info.value = data; + } + + // 通过id设置会话 + function setById(sessionId: number) { + const item = list.value.find((e) => e.id == sessionId); + set(item); + } + + // 清空会话 + function clear() { + info.value = undefined; + } + + return { + info, + list, + set, + setById, + clear, + pagination, + loaded, + loading, + refresh + }; +}); diff --git a/src/modules/cs/hooks/socket.ts b/src/modules/cs/hooks/socket.ts new file mode 100644 index 0000000..dce47a9 --- /dev/null +++ b/src/modules/cs/hooks/socket.ts @@ -0,0 +1,95 @@ +import type { Cs } from "../types"; +import { defineStore } from "pinia"; +import io, { type Socket } from "socket.io-client"; +import { useSession } from "./session"; +import { useBase } from "/$/base"; +import { config } from "/@/config"; +import { useMessage } from "./message"; +import { useCool } from "/@/cool"; + +export const useSocket = defineStore("cs.socket", () => { + const { mitt } = useCool(); + const { user } = useBase(); + const session = useSession(); + const message = useMessage(); + + let client = undefined as Socket | undefined; + + function connect() { + if (!user.token) { + return false; + } + + if (client) { + client.disconnect(); + client = undefined; + } + + if (!client) { + client = io(config.host + "/cs", { + transports: ["websocket", "polling"] + }); + + client.auth = { + isAdmin: true, + token: user.token + }; + + client.on("connect", () => { + console.log("[cs] connect"); + }); + + client.on("disconnect", () => { + console.log("[cs] disconnect"); + }); + + client.on("msg", (data: Cs.Msg) => { + if ((data.type == 1 && data.userId != user.info?.id) || data.type == 0) { + // 判断是否是当前会话 + if (data.sessionId == session.info?.id) { + message.append({ + ...data.user, + ...data + }); + } else { + // 设置其他会话信息 + session.list.forEach((e) => { + if (e.id == data.sessionId) { + if (!e.adminUnreadCount) { + e.adminUnreadCount = 0; + } + + e.adminUnreadCount += 1; + e.lastMsg = data; + } + }); + } + + // 发送事件 + mitt.emit("cs.msg", data); + } + }); + } + } + + function send(content: Cs.Content) { + if (client) { + // 发送事件 + client.emit("send", { + sessionId: session.info?.id, + content + }); + + // 追加消息 + message.append({ content }); + } else { + console.log("[cs] client error"); + } + } + + return { + connect, + client, + send + }; +}); diff --git a/src/modules/cs/hooks/tools.ts b/src/modules/cs/hooks/tools.ts new file mode 100644 index 0000000..354a862 --- /dev/null +++ b/src/modules/cs/hooks/tools.ts @@ -0,0 +1,28 @@ +import { defineStore } from "pinia"; +import { nextTick, ref } from "vue"; + +export const useTools = defineStore("cs.tools", () => { + // 是否展开 + const isExpand = ref(true); + + // 收起、展开 + function expand(value?: boolean) { + isExpand.value = value === undefined ? !isExpand.value : value; + } + + // 滚动到底部 + async function scrollToBottom() { + await nextTick(); + + document.querySelector(".cs-message .el-scrollbar__wrap")?.scroll({ + top: 9999, + behavior: "smooth" + }); + } + + return { + isExpand, + expand, + scrollToBottom + }; +}); diff --git a/src/modules/cs/index.ts b/src/modules/cs/index.ts new file mode 100644 index 0000000..b740056 --- /dev/null +++ b/src/modules/cs/index.ts @@ -0,0 +1,13 @@ +import { useMessage, useSession, useSocket } from "./hooks"; + +export function useCs() { + const socket = useSocket(); + const message = useMessage(); + const session = useSession(); + + return { + socket, + message, + session + }; +} diff --git a/src/modules/cs/static/audio/ding.wav b/src/modules/cs/static/audio/ding.wav new file mode 100644 index 0000000000000000000000000000000000000000..0479aa8fec4816e3ea95d6595d18caa2dbf60578 GIT binary patch literal 3534 zcmc(h4`iHo7{|XmC*ou$i;9Yfh^WdU(#ocyVv8)Aimck`s^Y9wXR3>;h>2(;qAIE? zs%omLDyn8;PE}RaOhrXiM8#A^MMOoMbIysl?(@9Ad-L9{mzgQMpZDJP{r;Xm-{<>0 z@4a`pr@Q;vr_y&{AN?l&<4}F2q)%9%eJ(W|l|rgiZ_knb8t|UJ?r(dJboWQO?maOa z|6*TD+n$y^t*ttBV(7#eQIz}0Yx5(=%N23tCjW7~{X45;yN~UJSi#5c+%n9+E_Dun zup5@GQnnuxE6(9^ewew~s#JczX50Ix&wrhQV17zP|2re&Tb5sehJJNrp0a*T(x$g^ zp4JY{DXX-S-b!(m#UjVCp_JCNqGgUXWx&c*j$ayNk+qCvL^~8^XI*c#gk2IZVO6Ew zYEqk8wNLxir(-&ZW5V;eZ!N5y{ z#OWYjPH$j4t*cx4Ch>lNUy3UC!tr~Z)u_&U)g9AU_-vxiLpV+5eI_)ed-{V&Iast) zOI*V`|NeqTsIFgM66qcLw>7?%WsHo!=@m8AqQoJ^@c989j;c$|{1~M)uLM<0C7CQ@9G9ouj(3^66l!JPHK#qUhp4;S2MP+iDQ=6sq-Fn zJO@dsf{u1!dyZO1h<{M^_#5ZT^upD*jKbAu*`{td#9$kPc?&c10yX?j-{CIn>ClUn zO~wG<2bh%#W@u)sv-eOj%JdxHs{8;->;hY-i+8-)nSkj6YILE^2o=ZSW$I*!^o;)6 zgiY}Mm_s-f3VAGFoDO5|K!8BG0a;}iOovKFI zkHf5-{>cV;v?<+94!g&_n)m}Ykyy@D@G|a*OD(X$Mn__Z0 z;dqV}qtYFf8Vtg&llm8lIDz7m%+LyP zdzrIgj$U+MhOH_nD!|wvfQ7fcjuUuIwJgvT9b6- z1;01#I_M`W&L$n@so&Iwa^)m?`XI zlKQ-9e1+|Oj(NB@!m|%$o3ODbJwY^k>gwp!0rr}iiWQ!LC#bxbe;T_`rpcS2Iq;{% zvEPCg(EZp)c`g>1zFFcVndwF*y9J&h;=ClvEOAmWZ3Jzit2y+1Nrrj$r5Yl%qGA;q zy7gnc8?4WGy3HPIvRY{I$*ebzx1h`mDPRXm-$I4WVhiv4TQEL>`d(7+Ui z@Ld9Z%Q-~f>p^RXmce=nbPZNftm{GT_dN-|N`_5hRD)G>&QZlXR+6h?bPfMBwM4Po zN%RPQE<%#42<)mk%HfvuBBY35=nnXnW05A8i%^kA%dkz8KS>o~vQ=SO!IizE);Tt) zq5>>RoDeq2Ei?jFfjR0(+uA9ep{%r3C?|KpTGjg$X>f*^dHm0>PIfCc`(RUzWA zAewQqeIIcwi?PBCJ`3~nKYn~amVfW|gXlh9oXtmr$lD{qj(=t6K!YH%<5#>}sE5z} h-=7OBF6C9cUwG$p4fJqGAs^r7|2>zgYkQ0u{{qX%l1l&p literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/angry-face.png b/src/modules/cs/static/emoji/angry-face.png new file mode 100644 index 0000000000000000000000000000000000000000..bd0b383210b1715293398434c1e93825742fdc99 GIT binary patch literal 3669 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/anguished-face.png b/src/modules/cs/static/emoji/anguished-face.png new file mode 100644 index 0000000000000000000000000000000000000000..fa4678b59d5aeeffa2134e8dc4d0838efddf7abf GIT binary patch literal 3584 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 BUP1r> literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/confused-face.png b/src/modules/cs/static/emoji/confused-face.png new file mode 100644 index 0000000000000000000000000000000000000000..b01d70e81b11b0b8d5e1204da063a07e1b8af876 GIT binary patch literal 3670 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 tehq(X% literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/crying-face.png b/src/modules/cs/static/emoji/crying-face.png new file mode 100644 index 0000000000000000000000000000000000000000..c5aae1ce8fba3448325f11fd2db2cca39b4b2559 GIT binary patch literal 3824 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/disappointed-but-relieved-face.png b/src/modules/cs/static/emoji/disappointed-but-relieved-face.png new file mode 100644 index 0000000000000000000000000000000000000000..f4f6e3eb594055bb555cfaa577fc9614a0a25f3c GIT binary patch literal 3709 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/disappointed-face.png b/src/modules/cs/static/emoji/disappointed-face.png new file mode 100644 index 0000000000000000000000000000000000000000..aa3e501ac4bf447b92e59dce19bc94451ff64f3a GIT binary patch literal 3584 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 BJEQ;r literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/drooling-face.png b/src/modules/cs/static/emoji/drooling-face.png new file mode 100644 index 0000000000000000000000000000000000000000..a7e254504ae64b40ac3305a4fce22cfd297abbd8 GIT binary patch literal 3552 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 a#sC0D%Jxpe2kQ+00000gj5k{$<_&TUv_0XfLbz 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-savouring-delicious-food.png b/src/modules/cs/static/emoji/face-savouring-delicious-food.png new file mode 100644 index 0000000000000000000000000000000000000000..7f3d4344c71b57a2451a506de0afdf55b38ad4b2 GIT binary patch literal 3701 zcmc&$`8O2)_nsMB*2XSNc9Td%*%C8U23dwgwyX_$E0Luz_ANB_Fo_x!Au5!mtYhEJ zkTlk;*>{H5$M>)J-gEBroO{lF&U1fx&be`>#y8nn1X&;u2s>I|4|Br$|JSGJPS8nN z_sI#--$&j;LLlWYSZR(7Co|Lsa}xzA?GavpKP{Ds
>B^>Cze2!Wz`ESf-q+7y!`Rv~~Oe?Nj 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 T@mRcn00000NkvXXu0mjfB!>uD literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-cowboy-hat.png b/src/modules/cs/static/emoji/face-with-cowboy-hat.png new file mode 100644 index 0000000000000000000000000000000000000000..5801afc7807f5e9b3eec3cf305e43426cae3c663 GIT binary patch literal 3683 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 B=M?|| literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-finger-covering-closed-lips.png b/src/modules/cs/static/emoji/face-with-finger-covering-closed-lips.png new file mode 100644 index 0000000000000000000000000000000000000000..3e37c47b1a547d5b1942199161ad28278fa6bff3 GIT binary patch literal 3797 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 LNkvXXu0mjfJjzFj literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-head-bandage.png b/src/modules/cs/static/emoji/face-with-head-bandage.png new file mode 100644 index 0000000000000000000000000000000000000000..00471c046c47524d8cc9df502bea2226716bb857 GIT binary patch literal 3586 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-look-of-triumph.png b/src/modules/cs/static/emoji/face-with-look-of-triumph.png new file mode 100644 index 0000000000000000000000000000000000000000..fa3bb8e3d5325e354b332f6c6df3a6b6bf6ba174 GIT binary patch literal 3809 zcmV<74j%D|P)Px&08mU+MMrQN5>3Fej#?Trlks}K046tYtS%+S=ZQv>y+7{G%@ zvzL3RsJ4Hh&;9=X|NsB`{r>;eEdI6_|K2SB#x?)ZDgCn%b2tF|tPKCyDfIgP;D`hN zyCVO}F!1*O{I3rG+bs8~3Gtu_({up-xFG+^Hvh>g|M+`kw#qYmZR^|DXZ?*+Bo@Dc$V;=#&KGj|JCz0srq`|Nh?p`>6lV zGXMS9|L983VZPxz>+6~y_~JqF+duQ^oa(?m(B19DrX|m{ zT>SE)(y}bvn;Grhis;K>ubhwl>WA{qGM&Qnw~#}odo=&y9?`jrxwf&gz|PdNcA|9@ z=dK_A%0QNAA?MFHzn^AdCII5>_2$rc(ZHVn+Fb48ZuPJqj~(Q6WOC*=C~;D@9pdB;GlCR-N;Njlm)Vpvz z1OLz&b3`4RePEG&bw=YXi~s-tKy*@0Qvm!FQ9v>N0S6(~IH=x-5&rOGbN*7@oA@5I zlls18pX`SF!pz#aZQb_#l)&xN-jB4W;q#C1(bd?wwS#x7wMe+i;{X5)Gf6~2R9M5T z)@e{v=Nboa2cZcLazSmK@dIu%{h+PAez1P%y)!Wc2;n4z3IPHbF#=WsAuItQWwGoA zv7oX95fH+@nu~zwm3{`#BfY8&nU=J01Yhriixq^(Q+D~i$1!(THQ$z^2~Z!6xG zl_e)JzdpRv=ucWy?L&uINy%juj~?B-chAugem#0r0T}DE{o2$&!R*|{N&@9x;DxM6 zndhlfp0Y@%s7vR-O7h#a^AA!K-9x)dfw^}nOLpSczn63CqGw^zIg--(1Cys3ZHh@s zt~eK!aN@X%lzfJaw*{ky*M|%@EZ}U4ETG5C#7S1tXPNA_kbad44 zLE5DK5l7h7BYo1lFxd(eiIG8n$r@huj7nAaTQfc=VJEgQ;r?$*WrV8tRjViDlsf2j zL(9uwmE#d_`Vw06|F=ZZ-HV;p$b^Q$QFHSM)k23%=D8f1tfNN~VJ_+IYfXr3C8hVy zN$X%vX;NrcVPSYUQ1_om%*}f{5@ZQ79C$G~3e4yrFyY~ag@w7vr8x)RbM7d+x0NQ@ z)xZvrfa>V2mPEYlh)hr?5*&%NrvVh~bag3`N^=gq`+3sd72{V{+?AD;<%AR2&?A+Y z_jE<7)AiC^65SFRP9Dw5>guYo^NZQFaeX~nd!e!1nwpv$K)Jg+xjQ)(E=VP+mT)po z3&;Tm4!OI-o*Oka`Ow_n_04VF7UO4GTnrGz-rgQ|J2@>>oA;^iDQnNzrkqm7gZ$#+;wT7`5SG;|c{1&e)2Wj5xRcBFH_)Ca*!#exH0LXw zcM7t7ke^>vR0Mn6qTEiS*iS!^PLuz+Q#CqmkJj9<5>-@`pP&EWosx79$ANNN2`n7H zfCHN{GBR>=b4~qn_H14CsMrCWnJTb^s2@51G&#ui)av5#7l#vk|3(iPhOwN}pm)ut9jALpW8XAjI2m8KT z$xWPY3OQKZ~o3ZUJZJCPa!$-~Ud*x1COey+K>xvf6X!Ff53&JKZfb8WCZ=HP5$W@hH$anc1K z715p+xxD00#$2B?U=!{n_j7bL|9`h(bdQ|njF#2FAjlXw0uQ?t=(-xBY`^Xk?8Rd8yj{fT#0e0zEPcDG9MSDUE{#T*`!jHMXz8czMlM)wzdjlv4K}bT> z)GJgL4!>j4jE@+;AQ7M-1o4J~XPKCoh@{Se#@c7Y!|m!v(yEsR zuZL>|8pqccCWJbJ5pC8yNT@qtNT@dkH0SHi({9dv+ zT#Fo2aL9tJqmoc_0GZadmsdx;F=%>&ZcRfFbZ`1SY~Z(u>hktl zQ_?{b@(MohYNNmWK+-luy?L?VL&@=ml7`DA@CB(qY-yowtiHYuQeQvT)-n-xR}!I= zlxM;gaG;>0sG#K$!Gr&c<&t>b-g)iO%82TryK>R%Y;^k#g&QZtv~h{sbbqKiLK)qj z=MA4lTu_{=&E{1PHvKcWBrNe3B}!>!<75*dAIt>FHW7Z{?wp-DB@!8QHdacNl0_bJ z(1gy%pV43SsMXm_mc#<7U81UNRE#W%h^BcAsgs>KI6++QA5k<`swC~mfl~$(GF<%y zkh&d}6zqkh@}Wjs-;gOeO|+D`3WAE}n+Wc{L0@0n#-Va13zamq%p0RO-FyylecKm8)a9BGlYZzr1+cI5fvlmcszAPn>?y@37nmkBPHjzWkge(qZ|+(Mze z(;6)ObKh-lDxIE+x6(v|DLw(#>FHKhX;xOO^z;Cq$3MfH5Of+&Dk1Ox2lc(shEMlh z7SU<25#Es?TCklQ^BKDKok=K# zD1DHS@WSu@@7=q=d#8*_8P63(X@;eSc4MCvrV6x}m$YR}(dD#yKEA$yDbDw2&TA0( z*C`)2w#axOaG9`Jb zyz4mOo_%wgF)3UbE!Eajit@ne&!(1Enq^IA{YNEPq?F4<@}YTqyb_K^JQYIFoacpO zW4+z$b~+DWoldvc)|F(A5IB+Rn|B5+n1nv!l)FWq2Sgn|Rk^x;eNZ^ToV2oHn15CJ@TLVkzwE#HjLk<%M6wn^g=!wsjm)w-iwZ}$BF Xucez-Fvj-$00000NkvXXu0mjf#!s=4 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-medical-mask.png b/src/modules/cs/static/emoji/face-with-medical-mask.png new file mode 100644 index 0000000000000000000000000000000000000000..8b237f5b51cc5b1f826766e7bd29468f1a987d59 GIT binary patch literal 3684 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-one-eyebrow-raised.png b/src/modules/cs/static/emoji/face-with-one-eyebrow-raised.png new file mode 100644 index 0000000000000000000000000000000000000000..c14f913df3a32a531b221a788075df65a8c30dd9 GIT binary patch literal 3483 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 RJHG$`002ovPDHLkV1jy~pG5!w literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-open-mouth.png b/src/modules/cs/static/emoji/face-with-open-mouth.png new file mode 100644 index 0000000000000000000000000000000000000000..6b9af1edfb0a617b49b2883a01b412297daadb3b GIT binary patch literal 3667 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-pleading-eyes.png b/src/modules/cs/static/emoji/face-with-pleading-eyes.png new file mode 100644 index 0000000000000000000000000000000000000000..39012990a2ae0fd407055a003fc90822a116a214 GIT binary patch literal 3498 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 Y0JlbC5vj`2HUIzs07*qoM6N<$g8E*?y#N3J literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-rolling-eyes.png b/src/modules/cs/static/emoji/face-with-rolling-eyes.png new file mode 100644 index 0000000000000000000000000000000000000000..3c1c93331f479b73ae3d015fd230ee7c000a1251 GIT binary patch literal 3461 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-stuck-out-tongue-and-winking-eye.png b/src/modules/cs/static/emoji/face-with-stuck-out-tongue-and-winking-eye.png new file mode 100644 index 0000000000000000000000000000000000000000..3c6c27629369c0c7a08b16cad9f09dcb8a1d9efe GIT binary patch literal 3561 zcmVPx&08mU+MMrQ<;(h?fa~{un6wYk`yL2+gd?M$c8@+ia z%6TBlX#ve_0LyX?%Weh6Y6SI(0Pl$e+j#)cbPDvI6|__s>WvcDcmTy{0_u<#%4`JI zbp`B(1;+?hm?t^ri2>1Z1Ilg(=YInHoC5cm3GR~YW9qztzCz2(%|e_oo>5ry2jzGw|^4|JX49)iM9hH1qQB z({%vR?0N8&4|M++T0Rj513*3YO?3@Vy_<#TX)c>#p|Ni0Vmj>R5 z0_Bnf|Mgz~_he5dXj<|GWhJY_e=J0RQ)I z|NO-N%q{PY0RQV?|EB`~;WeOO0hU++{@z~c>F58(4f{Na|NYwk(JTCY!2M&a?X4ev zK>_V6cJQ17{FMOzodNq(o&0yZpr4-O;^6=Lx&QmF^`-^?{LcT!C;!ku|LQ*chsgX& zmWD+G?3V-eodW;vY}we?icSH~&dmF*5AwDok5K~q`uG3ZL*}F#qh$uf!@-V|LunV@sF}HF#LqZs;a8q9A%s-CI8Y)|MQs3h!Xs9wzzr>oLd94vaz|S zQu{%U$0AOkj935ZeE-cO|IZ8mx(~;V4pt`xZ+ru;Q$0O9bN&njw z^xQz&q8hx56y>)%)RY$g))Tp!NAcWL$ZixD7Z?BaiS100$fz#q)J){ZN1sg;(Kc7U zY6Jh^8ag>Sb8>L^)G7Re#jc}e=R${+gI&x@MTy~ya{+t~2v#nj95j@8G(UHfk7000U^Nklc~leE9tUtF z2?+$yBXUIj<5~5zYM1|71405%p~{j#f{Hc*LO>ffkpxHt6ck9n1)f-T42#H)vMC^z z9Yg^uR5n@M>Qaxmx2MZ_?|1J^!m76Ky)Pz{%v?VH?(fb_f|l0*C$tZapwk#SIt&{9 zfBIi^DJE--jK2Ev%dd=#)|gOqY5$GF&^9*u>~n5U&q17?9`5I#85vU;e@~(7nizeN z4o3Ib;NZ`Y!NIX^(0slyGSQ{~4aRu&Dxu%O?!liEB}Mm|HuY_)E0QEy4|X5)6FRIm z{;QOZ$!e+4uY1f&(iH#A?jOkP4iD^04j$|F6G~Sb>--gmVuY$0tEjUJ-#shg`}!n< zP}QnMln)lgFj+5!NmVrM`vwvouEyY1INEckhY@nm}?!mx9Ut!F8%A6B%1xzgA+O=y%wY#>We68U$!MpImqM{-gD(jI(FPeR| z=r4>Bo-8c9dGjU=+J&-p%9|3l;zKY51{M}32xAs5p1tzX(w@q~!ouX_Om>!Rbfv)^srLsX;miu!2A&jkToL!Rs!f2^) zV`XJ!g{(j#xggmi*#q&InTd}4vi6)bNKU&lzw0&(Zxl&qbcwb+C`Kr11gWB;qM+ac z&YnGc;b$sC9l6S!lsLJ(R4$K8Ywvw|yRoq`fg2Mv?X1-PG&+VWN=P_!hLA{WFA)Up z-CNL++mIGl$}&gE%8Kg;DIr1RDUDvIx#Z|*W^9oclFpo|sw$}PF7QUV;7w4l@qU=E zOp}+gAPBmY71y3SF2c~%+~@{D(Na&5*G42&fmD^4`0cme-o!ur@UH48%ZV#BH^&YO zNh!+w+g>7GdUVio_2-?=M?v4Ed-6cqU{+mS9bAPez<^Wdb#6ln)Kkim=QO0Tki_cm z9q02seZC9Qtef^a4DkUXq~PGqkVG89**wq-1;M;he%-$PZoZslekm)htjm*!pmpjC zPJaPFY+v5V0|z#knVDjVu%^;cmL>ElDK)bsG&Hm%Gc`3T;MFrL6zkWy{dm<_?f-uIOAd&a%yW9j zeE~WO5Mvs&14F|CL1xX!gC9{IjMTh}$_B_9QO}jQB$hb~)*hWmFtghL9Y10?=23LqL^!oVebK+&>XNjTM4IpR|k9QKC zO9(PELlTY^{J`_jbC94@q{DDwySsMnSNk;|{O*uupfHwl%*KY%2-RwB&s$%#{q z9z)N28{mcnNl9xj%OCLa#4|hXW}!YIDNGmCBwP|7OB)Y>gk!3SS@qCwxn;`s{{Hrc zvfMWhy?AU67a(^Rruq`m*27KmAxjc@fdtpD+S0_d>Zjk{Gfv z7@vsUT+zBAiH{^NFG8Az5pSx8eu4}oc=3?~1zEbfiPx$>BZqD|umQK*`(SU%cw$^4< zB&luG1eL?x=E*8w0-)k64sbV)YIR{R2NywHhTe3lWRA2x7RO;*NABBWu!?BQ(5e{--VzUdP zqb@mPDV@`bB}jcP{w!9l*D4l=4=)^B> z=jrE=8q!(AaCw0*$VTBpqmwp1u}+4n$kbR7>d@Bd|D+u7sV6 zMR{=_Z%&0>$qWr`&G2zjJIs}|IS}-l0c~=2Lz1NfXE?hApxBGW#TPH$zmK=EVONrV zALXPwS&|NF5PcprLc*SF>8{S^vDOG|NaOAEa4KVN=cjt5#Px&08mU+MMrQyQ=3Y6SI(0L*$B%y0+FY60qk0@i^X$Z7-0X#xGD1;+?hm?t{L zX94et1juRv(QyOOas|a`1LA%K?1cpDmj?Hl3Gxc;WmIJU8OY@ctkuN*@tPrdbM*5@-`K1x1DlwxUI+aNXwOs)9q#F09826_c zycJWr6;c1sH2>N!^`;8@t_}auF{Wk#_^Jxi&>6Zon)hqwlG5`Ct|L{fs z%`E@AAOGz>{jw7O{n!7~Gyl~w|Ku*zb^!mxF#q_1|N5f;&nN%(UH|*K|NO!Jwi*A% zC;$D(|IIP~{^9&=v;X*e|M{5Qg#!QjsQtm|6m#U;&(60{{1R@Qnccl>q*>6a06(ky8QkEqed-PygRR z|K&^n?p^<#0f_|I|+Zr2>^!0ssH(|M7VL zssZh+A%#Q){DZ~QA6EbGWdG=8|I9!C++F|53B8ULt9J|k!3zJ=5631<>ne2TBWe6l zm)?8;<9-1D>3E=O2*PU%|L~9h@|onL9LSyGr){+(6oE!JG9Q)8F@Ua}}z9NrS1JZUOx;--g;25Y$6zk74F_Lppn_$PJe$1|kv`tm@^ zwZ62cub!&;&#u+g;`4%@@!gQ%oVF?OndsQ-RF3nr$HB$JZNOqa000UiNklKbLAk;vB28yhrfC`mO5Mm)j1R*>nvL-y1 zpsZ3WP=P8th{}$#6i`vXrPh6Uc6)vA`_8=yt4sUdKf^uup7Y&b{@-`by}1_^mH&@4 zUK+tFS9@n+ zRb+!nnwzJm=N^##G~qFHyLG2DL{=@_p!Pz}>V=S2sjnMqI%|(TO>uK`b^E;iLZq;0 zx!&_RddrK1kr&!O2f4bsQY_Cse$JYj_v}Z#*MT~3jEU0r-Uf(lNYI1wEkef-GruVyRyI1EI?ctHVz;^XV; z5-M`m8BQy8PDMfhlH!jZJxWl>0v(wqs)xcrSUhq9LMw8na)ZnO%+b*%HPvK78Y5~N+i_r)c8oQNMA8Wp;5a&(ak*iB zLAC%ouPBoKc>uWwm3RaOa?Lm#li){>heR<$O;>gh9e@v|D@{W&qM^r+OiVZ&-~@UA z$P_?|#jOh<3wK*Wa^Zj)Ma6QtL^AOOY6pIHklhK1L@qCmX~YSvGYbH{PEDJ} z>*tRF$lbPtSVB;-%-lRn(r6VGWu;V9R8*rR%iLU6-0=W8lp&tQ&SU5Q#-c&z0A%V; zSwd1%p8!C`y?AmI!sxcQ7(jyJrYS>nK^v7It_ufG5J(^Y5)~XAr4;P{@>oi~faAgq z3?!v?H{H1N*f|Brh=6i!gM$1J#N~`N^<_!>8dHObC<-dIu}_-S*EGg~83WD~NP{l6 zIK6PLc`7gece~n6x%F924xf!H6*6geE|VEUiPt^z^GN;74F?umD2gIw2%j;ka0A zP>3MH98f2RbnOR5h6$?^T3yVZ7Q6w)fFEfxM$VMqVGIKC9gF)bgQHHQY zK3WX=M_Ssi4 zE#D^{E%Dr|fBm)Rr%)$~q(mt(Y^40uGjh2aIbbL~g1>g^c?)t2M@x|0-Jv0g^tX|F z)r7`Y(3Fg7fMiLA{yjnxLqmy1Q*TC0)g^qN41Y{+P$Ck=#Kee-rI#*`+`ZQ`SWRiL z=ic3s%fDsyNpMso>B;5cl%+?A6_ZcR+e)yM?vIu(*TB82MuIeQhNPyrrlv+h@u3Hf z zB?CSt2W7h7B!^&RJv{|#ZEgMici7EWeHa)PY2 zEkFMZ==ycob(ya7}N(jn%e4c+FTG*T^mUu>DHLu!^T@}<8k{zaW`&I?ReWI zsysBuNNpYtf`EkCg=AR!@3h?b@JUz$XEchx&?zxP$x@y=}POo#sT4CH8OtIyA&NGyGWL@}C5w@wGp)kB~> zZ|^*8ubyq|%s@;;xH!>3pY=QrA*pDnuF&LzsgR72&g*CMz6Tx3%R7W^-r4J&5rFX# z^EH<-G@j2PBn_sTu0bMHD#C|={-G2BL3cPmafz-T^MxRaq@ilG1YC(bTl>Ai3XS>x zkavG;r&tV_m8vYldEtLE2#UopGSE!ye0nAss(gqDYMy-NX=kG5l9degm)My?LD5xp zmwo)~RDT><0A%vx`cFOkc$qG^d8wW0uHY(WeDr2pKdfo87~_)re|+;Jh8kr4Dt4xu zg1g;ofB!MR|G%4Y{P>LHGp;OpX(!3a0r3GIr3p=T! zWMD0vlE$hMK%i002ov JPDHLkV1kri#0mfa literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/face-with-thermometer.png b/src/modules/cs/static/emoji/face-with-thermometer.png new file mode 100644 index 0000000000000000000000000000000000000000..407c0c86298f222dda335ae47571760ed788d867 GIT binary patch literal 3560 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 OME{17Zn?HY)c*h$zg?vO literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/fearful-face.png b/src/modules/cs/static/emoji/fearful-face.png new file mode 100644 index 0000000000000000000000000000000000000000..f1927f0ffb0ba32a0bcb00fb7f2305b604a8a88b GIT binary patch literal 3366 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/flushed-face.png b/src/modules/cs/static/emoji/flushed-face.png new file mode 100644 index 0000000000000000000000000000000000000000..adad5ca268aaeda4a967192f2c764a161c0eaa7c GIT binary patch literal 3611 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/freezing-face.png b/src/modules/cs/static/emoji/freezing-face.png new file mode 100644 index 0000000000000000000000000000000000000000..97a93106a6969a1f345213a1150a54a7d5f25629 GIT binary patch literal 3615 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/frowning-face-with-open-mouth.png b/src/modules/cs/static/emoji/frowning-face-with-open-mouth.png new file mode 100644 index 0000000000000000000000000000000000000000..1ec8ac6f5064a382f5a439d86e22a5bb20d18b45 GIT binary patch literal 3666 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/grimacing-face.png b/src/modules/cs/static/emoji/grimacing-face.png new file mode 100644 index 0000000000000000000000000000000000000000..82ca911e9f63a2d18c8904bb16b44bab75f909e9 GIT binary patch literal 3695 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/grinning-face-with-smiling-eyes.png b/src/modules/cs/static/emoji/grinning-face-with-smiling-eyes.png new file mode 100644 index 0000000000000000000000000000000000000000..d90a62d4e9e8c343c85df443a511e74ffecd31ff GIT binary patch literal 3677 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/grinning-face-with-star-eyes.png b/src/modules/cs/static/emoji/grinning-face-with-star-eyes.png new file mode 100644 index 0000000000000000000000000000000000000000..9d3a0983168d1b61b2b401bf30252818017d205d GIT binary patch literal 3888 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/hugging-face.png b/src/modules/cs/static/emoji/hugging-face.png new file mode 100644 index 0000000000000000000000000000000000000000..edc9e01e49376d7f9f84ed3acad88b773a61bce7 GIT binary patch literal 4061 zcmV<34Px&08mU+MMrQ<&u5?REk4eF5}>1M-3g(P{$snHBSx z7uRh9`)GYtnLjR#1|Hw1{q!I0o z3IFq!|M7VL#Vgfp0srxW|Mj8&wkH3xB7sBz|Ct;4kq!U0F897V|M$87rxySF)c@~i z-h2i5&_4h3k^lVS@Tn>P(JTMoO6Z9P|MjQjg$n<_DgX1G;(!MK`pM;Z0`P|g|HC!^ z+AsXnLI3lN|MG|b`NaN}5&!K|>V5+L*-zMX0{_o0?xQIG<~WG5?eS{?teR-!`RX19Uh5|GFyp$v*%1zW@5q|NPtkyf6RFKmXS) z|GhPaN&)}dS^ugV?S%&Rzbo^)B>(7V|M7jwVF3TjEdR17|D^-}WcrtJ-CP*|Lbx8>U;mzWdEB3o?!(4yAtuM6YioF z|K4Z+%}D>A0sqP<#+M`Su^f+C24^h*|KorEyCwg|MfRN<^~68jg9QJ~P}Hw2=a>cV z&N$4XDZh^(|NiXfix%dlR{OKUiN>9UaFS( zu<3He>hMg_#pmkY(7Lq8xVhT;_xbUk-Pqyg@$vcN&eFEV)V=Dx(v{Kar~m*AD@jB_ zR9M5TmkCglXBvQkfP|x5f;j7{MZ8wG9d&lcX?JF4I+G9r6p&j^MN0w+sjeo-B?uyl z90`O#I24g^cK|6OiV;xmQx1_!&JhZ%f=B_6cHi&+|3qlDw!8bx0Dr#k&BObC-}`@m zqM-2qf&9@3Rk9jDR{dZ8i>l%}LxTf*_wL!VXYT>M%|!3%P$bi%S}!30{8uU3{=Q}Kx4Rje~I8x)AEZHr-u;-alf2pVRI~R;>w7a}S5Z}aeYIDdq{cnQP*UtKPx-4TV1Lfc2QPw*+ z0Gf(U-)~Q^;o9kI8U8E0rudTNU4|d6Pt|BwatUbKu6_$#LtqjF8p&{|8;lnW9Mn_% zNYiZovRe>8Sm%2d$jFena7#u88a?Zu(V*pza90MF2v|S^ zgD~aWR)osa`EWjJdS46(9`$|g;Nalvi^=fB@KkV#uP@97iNUnf=)?ISPo4kQ0OrDF zIC6;tnhvT&rVr#qHu$0&Jl_ER{{C_qvLS~r;ea`ac!^BA-oH+U2a`)4)xCWAGAJl$ zwTwXSfaxHJsuT=JcfYTpbaukNceS;hI(6y@fZLGHA6F0#GzTGVZEgL6ZmH(4`$=)P zbU-jU3IP%4&!2y1!*pvLr_P^`fUwa~NlEfP!(WLET9ZqX>PAO9JC7ef9uYy}qoZAn0qHK4UwD*uN(bV*>gwt`J3GGva2w_$h#(Yb7J?wOZc>oks=c;2 zvL+-n-qqK)vhol(iXeP)&}(c0MXao>^mTQm4@h^c?SXYWq}`0lE->{Kt`t5zgmV<< zJL2zAV)$WUVc|+2c)CzZYt~y8CDdM7S(%d)8ygF8ClH5D99jb!fMJ*}jE&97$*HWI zWOYk-tXm~g*(vSL9iPNZNEI6w7Z>Z~6nA0`*bq#_VjAR`Y%h_vYOgxyHt&;mv&Q@T zC!+u-jJesR#wowJRa~5&U7DXFrcvNz|M)mHx%F@Fx>D`0t;rNgJD9}cs3<=_Cnvw3 zhet+6(q{cG0&y2H@S%=yCqF;Gs3@_xzrTOH9ctUB`)==dv?g;T?d{jEUl)sATp)0I z_%<^;yEbj+)VLgo9Ww&jwICABy`Gbt&_+Re(!s$zy|*8}fL z6*1>iZs1JGhnA`;tgI~Dtdo*5Gg(L!qAuHfEhJ)irKhI{1}1r06h(J1n|nkeQBRL> zcsd1V79SFcgu?|^7M`Agfe`LhOy`t`hHQ|vwLXYAbUMS!%L}H``@D!88!eV9=@VAlgVYx9&|4x#v>p-~!oL;8-fNc?!yzUJ_c2M32U^C*tSnFk*g~W16+*M?1I5Tcvn~>h_Vn++Ve0TZ*Z`vv6ciC;juJV z3KD54uM$~;35P~y3=Me(dKU3`S=}*BYi+5TWnqZkWd?+NSHJ+Ic#ELj!g98bTC<2Ma!|WDuVv;DyGs(!Gi^wn@57^ zQlf18b@W5>Kop5QcMcymaWpL@L>5-7frQe%=+shEZ*x2lL_YUIFWn>>QdLO{iEM4r z^qo5}4th3=4fv;LRmCldmqLBOM2H-{k;p3( z?qG}6xFsdMN0b+$uda%V)HGqWZ7>nS#Qe1>t)`fj4k91(t^C=fhI z^py@FBCG4;wb^2e?fClb;sl7`RIx@X0&8m!v5IRZ7T@9~d1YIai8Ugx(@G>#0?Zvv z?MB`v)L;kQT0m5j@OGrR6dJ_@d9cckX{)KqzAF6XCQ1pdH|nF(S7{@5uefmbVqgyJ zDv8DYQ8-eo0jmVPVv8q~sIM#grh;n2Q;Lj4on2bfk#;*x^iG8%0Znt-?b_LzQd4Nj z$VhyJZf;YSAO6q)$3pTve4v*iy+O2aw~<+UJ59Lwiqyc$lmMBm2GXlVA!wK#WecUy zO8iLFjF8Z=4XeI7>3;g`WH2f&k`MRp#Kb~bhmgsHwBW#B{1I48wZmRAg|qC1i3wLz zK8VU~Jo19;NqVau{_8bjNCC)H&gb(jdwYLaV9)djh1ty7wA%z}wajdxaOq8ZjeT#g z844@6_sBaFbC;{P>XEIeslnye2cNMwGvix6&d-1FY(etVY>yov%ti=>=(=ZUPEs=w z@T9uel5c5d?{TKSp^R%R`$Vd&1h<@`w#MDN37E;Ox253e$%z_P``paD$WF$vsBvbl zomIT>?BtX03(RoO-7@YbIE%&Oa9##=Y)kQLMnG|I|I!2up1x>E2o(Tqy3VM$tk zdNb6~2y_fh&9z8aHFWw!@MRF$BM+&J%{5krteX|RX= zS4A;o)424*XoR#FGW#1@`u50v@wlxPLZMP&W9iUL!yoQrSw1 zN%(BDqRO@eHk%u8GT`x*>K~5fQ8=93TYQ#5B;{ z`pWn0r(YRwkKvM_8f@p+uUAo3H26D-%O%x=sJcp=6IK16R?OcuaVQSpMq?7A@ki|i zjp^x?E^@XGwq35jxa;KLCKNh2WX+O5Xb3oEYp_dj=#r&FLBUCf4go^~I^13-w+Jr% z6Xd=x+TL-!yXihl62kkue82fV-s3wnq*lFgiGcAN$>n9E-ZGZX?cLS2KD0zPqS&3i zttg3S86w1T++iw>g}3(Xktq5O5#o4 zcK1L!0a4sEOJ_So=Cl2MXgOn9zkNipNkLIHegaQT)mZfJ@9wVP%q9GLH&dd!l8HWhA4JD4nL-7Q7xr7KkiSDHM+4@fYo&2K5ty z2y;cqeWzoCNCxg9+pb|qS_`S(Xth*L5|+Jd@Z3OXzGy4(1uuMser+mHVdku4Dj-50 zS=YyX*AT#@s;xIv6+DKp>_6Qgk8b(mNQsobu*7SdgCYnV!#}0=J0m|Zjh)_nDF{Ya zg6Khp&{VD629My)ohNh{l`Vgy9HlZ*ESk=&x9lChczAZ2Vp654XbpjJZ|*Jz1ElIo zlGH}PwF{cTz;OGHtjowVjpx`zWjDFKT}G8ernDCm5syq)J`!2|=j91=ag1DGTO%M# z^*mqpz+_pzzp^1eng&yJyB~v+#F^8>QrwIL=?0SpAH>mQjG%Mp$JN1xYW4UcpTVhc z&2y)t)oNucCj2nzU^`i1*XLt06vK(^hr&KLi6~H%;PDbM*z-sVBQi9T2UsSXj>4`( zl4O?mCNl-C15M(~Y&MO@$lp3o*+h+-%${dUE`gyTWUzsf6cF63q3ERKO(x!g&(Y!7 z-?~_IxmM&W!D1%=2zo)}clqR3Cj8HiO)N_MGR-^5WPx&08mU+MMrQ<$9f(6lK{h`1M-Og^@;)eo(0l!1;J+n#%lz| zYX-z<1H^I<@16$ih6VMT4)UKH#ApJ|a0bqF59Wdf=#n1AX9C4%0mx|q>6ZxEdkpG_ z3g&(S?1cmHln?Zz8~m*f%4z}TeF5#59q4}n?SlgLlm+;q4f?4O%xnU`W&*!v0`iCg z|J^MAw;J}Q4FAtG|I#!6?m+RP3;wYX|Gq8%yea>>A^+Df|HU)tmInE$4*sqW{HzcE z!#4lmEaHy@|JgC{p9$@p2><)E|MzGA#w-4_6930F|I91@`l|o&N!ED)|F|Fj_Fn(h zGXK#q`>zc7s|xFx2HAfB;)??R!YTjbE&uwX|Fjj}hyvY(0nKdy|NE~0^;Y+&2m|NFZC<}v@oB>()x|IRS~{J;PF$p80l|E&VlbpZag761C4p<)65%rpP_ zjQ{w0|I{o0{nn>v0srSu|MgFKK?2cn0RPP;^_>F#;6VS;C;#9{|M`^v@Lp*z0RR2d z|JW^@UIL|M0d6z^|L{=begOaHMF0KU|LjQr?pOc41(;X?|HcP_L;?Tvi2uwr|LZ;f z{_OwgdjH~T{qKG6j{yJpbpNOV|C|B;(GCCckpI?F@R|bu*g*fd1FnD*|FZ`Fz!Hg0 z0{_|)?V=IsjRF7BJO85rhe`n0d;s;lEv9S=yoV9ct1JKge65Es|L8ma&_$hR2HLnx z+BBndy&&z%Os#qx_R&-H;bZ^a9sj!%l4%~Eav#c$ zALqX*@8)#6n@PTqDjl2F-2eapJ#xYy^_I}N|j8=+U5CVfW@<$sP@?8Y4+06*yPT!ve&f1)w2Kq3NcAUK~z}7yw?X* zQ&}DcaC(9?MP$Zu6x+@mopEhb_M9ca&>5*B5J(IH!8JjFPy`|@(lj8Uj367Ni4+S6 zK?WE>kRqM|X+u-##a`yvb=v;#eJ?;joSEG{`w2qcz5LF1@5_5UWMuv~QI`3?g+f(^ zP$~b7K~B(ot3UvzFreO|bY@-;G-@04C`+Wf_)^@@LnQC+3obfNU}wO4zKPF;#9D=Vv-D2jP??Q-dbChb+* zWd9(eu%2LErByk{oj>#~&Y|-M?aPXF2})Jrk2oruwBvyZsj@$Ph|IU-*VNduY@p(` zH_89*SyWd2ytN6Kve?t9sj2570N~H_si~(=r^Y=1s;TvJ)!*55#f{q44W;)gq7S0L z4`S;)Qh+&lFfJ}Gwl=b~p;~+6dpBKnojNQlJ25sk_UO^LIBcP2fPezb(W5Z5zw*w7 zc=eSF=64pX1m{lWetY}q=xD-{K{A@81uW1d~L9z@W2p ztpSqe>7=+$y}IFK#Xlg@45o_(7&LG7`hj4<^903g4#IB^&16EmS_M ziN8=cGJ>v$M^7I2KYmn;FoHt_GHM?^?(cu{s0t>Hj8tBYulc+Doac` z3wgZ4+_zQVNGGD|4Kb1{dfSiY2FBMcQJr?8+-P=u&}v%{FmcDK#pI!bm{nv zz{?--@n6Qfn+pq{KN}kxim%byFso2&Q}5V|=eeTpC&!NMn%ISM4CfLG@$E5mbYcPq zo^*?HpTDT9qqRy`TDwHOI=HT`i}&X7#3>@XccFjMfg`)g;fcp@cwKdMR@G9wq`Xe6 zCf=&I?%C+e;dWVc^v}T+*mCxi__YL*- zW~XImXTzklGbjhTr1OQsHMKX9wwYiD}tPHfgkyo zRL-1naeODRP;a zBE+CUm;+9IOG}`+v8B4!2a+pJ@e{2KOOr73WPJlxWMpJ!NJvjld3pI)IKbimiIkW3 z^n^fQ8mmP=IV#NL#|*8h^CrLby9~=vPlqT0A)%Q)Js3k=IU%6*Ofiux18^KXLx0Tp zZtDuk$=?nTGuYFCE^uX;*}|;h!H2^!hyOVzheL)R4i3WvTNcYzKzHyAHU((A5(#ZM z3lI$;2Rc0^B`+`UX&;Yw|DknE%weg-9JYRVpC{^g3R6;2=yYFzm;jw!MnX#ga!W7` zKH}?3&jM+)q-ZAK#M*-y;AexB*0-Hsue_exKoOM1x9%)G6X8 zC6$Q#?pZ^o6v(|kaY<4VU(_jaf@zN404-LSg06V)nT8t0`2{DEn)~kQ>RQi;uI|0Q z=A;t^`Qji}KS!isKybystKK|#s=pHskb@B}CUK{rTr5U;T-(=q)Pru9}As0ajOax&mGBY#8 zEG5Ov%yvf1(2D%LXADc9HOG=0)fC4!eW_8 zC^HlmTo(|5fF2cvpir8loA(tTuA1Z$k^i`j?e32&8Ha_LM@2;;$-uyXg<>W_SS&aM zi~uH>hlQa+urhackf!9KRb0*BSlD?(B~z0yEF~uk^j#4}SP}&seSshu9DqzrX^v=R zY{uUtPbJhf!XTfW{%%ky6eM$Vr@-W7eF&zel9L0mV;l-Q z;Dpj(r=gHdP`bv9=ZLkM45(!2y~l`#8}9sG8J zf$d=ujv*`z3e9@7RnioCuwk3@zTRWxxDS^?3;ao30?vsON&NfmXi>mG9Y-U}ufegf z2@+~5&3^Rz(2EgdV}T?~BUB0~Sm51e3DkiUU{D>2!)17B%y~kV-;xg7L`asX6cbtP z9puvZ1E>s>iR$bi9Zp;NmKAfJYVqPYNb(122T;f|8O-PB!*2q9XFFC1hIBlHnu{ou zIgs))wP^_`!U`U2?90#Z7-%OHV4#ja;V`_GD#%dgR#IH<#lVuCp&nY_4s03EKy{a zX{NUYao9cU0rE4x-K$*wE5>-LzBQK(q1-9%AiNJSA0S49a`e=~? z6`cA0)Px&08mU+MMrQt2z2w_VXS6K^aRt9Bq zEP<#{ykQJ;V-9pp4p~AKN@^&0c{GP-7WC3Dr0y=OjeymBtU9*!+p)mkKk`}6Q0*A_j+18Wgh5?{>GKP#qn`tC?;iK-~pzO0;##jYt zbT5OuW6rsB$jN-#fjf|S0g;Ucrkpdekq@bmMxMoe)ngcMTM=VL2uz9sqm>}5sZ70^ zEUuqcuC!vtqc*siP^Xw6ue55#t7W=X3|!fm=eBXhg8-bTMZBC-tBOCE#dg@1O{KeQ z%*}}0XBl;s8m+Nxz_(t^av*?S6KckT)WK}j!G6y`2}XuMl)!S)h%1`ho$1Yi-Z~3H zs86}BVZW0urGBLJ&-yOss4id!s$Npj)nlGLvo! zfs!<(bPk8jjN7()#hN~>o=2~77lMs8o|QbOX%2jM8;QMl%(HmFqfxV`SGSrrt&cvS zizAOVLUdA2Qvm%`BpLlT0SFW7IddY|{^i#*^<>~H{l~O^ z;XM~|rjz}*=`Hm5>|0FC#-L%N`|#p&%S1c?01MPfL_t(o z!=%<}P*dj`2k;E5zW&Pa|7FOZeY0tQnzbcq$#!aZ zQo%pJ`|Ohx*-%f{0Ai+VWcaB8AaAw$S6OOU__DPCc}desdg{!-+xZb?sAsTxW9UrRFEk}js&gQS)za-H0D%y(_Dvf8+q zMx)Rujv)G*Fcsuw^zNP4Dn zv6mNB8XlIE3RH)pr4j?DyzF6DLqo%_D1YF~UtzT&??gL;M#Ctr9UU@YlER=KFMzgK zkPLqSh)T7WhNp#aY@(~H>)g3>C(oUn4*UV@U0r$a)f5cF>eX4PC1Gjd zQUWs7C6O!%2qd7ArdUvq=6zls49soFac$5VY}}KV7g~;Ctf?f#NbT*Z(3<=QxR^`u z(FRBg6dN4M&Ca~en3$L=kbt>C8(@29-a|DFW3hn&2Zvy+;jY>&$01||f>R&b+1cso zc?;t5anMFj&(6+?n-q|w_(=faosmd8z7=}`0-N+%EUbL*#Dur^lP6Cybs)~O(Q$_L z%*;&Kw?eEMQetitqSPU0ri;&ATXMLV0Cn(CCHaSV*5FW$am?z6?hR393< z8QNS?p@@XQ!8~3rP_9vkg1ANj!^!2r9&~`BqM|tznqx1soc9yV8VP%yUfJjj^MqG`%xEiL^ogc6A)n#)BMaqzzIk;a=N zAa1loA{6%bx3tL0$}#qrdLMO0Dd-s*>hJFt3X`WH7;Io1r{jc6OeU8rkt7R+IsN@Z ztvwnVYw3bHkij|@t){(oXdoviCq$SW9Zfh;jH9E&bO?$EP&7CpIXMFZL#=8C#$GdL zw2cE<6iw^+z|Thy;}8_Yi*g0V!NJ`fpJ^~lz;ePRfI>nJAN~3C__(^9#x~KNeR)A= zjJEOJr?n6`(3Arp*CBWAzydIk>E#%a`r{ zTwx_qZ|v763c)^{Z6f~fGiyNZKK`V=$;(CMQ<+w; zEbjCM`Q(l81@w21PT}!=Nf+?D`;Nr-W{V1)9ZGY#5-90^KsiE5WUk8HB}D|z_-V34t=h)KRNQo9&Fnjr@7?pzv%2h?Uq$Fg-2R-O>gA@7P zul)Rp_`P(qv9Y1Qyd0mAkm3@d0;qpr_u#zCPu>{#yuPw+s6`Q+AFC4}N`L`+@CKbH zkiWL|L-F@})y;vqCp96ELB=_^}w$!u{4-2qQu`Cd!(xu0?J_N_*MdeoYdq_lajBhqTrTftbfvhm54 zlaKG8PrQaCbRXg+0pyRSitkkz>-z~OuRLi~HjnkFYm%M^&U-Fw z!TUr>Nb>2Nva|a(^^!aZw zcI=`y6vqKPT*Vlz+>*m0EyO`ktNj-`+~yx3*bE6;I4~{}ve<5cP(g@oPS3m2p+5&9$j9IB_ny!E!iqaP$H7Vt%jv~; z>L8)^o7Vh2Dtq_(Z=hcZVYKSEs|}Bomnno#rk7(NPp3) znAUNJC3NrV6w3<|O-Xa?xvD_&P~C?qjuP@iAzQKdZE<+uz12sGI=7EM4roFTch;xM z*d9yCA%hj#)H@)j>(l*{L?Uql>`8IvJ)=&=cA7{?u0QWSS|4C}Ud$pTL#tYl+!6XT z!(Y;DxWK9{hL`92#qw#7I;Pd^3`&jqQTKj`Tbp!wE|Sm6jgrqKRHsa*KhDPe@gZL0 zzW3>C`Ad7GkOoyXI7nzOjPGC1h!YAZZYimXt6Y5cdOeRt4{>_w(tL;szUe!OR1#TLH$S{$pW@?U z0hZio)JQad$eCz585VBI{H~9S#8b&aIu~V;lky@I?)JPx&08mU+MMrQ<%W(_Mdl~za0LOY8$bKfsf*<;;4eX#D z!+kgBj~c^k2*hUt!)OD~d>O}S0n2Lu$!!G8Y5>q~0Mc>=$Z-z)ngPXV0`i9d?1%>P zn-<1s0>p9;%Wekdf(G7z5BZb={GtZNX93D;0m*3r(s2Xqg#_w{3iPBK{H+ebX$H!6 z7UX;Z;d=tRW`l%7}q6_t<4g9ST z|G6UmuMz*hCjZ_o|JgC^oC*J%8vmmm_^1u|p$q@19RJ2P|Gq5$?nD32HUG;h|I0Sr zhXVYg5C5tn|M`MzG64VjqT-AL|NFN8#VY^%tpB?v|LQsa(lU2F0so*I|Li{h@k;;J zF#q|E|Mpk^v>E7^2KAi<|M-aiw;uof)&J%*|DGDze*y5H3fqAJ|N5Q(v>^Yk9{>87 zenA2M_g~g|0ROTQ@t6hekOKe1HvhW>|Ni3t`@H_475b_U&Tatz`?3F}9{s5i|NYAU z_j}ZK0ROKf=#&Qk+bxn)0ppMa;)eqN@O1yuEdSy_|NO@P`ltW7D*yb?hDrke{K5a} zH2=&n|KctGtpaj30RNa7|CJc)nFrBw0E|)u{-qY;egNf?2`U^B0RP)w{?SAK+Ef3{5dXju|J4t}l_1r9 z1OM%V|IU00vIO^3p|Knc&-4?;5Iox>y{+R*)=zaK~Bh2l$s8@U0{A$RWCl8vO8J+Q?CMzbp5( z9rmyq#)1s+(L7=&0H#ePg8%>kI&@M_Qvg>j{umJo0Rjb5W2~Mc`2Lge_@qt!#`UC_ zyEcNAV(;|a)aA6ZfT;FzKI_NZ&coKex9-?%iNoYgB~Z}-019bN8zq0DuE6fxT#kbD5?cBLD zJw2VYNF+TST6gZR?kJ8>n5}s2qH8Ul2h+-1vmcU{rm{8@3NUGDs3-QDrijdW3tqG4 zS_{Bws#@#j<`x+lNmvNOrC=m7(ha5U7D&qtU+diJTDx{_Y;5cVMg$>9iH(K+wbi_c z%nQq3QIqy!P2{Xuvu5AEeOPlNG#ob-0-Z3Vx(v3OgoSYmO*5y=GX|`l$ zJ%%4#$c$)gYbz=7*|1?lj!MHEL68RYZty87X=|%Gomu&z#*Blw7$(-x)rBOuTs}xc z9QM8>2*QArAg8OVF+!oaVA^e=%$;AEX<2pV%$bywUAuPu6~#y85y5Y92MkC_fuU8- znU(X^r|sBsD5$-vstQRdDc?X+NDqnR3jlCW3UXj@{cVNjpQoGwiaHeJUSEF}hMn#2 zZmBIVFR$(FN5xIbxBd4YppKUN{diz~{Xk}A-I5svHPnN2_C-q#pNrDNm31Hcc2bVL zD2GNAzU_XA1Kwcdt? z0a~Tz`x9qLeZHoGW(~EnYM=#m*guM;m(d_y?J|4V5ozG!+Q8LIq>e3g9G%TA-#L^ddnQ0!&#zvb#JQXru|-> zKqM|L0I963qa%0k-gWD~fB=Sm#`A>(Y6T^?qoV_cIyx7WvZ#W(_f-3)v8ql$^AN@*X`5jm{JqP>{W98+Qn&ISB0K+XUX-YxZSL6Yp%=#)!5baI#@UXN*nv+v% z>TbqCKgfY8K`ptA-KnWgPRM|G(0r)?Eg~yggdtxV4ro9*UA;3jIC$qOIGMJ1q`blCvj4x<&vA*m5Rv4jrOlug1B)QxcLFD$U&5BPB6WFq&RFFSN1mtBkDP9DWz=`u$K~Zd5L0EE#>Ojsl zLEFZTio(z#Yfl8R00PNNg+Sm)yR#7FNsPnL_ON6V)iYA#-9WTR2*Pt`#&Pn+EO&f^ ziSSZ+JbX{xS>k+7922jlwSQC;UTb63rPY4xJTW(af1qFuISCkY#GmGQWeoKVjXrcI zuDduN)(Gt`IWt(IWva)Bt`RH}o|`9x$>eYnD#WZ(6~wz&%a=-SJ{o>lT3X5yS0r#a zu)&ZhVrKEP#~<42CTMO*LdOi8{0denKqx#%$FhHRcb4;egd?M@QWmQMN`l!jnNdVR zfmyRA9{$Thk`tjMwj53+cA%gFXJyZE?+1VkdqVRaQy)z1i(C@ORGAX}LTv%9Gfh0RSfNTR$K?ak)N6a-2aY&(&wQ?!{=F!KGuUxsv$qyQL zHn|4I8_$~d)lX-3Oh};XCQ?EaRD~>*@Oz!*LdnQ^D9KLg9jbEnxU|L?%=m_^u?!}+ z2T3-#C~sd>mRKxi6|xGCJKt-86Zys!xt%hUPLH-BoGn@JyglRpjB4t}@hZuNN=H>8 z3;hEHp?MUv@VP=)xrhqos2rFWg)#%uTVxWy1xW!GxF}3{dk6XA1Szv+d||do1kAV- zW2B=-nOTzd>=;r)(~G8p1(*_PN}DIGNkZ8pJ9acGCjdEtAu$G8YLuA}MNP|KPRJG) z*X$$X6td`a>ZXxr629#Du#yecfO8}f>IsQiX!sHbnsn6+4d%on2^R%R6^$`!^gn9K zpZARhi)<|{a5>qoq@(*12uoVC(K@;8Nkm0q3t`c}{2Zhdv2ATF%z=S&z=@w@@CpY@ z6zxUESs{T?6i|d^fjA3{mj{WJ$~vnxrrTp$r;GzKgt0T{pmJF$ZO49n`mFtE2UW?KF#eC@B^J>{ z$f8zS4_mZWbsn!WH*rbuW?f)D+F}p^=M=*^M1Bdb{*nQB j?FfVveqzC41Y3CnCdE8-ty6IC00000NkvXXu0mjfJ2ePc literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/kissing-face-with-smiling-eyes.png b/src/modules/cs/static/emoji/kissing-face-with-smiling-eyes.png new file mode 100644 index 0000000000000000000000000000000000000000..f041364589496091ec783188c0c5f3d41f9f2fa6 GIT binary patch literal 3633 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 XcV4Z=Ih6l9)sTsSnf@m|o6!FORYkBe literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/money-mouth-face.png b/src/modules/cs/static/emoji/money-mouth-face.png new file mode 100644 index 0000000000000000000000000000000000000000..e72b3b88e4bec38b36a416c76b6260eec606a59c GIT binary patch literal 3680 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 Cg8mo) literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/nerd-face.png b/src/modules/cs/static/emoji/nerd-face.png new file mode 100644 index 0000000000000000000000000000000000000000..c9158633dd922344da30e1aa8a3ccb25a5f216d9 GIT binary patch literal 3673 zcmV-f4yN&mP)Px&08mU+MMrQ<`;-93ZV2Ok0LXqOPMe0LW+n?T7}h`1ITd>^oarTi2(el3-X;A$7li4 zas|Y456f-_=YImkX#>%72I__f%WDG2X#vA#0_ukf`kDjgeF5~B3-qKL?U)_FWdih# z1hQlS`l%82q7(6l0``;z_@NH?p$+w>3-Y52?VJb+2@L0z1=)WA{3u@L{wH4hID|I#V{`lSE38{&)u|M{5z`?BMZ1pn4C_^1v4 z&@}(MCI8wo|I;+hZ2Vqr-iQL#b^!aU51d^A|JyA8{^S4p zpa0Az`>qWC)++q15C6wC)_MT{_=x|>H>G6(|M`vo$twT-*8lv+|E>f6;yC~LlmFi` z7#SWSBPJRb5dXq5|Isi1-z;}M0sp-P^_~LJasbL%^tQ!CDWBA81|M+14@`nHM zlg_0c|KVx>=xNuwE&uRs_qQsoudY2lK4482*s3V+!!O{rEw89sI4u%DHyHozgzt|4 z|H2ag)ervbH*;SbigO-JJ{9-dGge6$R#i&o!!+iL0qT$e|HlUZ-4_4LHP@6E`_C~h zCKIxN7q^TTt(YnN{ru$Tu^utBzRdq8a9@DbkuH^Yrn2Y##m3MgQhX|Niau!6)IF7QvA! zsdo=-Q58)_Ex(Z!^X^=~q9@YT((lePoroml!zN}d0Fhh;%C0H@w+@DKFu#BofPY-) z;#pdcJK zz}8mnQ9GWVb{614SOUb*umlOXf<)O-ga8Q;M1-&efdC?_>;;q^MTb=psO*b?z*JF0 zL~s}PrM2zM{oYFmh{rB7XZ|O=oOkb+AOCyr_Z}}uN&RmmFZF*DiA0i@hr@rxkWrA` zq^9;4keb>iSp}ItMI$N6s(tgdD*4VyTzAN-Uw@+}tMo^EAtS5ycU53C78ahxRan?~ zGT@HtKh$Jp{t)I%RnLGkjfH20!$*U+y1FI?A01|&Eo?j!;HmoMrbWMZk_4vk>~OH2 z$KgHy!nNn{PP4>r|3Vxk`@0HMTDt*Z*uiE;4)2+(1G^H35o*Iy1=4TZ`6?zlppl)p z>j2VW5JI4LJm6tE+!zoY_P5_MC9{4*So9gk&}}ec>wm|~&dP!Cv@g%cWIxPB0=(k9E1Wjyhn}~r!@C8X$ z(LnlOpGfO*PLiqVUXTPfBM}3Vrlv_r1zV!Sq}P9FcCssDqEDtIB_(-znVQa_5h54> zG4=BDf@PuZ(J?D!KX8ry6&e&

`7Dx_|$EF9~h$w-N+mKz6@tVq!r-%1O_duN3BO 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 rP2t#b?e%cYPx<%oMA7^D+HBG7{M{ zBkLe5``8(W=cn&q@qJv^!gxcSNfTteF7s@kEXUv+u;hR-V#t zlDL}*h##(WP2TLi-VAebTstJT78h8b5B;BT`n9B!i-~*<&WQOC_(~+*!Ak_KfML7v z6s?SHmIzx8hgQO2eIBrHj_`T{UA+_hvnzcs@zh=(>p=;q63Fy@Yf4eaDbmuLxB+3iv)WapjHOZz60tdu%EJ3h?;(m7zu@S zxkKQn4g4h>zvBR%(%_&B99Do)ENGV6EoKCVwcsFcz#jrVn&nTTq zu%KNXbclj#UeF@?I|dGR@SvA%;@>f*ww;3jgA!oQ6)ZjiGuq&27%W`}%?!oPFhJ1< zGge^t7TClcra-IhIX^VbVZKFWr4Rcqisxg}DN5bBzw`LFfgE|BtPavprW4c7n8Iu?c5OrvbPDuU zNNtZ_0yQVWguudk>8)0+(R|TgpCmU2lzy75kp2~;!uN);DP*y~#ls;DqhmY`vX2a% zKmKZbZN6EUQJSC<>Qp15|upXw5;@--T_^PfTM3D!TdOskbARDigcKZ=fp=38?TL9+sGvl$02J5LV*XKqkKoM5Wuh3`HDXCf!PB z*-=whUmN+r@ZOZ}iswSQ61lOe;GVQW+rK>OU-`~0OM7rvk;z{^soA@0$o_Y2zFji3L6)>CSG_&~UafC+n^) zQ)NtRltROkzbX`6!*7d=ZAK-68JEbf+G86UW?B-clps#;9*V&6xawz*8_fcLCN3z| z2gu6^@Vh<7m?j^0mbBrzx_IR7Xq}jy-v3>vpTPbpp?gL@xbe@1rnlT3!h@n>qni}b z(dn=QbRaVPZD~SIG=DH(kB-eh>y9^oX>5lOua{9lU9r5Fbl>`y#+_Ka_g_=x6I6Z z*SfZvcaM)ZoAoE(I6U;z4+=e8BHP&MnFz@2Y;_cBSO&1mnAhl=T*NGpo#UDWP zl7U{G7vF~U{Js_|g}PcMCF^cO`DiKn?$nr}VQHX7RP^IERcdU{(=9ICy|41Y>bJc1 z812Pkh!;*ymvsgciBTcehFbM*6K^^T*jTD5g5jd`<{L~ob#=JT;l>x{iR{qZh7rT) zG6WR6VdOdG8E$^Sa-pHDnli~-9&xuUa_DV)Zr!>iU(;_jk%!HDr;Sk9^Tfp!&C2{o zP^3dHPMtBizL7I=Xxp#q*3OGd!@T9m&)7Q-OGh;hh93JgH3(`r|LliuU#{us<34M^ zotEm074Hx%6tw!q^Pe+sQN;CZ7x?d^%ddN}9Ge&~y_ z&VU~DIjR+2D3m16pXICrPgc{`7UnLYYp8gF5KlOhWPmxt>{XlgFnips`n;*P@Nh2) zFLHvDtBYwqJO1iDu#IVNloNOiKd;lKxv5uIJOIPqs-ZO4jM*!Oq<(3+s(S>xUOiDr31aSw+uEuxT*fc-?!G2h-E4 zWsH@RXu6sF@W|&!O1fe&YVBIOoI`}rV(#VRi*mDc-a33bW*lQ;0kL`0juGnm@j-D) ztcmJj+gs0nQt|%r9RW>y2~%n~1YWy0Q_G9sYUalplFup^!*rrxHFKlz5m^|`FD7S@ z+{$xG!iL6*RK}l(Bgj>Cee&87aZw7f55t_(L+TB9@<`dxTmtPcvR@6K&yR?o${iaW zGEOEJUkX3zh>1Z>BV(OQi3UG^&8m#NEVF-YCwWjDIyGJwd}#Cbb1;b)#60J>#4w}U zXI$Tu64UaH8b!5zb`4Ok`wI%rnb6F_%L@t$7JbDd8_8keMF|J!yz;YzY25I#F8rPM5F%8^585%?$6ro429mGyA}Xv zhM%m-f3wG<(e>K|dVD9XDgWg$uiu$i@ z?reO#>l6~{X$rH!nOtel{;UVD46|#pOj5t( zM(a#|_K) dhets$$i6{W8EJK!>)-PV;q;6zm+RPu{}1~OM2`Rf literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/overheated-face.png b/src/modules/cs/static/emoji/overheated-face.png new file mode 100644 index 0000000000000000000000000000000000000000..b8c9f63abc6adc642dd52fb2c0a54907dc33368d GIT binary patch literal 3440 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/persevering-face.png b/src/modules/cs/static/emoji/persevering-face.png new file mode 100644 index 0000000000000000000000000000000000000000..daa2cc031ea924d404ba9a3795d02ab3fda2d7e6 GIT binary patch literal 3628 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 yT7XHVwGdjDz#q#0000Z3n!t(*u&M!U8i5yO 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 A@&Et; literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/relieved-face.png b/src/modules/cs/static/emoji/relieved-face.png new file mode 100644 index 0000000000000000000000000000000000000000..502f517dacce540e92ffe0d8c380bc436a2fd833 GIT binary patch literal 3716 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

+ ); + }; + } + + // 不推荐用该方法,在 setup 中返回模板信息 + // render() { + // return
; + // } +}); diff --git a/src/modules/demo/views/crud/components/search/base.vue b/src/modules/demo/views/crud/components/search/base.vue new file mode 100644 index 0000000..4d8953e --- /dev/null +++ b/src/modules/demo/views/crud/components/search/base.vue @@ -0,0 +1,143 @@ + + + diff --git a/src/modules/demo/views/crud/components/search/custom.vue b/src/modules/demo/views/crud/components/search/custom.vue new file mode 100644 index 0000000..ece49b9 --- /dev/null +++ b/src/modules/demo/views/crud/components/search/custom.vue @@ -0,0 +1,176 @@ + + + diff --git a/src/modules/demo/views/crud/components/search/layout.vue b/src/modules/demo/views/crud/components/search/layout.vue new file mode 100644 index 0000000..db5ea97 --- /dev/null +++ b/src/modules/demo/views/crud/components/search/layout.vue @@ -0,0 +1,152 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/base.vue b/src/modules/demo/views/crud/components/table/base.vue new file mode 100644 index 0000000..f625e8c --- /dev/null +++ b/src/modules/demo/views/crud/components/table/base.vue @@ -0,0 +1,109 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/children.vue b/src/modules/demo/views/crud/components/table/children.vue new file mode 100644 index 0000000..fbf4cd7 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/children.vue @@ -0,0 +1,98 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/column-custom.vue b/src/modules/demo/views/crud/components/table/column-custom.vue new file mode 100644 index 0000000..f67a134 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/column-custom.vue @@ -0,0 +1,108 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/component/index.vue b/src/modules/demo/views/crud/components/table/component/index.vue new file mode 100644 index 0000000..3e1ee21 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/component/index.vue @@ -0,0 +1,108 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/component/user-info.vue b/src/modules/demo/views/crud/components/table/component/user-info.vue new file mode 100644 index 0000000..d095584 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/component/user-info.vue @@ -0,0 +1,30 @@ + + + + + + diff --git a/src/modules/demo/views/crud/components/table/context-menu.vue b/src/modules/demo/views/crud/components/table/context-menu.vue new file mode 100644 index 0000000..83be23d --- /dev/null +++ b/src/modules/demo/views/crud/components/table/context-menu.vue @@ -0,0 +1,191 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/dict.vue b/src/modules/demo/views/crud/components/table/dict.vue new file mode 100644 index 0000000..1c79e6f --- /dev/null +++ b/src/modules/demo/views/crud/components/table/dict.vue @@ -0,0 +1,156 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/hidden.vue b/src/modules/demo/views/crud/components/table/hidden.vue new file mode 100644 index 0000000..75db479 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/hidden.vue @@ -0,0 +1,127 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/op.vue b/src/modules/demo/views/crud/components/table/op.vue new file mode 100644 index 0000000..02dee22 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/op.vue @@ -0,0 +1,163 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/plugin/column.tsx b/src/modules/demo/views/crud/components/table/plugin/column.tsx new file mode 100644 index 0000000..ad54b28 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/plugin/column.tsx @@ -0,0 +1,48 @@ +import { merge } from "lodash-es"; +import { defineComponent } from "vue"; + +const columns = { + UserInfo: { + label: "用户信息", + minWidth: 200, + component: { + vm: defineComponent({ + name: "user-info", + + props: { + scope: null + }, + + setup(props) { + return () => { + return ( +
+

{props.scope.name}

+

{props.scope.phone}

+
+ ); + }; + } + }) + } + } +} as { [key: string]: DeepPartial }; + +/** + * 列标签匹配,方便多个列表公用同一个组件 + * @returns + */ +export function setColumn(): ClTable.Plugin { + return ({ exposed }) => { + function deep(arr: ClTable.Column[]) { + arr.forEach((e) => { + if (e.tag) { + merge(e, columns[e.tag]); + } + deep(e.children || []); + }); + } + + deep(exposed.columns); + }; +} diff --git a/src/modules/demo/views/crud/components/table/plugin/index.vue b/src/modules/demo/views/crud/components/table/plugin/index.vue new file mode 100644 index 0000000..5cae6fb --- /dev/null +++ b/src/modules/demo/views/crud/components/table/plugin/index.vue @@ -0,0 +1,86 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/search.vue b/src/modules/demo/views/crud/components/table/search.vue new file mode 100644 index 0000000..781dd97 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/search.vue @@ -0,0 +1,143 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/selection.vue b/src/modules/demo/views/crud/components/table/selection.vue new file mode 100644 index 0000000..26417dd --- /dev/null +++ b/src/modules/demo/views/crud/components/table/selection.vue @@ -0,0 +1,110 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/slot.vue b/src/modules/demo/views/crud/components/table/slot.vue new file mode 100644 index 0000000..1a5389d --- /dev/null +++ b/src/modules/demo/views/crud/components/table/slot.vue @@ -0,0 +1,97 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/span-method.vue b/src/modules/demo/views/crud/components/table/span-method.vue new file mode 100644 index 0000000..6795537 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/span-method.vue @@ -0,0 +1,116 @@ + + + diff --git a/src/modules/demo/views/crud/components/table/summary.vue b/src/modules/demo/views/crud/components/table/summary.vue new file mode 100644 index 0000000..11af421 --- /dev/null +++ b/src/modules/demo/views/crud/components/table/summary.vue @@ -0,0 +1,95 @@ + + + diff --git a/src/modules/demo/views/crud/components/upsert/base.vue b/src/modules/demo/views/crud/components/upsert/base.vue new file mode 100644 index 0000000..1841670 --- /dev/null +++ b/src/modules/demo/views/crud/components/upsert/base.vue @@ -0,0 +1,133 @@ + + + diff --git a/src/modules/demo/views/crud/components/upsert/event.vue b/src/modules/demo/views/crud/components/upsert/event.vue new file mode 100644 index 0000000..1f8ac61 --- /dev/null +++ b/src/modules/demo/views/crud/components/upsert/event.vue @@ -0,0 +1,210 @@ + + + diff --git a/src/modules/demo/views/crud/components/upsert/hook/index.vue b/src/modules/demo/views/crud/components/upsert/hook/index.vue new file mode 100644 index 0000000..5b6c0b6 --- /dev/null +++ b/src/modules/demo/views/crud/components/upsert/hook/index.vue @@ -0,0 +1,200 @@ + + + diff --git a/src/modules/demo/views/crud/components/upsert/hook/reg-pca2.ts b/src/modules/demo/views/crud/components/upsert/hook/reg-pca2.ts new file mode 100644 index 0000000..1c28981 --- /dev/null +++ b/src/modules/demo/views/crud/components/upsert/hook/reg-pca2.ts @@ -0,0 +1,14 @@ +import { registerFormHook } from "@cool-vue/crud"; + +// 注册 hook +registerFormHook("pca2", (value, { method, form, prop }) => { + if (method == "bind") { + return [form.province, form.city, form.district]; + } else { + const [province, city, district] = value || []; + form.province = province; + form.city = city; + form.district = district; + form[prop] = undefined; + } +}); diff --git a/src/modules/demo/views/crud/components/upsert/mode.vue b/src/modules/demo/views/crud/components/upsert/mode.vue new file mode 100644 index 0000000..3ee388b --- /dev/null +++ b/src/modules/demo/views/crud/components/upsert/mode.vue @@ -0,0 +1,146 @@ + + + diff --git a/src/modules/demo/views/crud/index.vue b/src/modules/demo/views/crud/index.vue new file mode 100644 index 0000000..8188a51 --- /dev/null +++ b/src/modules/demo/views/crud/index.vue @@ -0,0 +1,292 @@ + + + + + diff --git a/src/modules/demo/views/home/components/count-category.vue b/src/modules/demo/views/home/components/count-category.vue new file mode 100644 index 0000000..1789763 --- /dev/null +++ b/src/modules/demo/views/home/components/count-category.vue @@ -0,0 +1,128 @@ + + + + + diff --git a/src/modules/demo/views/home/components/count-effect.vue b/src/modules/demo/views/home/components/count-effect.vue new file mode 100644 index 0000000..85a0472 --- /dev/null +++ b/src/modules/demo/views/home/components/count-effect.vue @@ -0,0 +1,100 @@ + + + + + diff --git a/src/modules/demo/views/home/components/count-paid.vue b/src/modules/demo/views/home/components/count-paid.vue new file mode 100644 index 0000000..0896713 --- /dev/null +++ b/src/modules/demo/views/home/components/count-paid.vue @@ -0,0 +1,125 @@ + + + + + diff --git a/src/modules/demo/views/home/components/count-user.vue b/src/modules/demo/views/home/components/count-user.vue new file mode 100644 index 0000000..22665b8 --- /dev/null +++ b/src/modules/demo/views/home/components/count-user.vue @@ -0,0 +1,84 @@ + + + + + diff --git a/src/modules/demo/views/home/components/goods-rank.vue b/src/modules/demo/views/home/components/goods-rank.vue new file mode 100644 index 0000000..c09e576 --- /dev/null +++ b/src/modules/demo/views/home/components/goods-rank.vue @@ -0,0 +1,93 @@ + + + + + diff --git a/src/modules/demo/views/home/components/tab-chart.vue b/src/modules/demo/views/home/components/tab-chart.vue new file mode 100644 index 0000000..256af30 --- /dev/null +++ b/src/modules/demo/views/home/components/tab-chart.vue @@ -0,0 +1,200 @@ + + + + + diff --git a/src/modules/demo/views/home/index.vue b/src/modules/demo/views/home/index.vue new file mode 100644 index 0000000..8f594e0 --- /dev/null +++ b/src/modules/demo/views/home/index.vue @@ -0,0 +1,114 @@ + + + + + diff --git a/src/modules/dict/config.ts b/src/modules/dict/config.ts new file mode 100644 index 0000000..8d2a998 --- /dev/null +++ b/src/modules/dict/config.ts @@ -0,0 +1,13 @@ +import type { ModuleConfig } from "/@/cool"; +import { useDict } from "./index"; + +export default (): ModuleConfig => { + return { + onLoad({ hasToken }) { + const { dict } = useDict(); + hasToken(() => { + dict.refresh(); + }); + } + }; +}; diff --git a/src/modules/dict/index.ts b/src/modules/dict/index.ts new file mode 100644 index 0000000..6227d00 --- /dev/null +++ b/src/modules/dict/index.ts @@ -0,0 +1,7 @@ +import { useStore } from "./store"; + +export function useDict() { + return { + ...useStore() + }; +} diff --git a/src/modules/dict/store/dict.ts b/src/modules/dict/store/dict.ts new file mode 100644 index 0000000..1e70083 --- /dev/null +++ b/src/modules/dict/store/dict.ts @@ -0,0 +1,67 @@ +import { defineStore } from "pinia"; +import { computed, reactive, toRaw } from "vue"; +import { Dict } from "../types"; +import { service } from "/@/cool"; +import { deepTree } from "/@/cool/utils"; +import { isDev } from "/@/config"; +import { isArray } from "lodash-es"; +import { deepFind } from "../utils"; + +const useDictStore = defineStore("dict", () => { + // 对象数据 + const data = reactive({}); + + // 获取 + function get(name: string) { + return computed(() => data[name] || []); + } + + // 查找 + function find(name: string, value: any | any[]) { + const arr = isArray(value) ? value : [value]; + return arr.filter((e) => e !== undefined).map((v) => deepFind(v, get(name).value)); + } + + // 刷新 + async function refresh(types?: string[]) { + return service.dict.info + .data({ + types + }) + .then((res: Dict.Data) => { + const d = {}; + + for (const [i, arr] of Object.entries(res)) { + arr.forEach((e) => { + e.label = e.name; + + if (e.value === undefined || e.value === "" || e.value === null) { + e.value = e.id; + } + }); + + // @ts-ignore + d[i] = deepTree(arr, "desc"); + } + + Object.assign(data, d); + + if (isDev) { + console.group("字典数据"); + console.log(toRaw(data)); + console.groupEnd(); + } + + return data; + }); + } + + return { + data, + get, + find, + refresh + }; +}); + +export { useDictStore }; diff --git a/src/modules/dict/store/index.ts b/src/modules/dict/store/index.ts new file mode 100644 index 0000000..38221ef --- /dev/null +++ b/src/modules/dict/store/index.ts @@ -0,0 +1,9 @@ +import { useDictStore } from "./dict"; + +export function useStore() { + const dict = useDictStore(); + + return { + dict + }; +} diff --git a/src/modules/dict/types/index.d.ts b/src/modules/dict/types/index.d.ts new file mode 100644 index 0000000..272f608 --- /dev/null +++ b/src/modules/dict/types/index.d.ts @@ -0,0 +1,13 @@ +export namespace Dict { + interface Item { + id: string; + label: string; + value: any; + children?: Item[]; + [key: string]: any; + } + + interface Data { + [key: string]: Item[]; + } +} diff --git a/src/modules/dict/utils/index.ts b/src/modules/dict/utils/index.ts new file mode 100644 index 0000000..a46d699 --- /dev/null +++ b/src/modules/dict/utils/index.ts @@ -0,0 +1,17 @@ +export function deepFind(value: any, list: any[]) { + function deep(arr: any[]): any | undefined { + for (const e of arr) { + if (e.value === value) { + return e; + } else if (e.children) { + const d = deep(e.children); + if (d !== undefined) { + return d; + } + } + } + return undefined; + } + + return deep(list); +} diff --git a/src/modules/dict/views/list.vue b/src/modules/dict/views/list.vue new file mode 100644 index 0000000..c396fbe --- /dev/null +++ b/src/modules/dict/views/list.vue @@ -0,0 +1,295 @@ + + + + + diff --git a/src/modules/goods/components/select.vue b/src/modules/goods/components/select.vue new file mode 100644 index 0000000..ca689cd --- /dev/null +++ b/src/modules/goods/components/select.vue @@ -0,0 +1,371 @@ + + + + + diff --git a/src/modules/goods/components/spec.vue b/src/modules/goods/components/spec.vue new file mode 100644 index 0000000..e4a0293 --- /dev/null +++ b/src/modules/goods/components/spec.vue @@ -0,0 +1,322 @@ + + + + + diff --git a/src/modules/goods/views/comment.vue b/src/modules/goods/views/comment.vue new file mode 100644 index 0000000..7f289bb --- /dev/null +++ b/src/modules/goods/views/comment.vue @@ -0,0 +1,72 @@ + + + diff --git a/src/modules/goods/views/info.vue b/src/modules/goods/views/info.vue new file mode 100644 index 0000000..dea0e0d --- /dev/null +++ b/src/modules/goods/views/info.vue @@ -0,0 +1,276 @@ + + + diff --git a/src/modules/goods/views/search-keyword.vue b/src/modules/goods/views/search-keyword.vue new file mode 100644 index 0000000..915b788 --- /dev/null +++ b/src/modules/goods/views/search-keyword.vue @@ -0,0 +1,85 @@ + + + diff --git a/src/modules/goods/views/type.vue b/src/modules/goods/views/type.vue new file mode 100644 index 0000000..117fd97 --- /dev/null +++ b/src/modules/goods/views/type.vue @@ -0,0 +1,162 @@ + + + diff --git a/src/modules/helper/components/auto-menu/btn.vue b/src/modules/helper/components/auto-menu/btn.vue new file mode 100644 index 0000000..4e1402e --- /dev/null +++ b/src/modules/helper/components/auto-menu/btn.vue @@ -0,0 +1,43 @@ + + + + + diff --git a/src/modules/helper/components/auto-menu/index.vue b/src/modules/helper/components/auto-menu/index.vue new file mode 100644 index 0000000..3c42f2e --- /dev/null +++ b/src/modules/helper/components/auto-menu/index.vue @@ -0,0 +1,8 @@ + + + diff --git a/src/modules/helper/components/auto-menu/quick.vue b/src/modules/helper/components/auto-menu/quick.vue new file mode 100644 index 0000000..d0a1169 --- /dev/null +++ b/src/modules/helper/components/auto-menu/quick.vue @@ -0,0 +1,255 @@ + + + diff --git a/src/modules/helper/components/auto-perms/index.vue b/src/modules/helper/components/auto-perms/index.vue new file mode 100644 index 0000000..93451a2 --- /dev/null +++ b/src/modules/helper/components/auto-perms/index.vue @@ -0,0 +1,206 @@ + + + + + diff --git a/src/modules/helper/config.ts b/src/modules/helper/config.ts new file mode 100644 index 0000000..00a83fa --- /dev/null +++ b/src/modules/helper/config.ts @@ -0,0 +1,27 @@ +import type { ModuleConfig } from "/@/cool"; +import { getRules } from "./utils"; + +export default (): ModuleConfig => { + return { + options: { + host: "https://service.cool-js.com/api" + }, + toolbar: { + order: 1, + component: import("./components/auto-menu/btn.vue") + }, + pages: [ + { + path: "/helper/ai-code", + meta: { + label: "Ai 极速编码", + keepAlive: true + }, + component: () => import("./views/ai-code.vue") + } + ], + onLoad() { + getRules(); + } + }; +}; diff --git a/src/modules/helper/dict/index.ts b/src/modules/helper/dict/index.ts new file mode 100644 index 0000000..1354740 --- /dev/null +++ b/src/modules/helper/dict/index.ts @@ -0,0 +1,3 @@ +import type { PropRule } from "../types"; + +export const PropRules: PropRule[] = []; diff --git a/src/modules/helper/hooks/ai.ts b/src/modules/helper/hooks/ai.ts new file mode 100644 index 0000000..c83831e --- /dev/null +++ b/src/modules/helper/hooks/ai.ts @@ -0,0 +1,102 @@ +import { module } from "/@/cool"; +import { useBase } from "/$/base"; + +export function useAi() { + const { host } = module.config("helper"); + const { user } = useBase(); + + // 调用流程 + async function invokeFlow( + label: string, + params: any, + streamCb?: ({ isEnd, content }: { isEnd: boolean; content: string }) => void + ): Promise { + const stream = !!streamCb; + + let cacheText = ""; + + return new Promise((resolve, reject) => { + fetch(host + "/open/code/gen/data", { + method: "POST", + headers: { + Authorization: user.token, + "Content-Type": "application/json" + }, + body: JSON.stringify({ + params, + label, + stream + }) + }) + .then((res) => { + if (res.body) { + if (stream) { + const reader = res.body.getReader(); + const decoder = new TextDecoder("utf-8"); + const stream = new ReadableStream({ + start(controller) { + function push() { + reader.read().then(({ done, value }) => { + if (done) { + controller.close(); + return; + } + + let text = decoder.decode(value, { stream: true }); + + if (streamCb) { + if (cacheText) { + text = cacheText + text; + } + + if (text.indexOf("data:") == 0) { + text = "\n\n" + text; + } + + try { + const arr = text + .split(/\n\ndata:/g) + .filter(Boolean) + .map((e) => JSON.parse(e)); + + arr.forEach(streamCb); + + cacheText = ""; + } catch (err) { + cacheText = text; + } + } + + controller.enqueue(text); + push(); + }); + } + push(); + } + }); + + return new Response(stream); + } else { + return res.json(); + } + } + }) + .then((res) => { + if (stream) { + return res; + } + + if (res.code == 1000) { + resolve(res.data.result); + } else { + reject(res); + } + }) + .catch(reject); + }); + } + + return { + invokeFlow + }; +} diff --git a/src/modules/helper/hooks/code.ts b/src/modules/helper/hooks/code.ts new file mode 100644 index 0000000..e406f2c --- /dev/null +++ b/src/modules/helper/hooks/code.ts @@ -0,0 +1,454 @@ +import { isEmpty, isFunction, isRegExp, isString } from "lodash-es"; +import { PropRules } from "../dict"; +import type { EpsColumn, EpsModule } from "../types"; + +export function useCode() { + // 特殊情况处理 + const handler = { + // 单选 + dict({ comment }: EpsColumn) { + const [label, ...arr] = comment.split(" "); + + // 选项列表 + const list: any[] = arr.map((e) => { + const [value, label] = e.split("-"); + + return { + label, + value: isNaN(Number(value)) ? value : Number(value) + }; + }); + + // boolean + if (list.length == 2) { + list.forEach((e) => { + if (e.value == 1) { + e.type = "success"; + } else { + e.type = "danger"; + } + }); + } + + const d = { + table: { + label, + dict: list, + dictColor: true, + minWidth: 120 + }, + form: { + label, + component: { + name: "", + options: list + } + } as ClForm.Item + }; + + // 默认值 + if (list[0]) { + d.form.value = list[0].value; + } + + // 匹配组件 + d.form.component!.name = arr.length > 4 ? "el-select" : "el-radio-group"; + + return d; + }, + + // 多选 + dict_multiple(column: EpsColumn) { + const { table, form } = handler.dict(column); + + if (!form.component?.props) { + form.component!.props = {}; + } + + if (!form.value) { + form.value = []; + } + + switch (form.component?.name) { + case "el-select": + Object.assign(form.component.props, { + multiple: true, + filterable: true + }); + break; + + case "el-radio-group": + form.component.name = "el-checkbox-group"; + break; + } + + return { + table, + form + }; + } + }; + + // 创建组件 + function createComponent(column: EpsColumn, columns: EpsColumn[]) { + const prop = column.propertyName; + let label = column.comment || ""; + let d: any; + let isHidden = false; + + // 根据规则匹配组件 + PropRules.find((r) => { + let s = false; + + // 已知组件的情况下 + if (column.component) { + if (column.component == r.value) { + s = true; + } + } else { + // 根据规则匹配 + if (r.test) { + s = !!r.test.find((e) => { + if (isRegExp(e)) { + return e.test(prop); + } + + if (isFunction(e)) { + return e(prop); + } + + if (isString(e)) { + if (e == prop) { + return true; + } + + const re = new RegExp(`${e}$`); + return re.test(prop.toLocaleLowerCase()); + } + + return false; + }); + } + } + + if (r.group) { + if ( + r.group.includes(prop) && + r.group.some((e) => columns.find((c) => c.propertyName == e)) + ) { + if (r.group[0] == prop) { + s = true; + } else { + isHidden = true; + } + } + } + + if (s) { + if (r.handler) { + // @ts-ignore + const fn = isString(r.handler) ? handler[r.handler] : r.handler; + + if (isFunction(fn)) { + d = fn(column); + } + } else { + d = { + ...r, + test: undefined + }; + } + } + + return s; + }); + + // 没找到则默认input + if (!d) { + d = PropRules.find((e) => e.value == "input")?.render; + } + + // 格式化标题 + label = label.split(" ")[0]; + + return { + column: { + label, + prop, + ...d?.table + }, + item: { + label, + prop, + ...d?.form + }, + isHidden + }; + } + + // 创建 vue 代码 + function createVue({ + router = "", + columns = [], + prefix = "", + api = [], + fieldEq = [], + keyWordLikeFields = [] + }: EpsModule) { + // 新增、编辑 + const upsert = { + items: [] as DeepPartial[] + }; + + // 表格 + const table = { + columns: [] as DeepPartial[] + }; + + // 选项 + const options = {}; + + // 遍历 + columns.forEach((e) => { + // 创建组件 + const { item, column, isHidden } = createComponent(e, columns); + + // 过滤隐藏 + if (isHidden) { + return false; + } + + // 验证规则 + if (!e.nullable) { + item.required = true; + } + + // 字典 + const dict = item.component?.options || column.dict; + + if (!isEmpty(dict)) { + options[item.prop] = dict; + + const str = `$$options.${item.prop}$$`; + + if (!column.component) { + column.dict = str; + } + + item.component.options = str; + } + + // 表单忽略 + if (!["createTime", "updateTime", "id", "endTime", "endDate"].includes(item.prop)) { + upsert.items.push(item); + } + + // 表格忽略 + if (!["id"].includes(item.prop)) { + // 默认排序 + if (item.prop == "createTime") { + column.sortable = "desc"; + } + + table.columns.push(column); + } + + // 时间范围处理 + if (["startTime", "startDate"].includes(item.prop)) { + const key = item.prop.replace("start", ""); + + if (columns.find((e) => e.propertyName == "end" + key)) { + item.prop = key.toLocaleLowerCase(); + const isTime = item.prop == "time"; + item.label = isTime ? "时间范围" : "日期范围"; + item.hook = "datetimeRange"; + item.component = { + name: "el-date-picker", + props: { + type: isTime ? "datetimerange" : "daterange", + valueFormat: isTime ? "YYYY-MM-DD HH:mm:ss" : "YYYY-MM-DD 00:00:00", + defaultTime: [ + new Date(2000, 1, 1, 0, 0, 0), + new Date(2000, 1, 1, 23, 59, 59) + ] + } + }; + } + } + }); + + // 请求服务 + const service = prefix.replace("/admin", "service").split("/"); + + // 请求地址 + const paths = api.map((e) => e.path); + + // 权限 + const perms = { + add: paths.includes("/add"), + del: paths.includes("/delete"), + update: paths.includes("/info") && paths.includes("/update"), + page: paths.includes("/page"), + upsert: true + }; + perms.upsert = perms.add || perms.update; + + // 是否有操作栏 + if (perms.del || perms.upsert) { + const buttons: ClTable.OpButton = []; + + if (perms.upsert) { + buttons.push("edit"); + } + + if (perms.del) { + buttons.push("delete"); + } + + table.columns.push({ + type: "op", + buttons + }); + } + + // 是否多选、序号 + if (perms.del) { + table.columns.unshift({ + type: "selection" + }); + } else { + table.columns.unshift({ + label: "#", + type: "index" + }); + } + + // 筛选 + const clFilter = fieldEq + .map((field) => { + if (isEmpty(options[field])) { + return ""; + } + + const item = upsert.items.find((e) => e.prop == field); + + if (!item) { + return ""; + } + + return `\n\n\n`; + }) + .filter(Boolean) + .join("\n"); + + // 关键字搜索 + const clSearchKeyPlaceholder = keyWordLikeFields + .map((field) => { + return table.columns.find((e) => e.prop == field)?.label; + }) + .filter((e) => !!e) + .join("、"); + + // 选项 + const ConstOptions = `${ + isEmpty(options) + ? "" + : "\n// 选项\nconst options = reactive(" + toCodeString(options) + ")\n" + }`; + + // Vue 依赖 + let ImportVue = ""; + + if (ConstOptions) { + ImportVue = "import { reactive } from 'vue';\n"; + } + + // 代码模板 + const temp = ` + +`; + + return temp.replace(/"\$\$|\$\$"/g, ""); + } + + // 转成代码字符串 + function toCodeString(data: any) { + const arr: string[][] = []; + + let code = JSON.stringify(data, (key, value) => { + if (isFunction(value)) { + const str = value.toString(); + arr.push([JSON.stringify({ [key]: str }), str]); + return str; + } else { + return value; + } + }); + + arr.forEach((e) => { + code = code.replace(e[0].substring(1, e[0].length - 1), e[1]); + }); + + return code; + } + + return { + handler, + createVue, + createComponent, + toCodeString + }; +} diff --git a/src/modules/helper/hooks/index.ts b/src/modules/helper/hooks/index.ts new file mode 100644 index 0000000..d73e809 --- /dev/null +++ b/src/modules/helper/hooks/index.ts @@ -0,0 +1,3 @@ +export * from "./ai"; +export * from "./menu"; +export * from "./code"; diff --git a/src/modules/helper/hooks/menu.ts b/src/modules/helper/hooks/menu.ts new file mode 100644 index 0000000..8cf5815 --- /dev/null +++ b/src/modules/helper/hooks/menu.ts @@ -0,0 +1,85 @@ +import { ElMessage } from "element-plus"; +import type { EpsModule } from "../types"; +import { service } from "/@/cool"; +import { useCode } from "./code"; + +export function useMenu() { + const { createVue } = useCode(); + + // 创建菜单、权限、文件 + function create(data: EpsModule): Promise<() => void> { + return new Promise((resolve, reject) => { + // 视图文件路径 + data.viewPath = `modules/${data.module}/views${data.router?.replace( + `/${data.module}`, + "" + )}.vue`; + + // 添加菜单 + service.base.sys.menu + .add({ + type: 1, + isShow: true, + keepAlive: true, + ...data, + api: undefined, + code: undefined, + columns: undefined + }) + .then((res) => { + // 权限列表 + const perms = data.api?.map((e) => { + const d = { + type: 2, + parentId: res.id, + name: e.summary || e.path, + perms: [e.path] + }; + + if (e.path == "/update") { + if (data.api?.find((a) => a.path == "/info")) { + d.perms.push("/info"); + } + } + + return { + ...d, + perms: d.perms + .map((e) => + (data.prefix?.replace("/admin/", "") + e).replace(/\//g, ":") + ) + .join(",") + }; + }); + + // 批量插入权限 + service.base.sys.menu.add(perms).then(() => { + resolve(() => { + service + .request({ + url: "/__cool_createMenu", + method: "POST", + proxy: false, + data: { + code: createVue(data), + ...data + } + }) + .then(() => { + location.reload(); + }); + }); + }); + }) + .catch((err) => { + ElMessage.error(err.message); + reject(); + }); + }); + } + + return { + create, + createVue + }; +} diff --git a/src/modules/helper/static/index.scss b/src/modules/helper/static/index.scss new file mode 100644 index 0000000..9cc75b6 --- /dev/null +++ b/src/modules/helper/static/index.scss @@ -0,0 +1,109 @@ +.plugins { + padding: 10px; + + &__wrapper { + background-color: var(--el-bg-color); + } + + .scope { + border-radius: 8px; + margin-bottom: 10px; + border: 1px solid var(--el-border-color-light); + height: 200px; + width: 100%; + box-sizing: border-box; + cursor: pointer; + + .c { + display: flex; + box-sizing: border-box; + padding: 15px; + height: calc(100% - 50px); + position: relative; + + .set { + position: absolute; + right: 10px; + top: 10px; + font-size: 18px; + color: var(--el-color-info); + } + + .logo { + height: 40px; + width: 40px; + margin-right: 15px; + } + + .det { + display: flex; + flex-direction: column; + flex: 1; + + .tag { + margin-bottom: 10px; + + .el-tag { + margin-right: 5px; + } + } + + .title { + display: flex; + align-items: center; + margin-bottom: 5px; + font-size: 14px; + line-height: 1; + font-weight: bold; + } + + .desc { + font-size: 12px; + flex: 1; + display: -webkit-box; + -webkit-line-clamp: 3; + -webkit-box-orient: vertical; + overflow: hidden; + color: var(--el-text-color-regular); + } + + .link { + display: flex; + align-items: center; + } + + .author { + font-size: 12px; + color: var(--el-text-color-secondary); + } + } + } + + .f { + display: flex; + align-items: center; + justify-content: space-between; + padding: 10px; + height: 30px; + } + + &.is-add { + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + background-color: var(--el-disabled-bg-color); + border-color: var(--el-disabled-bg-color); + width: 180px; + + .el-icon { + font-size: 36px; + color: #666; + } + } + + &:not(.is-add):hover { + box-shadow: 0px 0px 10px 1px var(--el-color-info-light-9); + } + } +} diff --git a/src/modules/helper/static/svg/arrow-down.svg b/src/modules/helper/static/svg/arrow-down.svg new file mode 100644 index 0000000..3348788 --- /dev/null +++ b/src/modules/helper/static/svg/arrow-down.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/helper/static/svg/code.svg b/src/modules/helper/static/svg/code.svg new file mode 100644 index 0000000..12e1ae3 --- /dev/null +++ b/src/modules/helper/static/svg/code.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/helper/static/svg/enter.svg b/src/modules/helper/static/svg/enter.svg new file mode 100644 index 0000000..64f5500 --- /dev/null +++ b/src/modules/helper/static/svg/enter.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/modules/helper/static/svg/icon-vue.svg b/src/modules/helper/static/svg/icon-vue.svg new file mode 100644 index 0000000..6e08773 --- /dev/null +++ b/src/modules/helper/static/svg/icon-vue.svg @@ -0,0 +1,18 @@ + + + + + diff --git a/src/modules/helper/types/index.d.ts b/src/modules/helper/types/index.d.ts new file mode 100644 index 0000000..d479403 --- /dev/null +++ b/src/modules/helper/types/index.d.ts @@ -0,0 +1,50 @@ +export declare interface EpsColumn { + comment: string; + length: number; + nullable: boolean; + propertyName: string; + type: string; + component: string; +} + +export declare interface EpsApi { + path: string; + summary: string; + method?: string; + prefix?: string; + tag?: string; + dts?: {}; + [key: string]: any; +} + +export declare interface EpsModule { + api: EpsApi[]; + columns: EpsColumn[]; + prefix: string; + router: string; + module: string; + fieldEq?: string[]; + keyWordLikeFields?: string[]; + [key: string]: any; +} + +export declare interface EpsData { + [key: string]: EpsModule[]; +} + +export interface CodeItem { + label: string; + value: string; + content: string; + [key: string]: any; +} + +export declare interface PropRule { + value?: string; + test?: any[]; + group?: string[]; + table?: DeepPartial; + form?: ClForm.Item; + handler?: string; + [key: string]: any; +} diff --git a/src/modules/helper/utils/index.ts b/src/modules/helper/utils/index.ts new file mode 100644 index 0000000..2922e68 --- /dev/null +++ b/src/modules/helper/utils/index.ts @@ -0,0 +1,52 @@ +import axios from "axios"; +import { module } from "/@/cool"; +import { PropRules } from "../dict"; +import type { PropRule } from "../types"; + +// 统一请求 +export async function request(options: any): Promise { + const { host } = module.config("helper"); + + return new Promise((resolve, reject) => { + axios({ + ...options, + url: host + options.url + }) + .then((res) => { + const { code, data, message } = res.data; + + if (code == 1000) { + resolve(data); + } else { + reject({ message }); + } + }) + .catch(reject); + }); +} + +// 获取匹配规则 +export async function getRules() { + const res = await request({ + url: "/app/base/comm/param", + params: { + key: "epsFieldType" + } + }); + + let eps: { components: PropRule[] }; + + eval(` + ${res} + eps = EPS + `); + + const arr = eps!.components.map((e) => { + return { + ...e, + ...e.render + }; + }); + + PropRules.unshift(...arr); +} diff --git a/src/modules/helper/views/ai-code.vue b/src/modules/helper/views/ai-code.vue new file mode 100644 index 0000000..8ca50c2 --- /dev/null +++ b/src/modules/helper/views/ai-code.vue @@ -0,0 +1,1631 @@ + + + + + diff --git a/src/modules/helper/views/plugins/serve.vue b/src/modules/helper/views/plugins/serve.vue new file mode 100644 index 0000000..d51ee1d --- /dev/null +++ b/src/modules/helper/views/plugins/serve.vue @@ -0,0 +1,351 @@ + + + + + diff --git a/src/modules/helper/views/plugins/vue.vue b/src/modules/helper/views/plugins/vue.vue new file mode 100644 index 0000000..18f0c2a --- /dev/null +++ b/src/modules/helper/views/plugins/vue.vue @@ -0,0 +1,146 @@ + + + + + diff --git a/src/modules/info/views/banner.vue b/src/modules/info/views/banner.vue new file mode 100644 index 0000000..06f388f --- /dev/null +++ b/src/modules/info/views/banner.vue @@ -0,0 +1,137 @@ + + + diff --git a/src/modules/market/views/coupon/info.vue b/src/modules/market/views/coupon/info.vue new file mode 100644 index 0000000..172b12a --- /dev/null +++ b/src/modules/market/views/coupon/info.vue @@ -0,0 +1,282 @@ + + + + + diff --git a/src/modules/market/views/coupon/user.vue b/src/modules/market/views/coupon/user.vue new file mode 100644 index 0000000..fbca95a --- /dev/null +++ b/src/modules/market/views/coupon/user.vue @@ -0,0 +1,129 @@ + + + diff --git a/src/modules/order/components/discount-item.vue b/src/modules/order/components/discount-item.vue new file mode 100644 index 0000000..234626a --- /dev/null +++ b/src/modules/order/components/discount-item.vue @@ -0,0 +1,45 @@ + + + + + diff --git a/src/modules/order/components/goods.vue b/src/modules/order/components/goods.vue new file mode 100644 index 0000000..47dd51c --- /dev/null +++ b/src/modules/order/components/goods.vue @@ -0,0 +1,145 @@ + + + + + diff --git a/src/modules/order/components/info-item.vue b/src/modules/order/components/info-item.vue new file mode 100644 index 0000000..ec2a9e1 --- /dev/null +++ b/src/modules/order/components/info-item.vue @@ -0,0 +1,24 @@ + + + + + diff --git a/src/modules/order/components/logistics-item.vue b/src/modules/order/components/logistics-item.vue new file mode 100644 index 0000000..c6b183b --- /dev/null +++ b/src/modules/order/components/logistics-item.vue @@ -0,0 +1,229 @@ + + + + + diff --git a/src/modules/order/components/pay-item.vue b/src/modules/order/components/pay-item.vue new file mode 100644 index 0000000..87c0298 --- /dev/null +++ b/src/modules/order/components/pay-item.vue @@ -0,0 +1,20 @@ + + + diff --git a/src/modules/order/components/remark-item.vue b/src/modules/order/components/remark-item.vue new file mode 100644 index 0000000..adf5fa0 --- /dev/null +++ b/src/modules/order/components/remark-item.vue @@ -0,0 +1,26 @@ + + + + + diff --git a/src/modules/order/components/user-item.vue b/src/modules/order/components/user-item.vue new file mode 100644 index 0000000..88bc284 --- /dev/null +++ b/src/modules/order/components/user-item.vue @@ -0,0 +1,20 @@ + + + + + diff --git a/src/modules/order/dict/index.ts b/src/modules/order/dict/index.ts new file mode 100644 index 0000000..667e175 --- /dev/null +++ b/src/modules/order/dict/index.ts @@ -0,0 +1,16 @@ +export const OrderStatus = [ + { label: "待付款", value: 0, type: "info" }, + { label: "待发货", value: 1, type: "warning" }, + { label: "待收货", value: 2 }, + { label: "待评价", value: 3, type: "success" }, + { label: "交易完成", value: 4, type: "success" }, + { label: "退款中", value: 5, type: "danger" }, + { label: "已退款", value: 6, type: "danger" }, + { label: "已关闭", value: 7, type: "danger" } +]; + +export const PayType = [ + { label: "待支付", value: 0, type: "info" }, + { label: "微信", value: 1, color: "#2aae67" }, + { label: "支付宝", value: 2, color: "#1678FF" } +]; diff --git a/src/modules/order/views/info.vue b/src/modules/order/views/info.vue new file mode 100644 index 0000000..8865899 --- /dev/null +++ b/src/modules/order/views/info.vue @@ -0,0 +1,165 @@ + + + diff --git a/src/modules/order/views/refund.vue b/src/modules/order/views/refund.vue new file mode 100644 index 0000000..59f4275 --- /dev/null +++ b/src/modules/order/views/refund.vue @@ -0,0 +1,272 @@ + + + diff --git a/src/modules/recycle/views/data.vue b/src/modules/recycle/views/data.vue new file mode 100644 index 0000000..d185c8a --- /dev/null +++ b/src/modules/recycle/views/data.vue @@ -0,0 +1,125 @@ + + + diff --git a/src/modules/space/components/space-inner.vue b/src/modules/space/components/space-inner.vue new file mode 100644 index 0000000..46ca62c --- /dev/null +++ b/src/modules/space/components/space-inner.vue @@ -0,0 +1,447 @@ + + + + + diff --git a/src/modules/space/components/space.vue b/src/modules/space/components/space.vue new file mode 100644 index 0000000..dd2d8ee --- /dev/null +++ b/src/modules/space/components/space.vue @@ -0,0 +1,165 @@ + + + + + diff --git a/src/modules/space/config.ts b/src/modules/space/config.ts new file mode 100644 index 0000000..6a250d8 --- /dev/null +++ b/src/modules/space/config.ts @@ -0,0 +1,20 @@ +import type { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return { + components: [ + () => import("./components/space.vue"), + () => import("./components/space-inner.vue") + ], + + views: [ + { + meta: { + label: "文件空间" + }, + path: "/space/list", + component: () => import("./views/list.vue") + } + ] + }; +}; diff --git a/src/modules/space/hooks/index.ts b/src/modules/space/hooks/index.ts new file mode 100644 index 0000000..43e9112 --- /dev/null +++ b/src/modules/space/hooks/index.ts @@ -0,0 +1,19 @@ +import { type Ref, inject } from "vue"; + +declare interface Space { + loading: boolean; + category: { + id: number | undefined; + visible: boolean; + }; + list: Ref; + selection: Ref; + refresh(params: any): Promise; + preview(item: Eps.SpaceInfoEntity): void; +} + +export function useSpace() { + const space = inject("space") as Space; + + return { space }; +} diff --git a/src/modules/space/views/list.vue b/src/modules/space/views/list.vue new file mode 100644 index 0000000..3c84134 --- /dev/null +++ b/src/modules/space/views/list.vue @@ -0,0 +1,7 @@ + + + diff --git a/src/modules/task/components/logs.vue b/src/modules/task/components/logs.vue new file mode 100644 index 0000000..68eabe6 --- /dev/null +++ b/src/modules/task/components/logs.vue @@ -0,0 +1,113 @@ + + + diff --git a/src/modules/task/views/list.vue b/src/modules/task/views/list.vue new file mode 100644 index 0000000..304dea9 --- /dev/null +++ b/src/modules/task/views/list.vue @@ -0,0 +1,502 @@ + + + + + diff --git a/src/modules/user/components/select.vue b/src/modules/user/components/select.vue new file mode 100644 index 0000000..f280576 --- /dev/null +++ b/src/modules/user/components/select.vue @@ -0,0 +1,373 @@ + + + + + diff --git a/src/modules/user/config.ts b/src/modules/user/config.ts new file mode 100644 index 0000000..fddae1f --- /dev/null +++ b/src/modules/user/config.ts @@ -0,0 +1,5 @@ +import type { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return {}; +}; diff --git a/src/modules/user/views/list.vue b/src/modules/user/views/list.vue new file mode 100644 index 0000000..a1aa653 --- /dev/null +++ b/src/modules/user/views/list.vue @@ -0,0 +1,216 @@ + + + diff --git a/src/plugins/crud/components/column-custom/index.vue b/src/plugins/crud/components/column-custom/index.vue new file mode 100644 index 0000000..562f13e --- /dev/null +++ b/src/plugins/crud/components/column-custom/index.vue @@ -0,0 +1,196 @@ + + + + + diff --git a/src/plugins/crud/components/date/picker.vue b/src/plugins/crud/components/date/picker.vue new file mode 100644 index 0000000..94473ae --- /dev/null +++ b/src/plugins/crud/components/date/picker.vue @@ -0,0 +1,175 @@ + + + + + diff --git a/src/plugins/crud/components/date/text.vue b/src/plugins/crud/components/date/text.vue new file mode 100644 index 0000000..550ca75 --- /dev/null +++ b/src/plugins/crud/components/date/text.vue @@ -0,0 +1,30 @@ + + + diff --git a/src/plugins/crud/components/select/index.tsx b/src/plugins/crud/components/select/index.tsx new file mode 100644 index 0000000..9a6890a --- /dev/null +++ b/src/plugins/crud/components/select/index.tsx @@ -0,0 +1,143 @@ +import { useCrud } from "@cool-vue/crud"; +import { isEmpty, isString } from "lodash-es"; +import { computed, defineComponent, type PropType, type Ref, toValue, useModel } from "vue"; +import { parsePx } from "/@/cool/utils"; +import type { Dict } from "/$/dict/types"; + +export default defineComponent({ + name: "cl-select", + + props: { + modelValue: [String, Number, Array], + options: { + type: [Array, Object] as PropType>, + default: () => [] + }, + prop: String, + scope: null, + labelKey: { + type: String, + default: "label" + }, + valueKey: { + type: String, + default: "value" + }, + width: [String, Number], + // 是否树形 + tree: Boolean, + // 是否返回选中层级下的所有值 + allLevelsId: Boolean, + // 是否父子不互相关联 + checkStrictly: Boolean + }, + + emits: ["update:modelValue", "change"], + + setup(props, { emit }) { + // cl-crud + const Crud = useCrud(); + + // 选中值 + const value = useModel(props, "modelValue"); + + // 列表 + const list = computed(() => { + return toValue(props.options) || []; + }); + + // 获取值 + function getValue(val: any): any | any[] { + if (props.allLevelsId) { + const ids: any[] = []; + + // 获取所有的值 + function deep(arr: Dict.Item[], f: boolean) { + arr.forEach((e) => { + const f2 = e[props.valueKey] == val; + + if (f || f2) { + ids.push(e[props.valueKey]); + } + + if (e.children) { + deep(e.children, f || f2); + } + }); + } + + deep(list.value, false); + + return isEmpty(ids) ? undefined : ids; + } else { + return val === "" ? undefined : val; + } + } + + // 值改变 + function onChange(val: any) { + const v = getValue(val); + + emit("update:modelValue", v); + emit("change", v); + + if (props.prop && !props.scope) { + Crud.value?.refresh({ page: 1, [props.prop]: v }); + } + } + + return () => { + // 样式 + const style = { + width: parsePx(props.width!) + }; + + // 占位符 + const placeholder = props.prop ? "选择搜索" : "请选择"; + + // 树形下拉框 + const TreeSelect = ( + + ); + + // 普通下拉框 + const Select = ( + + {list.value?.map((e) => { + return isString(e) ? ( + + ) : ( + + ); + })} + + ); + + return props.tree ? TreeSelect : Select; + }; + } +}); diff --git a/src/plugins/crud/components/switch/index.tsx b/src/plugins/crud/components/switch/index.tsx new file mode 100644 index 0000000..331e6da --- /dev/null +++ b/src/plugins/crud/components/switch/index.tsx @@ -0,0 +1,111 @@ +import { useCrud } from "@cool-vue/crud"; +import { ElMessage } from "element-plus"; +import { defineComponent, ref, watch } from "vue"; +import { isBoolean, isFunction } from "lodash-es"; + +export default defineComponent({ + name: "cl-switch", + + props: { + scope: null, + column: null, + modelValue: [Number, String, Boolean], + activeValue: { + type: [Number, String, Boolean], + default: 1 + }, + inactiveValue: { + type: [Number, String, Boolean], + default: 0 + }, + api: Function + }, + + emits: ["update:modelValue", "change"], + + setup(props, { emit }) { + // cl-crud + const Crud = useCrud(); + + // 状态 + const status = ref(); + + // 选中值类型 + const activeValue = ref(); + const inactiveValue = ref(); + + // 监听值 + watch( + () => props.modelValue, + (val) => { + status.value = val; + + if (val !== undefined) { + if (isBoolean(val)) { + activeValue.value = true; + inactiveValue.value = false; + + return true; + } + } + + activeValue.value = props.activeValue; + inactiveValue.value = props.inactiveValue; + }, + { + immediate: true + } + ); + + // 监听改变 + function onChange(val: boolean | string | number) { + if (props.column && props.scope) { + if (val !== undefined) { + if ( + status.value === activeValue.value || + status.value === inactiveValue.value + ) { + const params = { + id: props.scope.id, + [props.column.property]: val + }; + + const req: Promise = isFunction(props.api) + ? props.api(params) + : Crud.value?.service.update(params); + + if (req) { + req.then(() => { + emit("update:modelValue", val); + emit("change", val); + ElMessage.success("更新成功"); + }).catch((err) => { + ElMessage.error(err.message); + }); + } + } + } + } else { + emit("update:modelValue", val); + emit("change", val); + } + } + + // 点击事件, 阻止冒泡 + function onClick(event: MouseEvent) { + event.stopPropagation(); + } + + return () => { + return ( + + ); + }; + } +}); diff --git a/src/plugins/crud/config.ts b/src/plugins/crud/config.ts new file mode 100644 index 0000000..7f9b378 --- /dev/null +++ b/src/plugins/crud/config.ts @@ -0,0 +1,51 @@ +import type { Merge, ModuleConfig } from "/@/cool"; + +// npm +import Crud, { locale, setFocus } from "@cool-vue/crud"; +import "@cool-vue/crud/dist/index.css"; + +export default (): Merge => { + return { + label: "CRUD", + description: "快速增删改查及一系列辅助组件", + author: "COOL", + version: "1.0.6", + updateTime: "2024-03-19", + demo: "/demo/crud", + + // 组件全注册 + components: Object.values(import.meta.glob("./components/**/*.{vue,tsx}")), + + // 配置参数,具体配置点 CrudOptions 查看 + options: { + style: { + table: { + // 插件列表 + plugins: [] + + // 右键菜单,为空则关闭 + // contextMenu: [] + }, + form: { + // 插件列表 + plugins: [ + // 自动聚焦插件 + setFocus() + ] + } + }, + dict: { + // 排序字段 + sort: { + prop: "order", + order: "sort" + }, + // 按钮及提示文案 + label: locale.zhCn + } + }, + + // 安装 + install: Crud.install + }; +}; diff --git a/src/plugins/distpicker/components/index.tsx b/src/plugins/distpicker/components/index.tsx new file mode 100644 index 0000000..91d4e47 --- /dev/null +++ b/src/plugins/distpicker/components/index.tsx @@ -0,0 +1,40 @@ +import { type PropType, defineComponent } from "vue"; +import data from "../data/pca.json"; + +export default defineComponent({ + name: "cl-distpicker", + + props: { + props: Object, + type: { + type: String as PropType<"pc" | "pca">, + default: "pca" + } + }, + + setup(props) { + return () => { + return ( + { + if (props.type === "pc") { + return { + ...e, + children: e.children.map((a) => { + return { + ...a, + children: undefined + }; + }) + }; + } + + return e; + })} + props={{ label: "name", value: "name", ...props.props }} + /> + ); + }; + } +}); diff --git a/src/plugins/distpicker/config.ts b/src/plugins/distpicker/config.ts new file mode 100644 index 0000000..8bda75c --- /dev/null +++ b/src/plugins/distpicker/config.ts @@ -0,0 +1,36 @@ +import type { ModuleConfig } from "/@/cool"; +import { registerFormHook } from "@cool-vue/crud"; + +// 注册hook +registerFormHook("pca", (value, { method, form, prop }) => { + if (method == "bind") { + return [form.province, form.city, form.district].filter(Boolean); + } else { + const [province, city, district] = value || []; + form.province = province; + form.city = city; + form.district = district; + form[prop] = undefined; + } +}); + +export default (): ModuleConfig => { + return { + label: "省市区选择器", + description: "快速增删改查及一系列辅助组件", + author: "COOL", + version: "1.0.1", + updateTime: "2024-02-04", + demo: [ + { + name: "基础用法", + component: () => import("./demo/base.vue") + } + ], + + components: [ + // 省市区选择 https://github.com/modood/Administrative-divisions-of-China + () => import("./components/index") + ] + }; +}; diff --git a/src/plugins/distpicker/data/pca.json b/src/plugins/distpicker/data/pca.json new file mode 100644 index 0000000..300f132 --- /dev/null +++ b/src/plugins/distpicker/data/pca.json @@ -0,0 +1,5300 @@ +[ + { + "code": "11", + "name": "北京市", + "children": [ + { + "code": "1101", + "name": "市辖区", + "children": [ + { "code": "110101", "name": "东城区" }, + { "code": "110102", "name": "西城区" }, + { "code": "110105", "name": "朝阳区" }, + { "code": "110106", "name": "丰台区" }, + { "code": "110107", "name": "石景山区" }, + { "code": "110108", "name": "海淀区" }, + { "code": "110109", "name": "门头沟区" }, + { "code": "110111", "name": "房山区" }, + { "code": "110112", "name": "通州区" }, + { "code": "110113", "name": "顺义区" }, + { "code": "110114", "name": "昌平区" }, + { "code": "110115", "name": "大兴区" }, + { "code": "110116", "name": "怀柔区" }, + { "code": "110117", "name": "平谷区" }, + { "code": "110118", "name": "密云区" }, + { "code": "110119", "name": "延庆区" } + ] + } + ] + }, + { + "code": "12", + "name": "天津市", + "children": [ + { + "code": "1201", + "name": "市辖区", + "children": [ + { "code": "120101", "name": "和平区" }, + { "code": "120102", "name": "河东区" }, + { "code": "120103", "name": "河西区" }, + { "code": "120104", "name": "南开区" }, + { "code": "120105", "name": "河北区" }, + { "code": "120106", "name": "红桥区" }, + { "code": "120110", "name": "东丽区" }, + { "code": "120111", "name": "西青区" }, + { "code": "120112", "name": "津南区" }, + { "code": "120113", "name": "北辰区" }, + { "code": "120114", "name": "武清区" }, + { "code": "120115", "name": "宝坻区" }, + { "code": "120116", "name": "滨海新区" }, + { "code": "120117", "name": "宁河区" }, + { "code": "120118", "name": "静海区" }, + { "code": "120119", "name": "蓟州区" } + ] + } + ] + }, + { + "code": "13", + "name": "河北省", + "children": [ + { + "code": "1301", + "name": "石家庄市", + "children": [ + { "code": "130102", "name": "长安区" }, + { "code": "130104", "name": "桥西区" }, + { "code": "130105", "name": "新华区" }, + { "code": "130107", "name": "井陉矿区" }, + { "code": "130108", "name": "裕华区" }, + { "code": "130109", "name": "藁城区" }, + { "code": "130110", "name": "鹿泉区" }, + { "code": "130111", "name": "栾城区" }, + { "code": "130121", "name": "井陉县" }, + { "code": "130123", "name": "正定县" }, + { "code": "130125", "name": "行唐县" }, + { "code": "130126", "name": "灵寿县" }, + { "code": "130127", "name": "高邑县" }, + { "code": "130128", "name": "深泽县" }, + { "code": "130129", "name": "赞皇县" }, + { "code": "130130", "name": "无极县" }, + { "code": "130131", "name": "平山县" }, + { "code": "130132", "name": "元氏县" }, + { "code": "130133", "name": "赵县" }, + { "code": "130171", "name": "石家庄高新技术产业开发区" }, + { "code": "130172", "name": "石家庄循环化工园区" }, + { "code": "130181", "name": "辛集市" }, + { "code": "130183", "name": "晋州市" }, + { "code": "130184", "name": "新乐市" } + ] + }, + { + "code": "1302", + "name": "唐山市", + "children": [ + { "code": "130202", "name": "路南区" }, + { "code": "130203", "name": "路北区" }, + { "code": "130204", "name": "古冶区" }, + { "code": "130205", "name": "开平区" }, + { "code": "130207", "name": "丰南区" }, + { "code": "130208", "name": "丰润区" }, + { "code": "130209", "name": "曹妃甸区" }, + { "code": "130224", "name": "滦南县" }, + { "code": "130225", "name": "乐亭县" }, + { "code": "130227", "name": "迁西县" }, + { "code": "130229", "name": "玉田县" }, + { "code": "130271", "name": "河北唐山芦台经济开发区" }, + { "code": "130272", "name": "唐山市汉沽管理区" }, + { "code": "130273", "name": "唐山高新技术产业开发区" }, + { "code": "130274", "name": "河北唐山海港经济开发区" }, + { "code": "130281", "name": "遵化市" }, + { "code": "130283", "name": "迁安市" }, + { "code": "130284", "name": "滦州市" } + ] + }, + { + "code": "1303", + "name": "秦皇岛市", + "children": [ + { "code": "130302", "name": "海港区" }, + { "code": "130303", "name": "山海关区" }, + { "code": "130304", "name": "北戴河区" }, + { "code": "130306", "name": "抚宁区" }, + { "code": "130321", "name": "青龙满族自治县" }, + { "code": "130322", "name": "昌黎县" }, + { "code": "130324", "name": "卢龙县" }, + { "code": "130371", "name": "秦皇岛市经济技术开发区" }, + { "code": "130372", "name": "北戴河新区" } + ] + }, + { + "code": "1304", + "name": "邯郸市", + "children": [ + { "code": "130402", "name": "邯山区" }, + { "code": "130403", "name": "丛台区" }, + { "code": "130404", "name": "复兴区" }, + { "code": "130406", "name": "峰峰矿区" }, + { "code": "130407", "name": "肥乡区" }, + { "code": "130408", "name": "永年区" }, + { "code": "130423", "name": "临漳县" }, + { "code": "130424", "name": "成安县" }, + { "code": "130425", "name": "大名县" }, + { "code": "130426", "name": "涉县" }, + { "code": "130427", "name": "磁县" }, + { "code": "130430", "name": "邱县" }, + { "code": "130431", "name": "鸡泽县" }, + { "code": "130432", "name": "广平县" }, + { "code": "130433", "name": "馆陶县" }, + { "code": "130434", "name": "魏县" }, + { "code": "130435", "name": "曲周县" }, + { "code": "130471", "name": "邯郸经济技术开发区" }, + { "code": "130473", "name": "邯郸冀南新区" }, + { "code": "130481", "name": "武安市" } + ] + }, + { + "code": "1305", + "name": "邢台市", + "children": [ + { "code": "130502", "name": "襄都区" }, + { "code": "130503", "name": "信都区" }, + { "code": "130505", "name": "任泽区" }, + { "code": "130506", "name": "南和区" }, + { "code": "130522", "name": "临城县" }, + { "code": "130523", "name": "内丘县" }, + { "code": "130524", "name": "柏乡县" }, + { "code": "130525", "name": "隆尧县" }, + { "code": "130528", "name": "宁晋县" }, + { "code": "130529", "name": "巨鹿县" }, + { "code": "130530", "name": "新河县" }, + { "code": "130531", "name": "广宗县" }, + { "code": "130532", "name": "平乡县" }, + { "code": "130533", "name": "威县" }, + { "code": "130534", "name": "清河县" }, + { "code": "130535", "name": "临西县" }, + { "code": "130571", "name": "河北邢台经济开发区" }, + { "code": "130581", "name": "南宫市" }, + { "code": "130582", "name": "沙河市" } + ] + }, + { + "code": "1306", + "name": "保定市", + "children": [ + { "code": "130602", "name": "竞秀区" }, + { "code": "130606", "name": "莲池区" }, + { "code": "130607", "name": "满城区" }, + { "code": "130608", "name": "清苑区" }, + { "code": "130609", "name": "徐水区" }, + { "code": "130623", "name": "涞水县" }, + { "code": "130624", "name": "阜平县" }, + { "code": "130626", "name": "定兴县" }, + { "code": "130627", "name": "唐县" }, + { "code": "130628", "name": "高阳县" }, + { "code": "130629", "name": "容城县" }, + { "code": "130630", "name": "涞源县" }, + { "code": "130631", "name": "望都县" }, + { "code": "130632", "name": "安新县" }, + { "code": "130633", "name": "易县" }, + { "code": "130634", "name": "曲阳县" }, + { "code": "130635", "name": "蠡县" }, + { "code": "130636", "name": "顺平县" }, + { "code": "130637", "name": "博野县" }, + { "code": "130638", "name": "雄县" }, + { "code": "130671", "name": "保定高新技术产业开发区" }, + { "code": "130672", "name": "保定白沟新城" }, + { "code": "130681", "name": "涿州市" }, + { "code": "130682", "name": "定州市" }, + { "code": "130683", "name": "安国市" }, + { "code": "130684", "name": "高碑店市" } + ] + }, + { + "code": "1307", + "name": "张家口市", + "children": [ + { "code": "130702", "name": "桥东区" }, + { "code": "130703", "name": "桥西区" }, + { "code": "130705", "name": "宣化区" }, + { "code": "130706", "name": "下花园区" }, + { "code": "130708", "name": "万全区" }, + { "code": "130709", "name": "崇礼区" }, + { "code": "130722", "name": "张北县" }, + { "code": "130723", "name": "康保县" }, + { "code": "130724", "name": "沽源县" }, + { "code": "130725", "name": "尚义县" }, + { "code": "130726", "name": "蔚县" }, + { "code": "130727", "name": "阳原县" }, + { "code": "130728", "name": "怀安县" }, + { "code": "130730", "name": "怀来县" }, + { "code": "130731", "name": "涿鹿县" }, + { "code": "130732", "name": "赤城县" }, + { "code": "130771", "name": "张家口经济开发区" }, + { "code": "130772", "name": "张家口市察北管理区" }, + { "code": "130773", "name": "张家口市塞北管理区" } + ] + }, + { + "code": "1308", + "name": "承德市", + "children": [ + { "code": "130802", "name": "双桥区" }, + { "code": "130803", "name": "双滦区" }, + { "code": "130804", "name": "鹰手营子矿区" }, + { "code": "130821", "name": "承德县" }, + { "code": "130822", "name": "兴隆县" }, + { "code": "130824", "name": "滦平县" }, + { "code": "130825", "name": "隆化县" }, + { "code": "130826", "name": "丰宁满族自治县" }, + { "code": "130827", "name": "宽城满族自治县" }, + { "code": "130828", "name": "围场满族蒙古族自治县" }, + { "code": "130871", "name": "承德高新技术产业开发区" }, + { "code": "130881", "name": "平泉市" } + ] + }, + { + "code": "1309", + "name": "沧州市", + "children": [ + { "code": "130902", "name": "新华区" }, + { "code": "130903", "name": "运河区" }, + { "code": "130921", "name": "沧县" }, + { "code": "130922", "name": "青县" }, + { "code": "130923", "name": "东光县" }, + { "code": "130924", "name": "海兴县" }, + { "code": "130925", "name": "盐山县" }, + { "code": "130926", "name": "肃宁县" }, + { "code": "130927", "name": "南皮县" }, + { "code": "130928", "name": "吴桥县" }, + { "code": "130929", "name": "献县" }, + { "code": "130930", "name": "孟村回族自治县" }, + { "code": "130971", "name": "河北沧州经济开发区" }, + { "code": "130972", "name": "沧州高新技术产业开发区" }, + { "code": "130973", "name": "沧州渤海新区" }, + { "code": "130981", "name": "泊头市" }, + { "code": "130982", "name": "任丘市" }, + { "code": "130983", "name": "黄骅市" }, + { "code": "130984", "name": "河间市" } + ] + }, + { + "code": "1310", + "name": "廊坊市", + "children": [ + { "code": "131002", "name": "安次区" }, + { "code": "131003", "name": "广阳区" }, + { "code": "131022", "name": "固安县" }, + { "code": "131023", "name": "永清县" }, + { "code": "131024", "name": "香河县" }, + { "code": "131025", "name": "大城县" }, + { "code": "131026", "name": "文安县" }, + { "code": "131028", "name": "大厂回族自治县" }, + { "code": "131071", "name": "廊坊经济技术开发区" }, + { "code": "131081", "name": "霸州市" }, + { "code": "131082", "name": "三河市" } + ] + }, + { + "code": "1311", + "name": "衡水市", + "children": [ + { "code": "131102", "name": "桃城区" }, + { "code": "131103", "name": "冀州区" }, + { "code": "131121", "name": "枣强县" }, + { "code": "131122", "name": "武邑县" }, + { "code": "131123", "name": "武强县" }, + { "code": "131124", "name": "饶阳县" }, + { "code": "131125", "name": "安平县" }, + { "code": "131126", "name": "故城县" }, + { "code": "131127", "name": "景县" }, + { "code": "131128", "name": "阜城县" }, + { "code": "131171", "name": "河北衡水高新技术产业开发区" }, + { "code": "131172", "name": "衡水滨湖新区" }, + { "code": "131182", "name": "深州市" } + ] + } + ] + }, + { + "code": "14", + "name": "山西省", + "children": [ + { + "code": "1401", + "name": "太原市", + "children": [ + { "code": "140105", "name": "小店区" }, + { "code": "140106", "name": "迎泽区" }, + { "code": "140107", "name": "杏花岭区" }, + { "code": "140108", "name": "尖草坪区" }, + { "code": "140109", "name": "万柏林区" }, + { "code": "140110", "name": "晋源区" }, + { "code": "140121", "name": "清徐县" }, + { "code": "140122", "name": "阳曲县" }, + { "code": "140123", "name": "娄烦县" }, + { "code": "140171", "name": "山西转型综合改革示范区" }, + { "code": "140181", "name": "古交市" } + ] + }, + { + "code": "1402", + "name": "大同市", + "children": [ + { "code": "140212", "name": "新荣区" }, + { "code": "140213", "name": "平城区" }, + { "code": "140214", "name": "云冈区" }, + { "code": "140215", "name": "云州区" }, + { "code": "140221", "name": "阳高县" }, + { "code": "140222", "name": "天镇县" }, + { "code": "140223", "name": "广灵县" }, + { "code": "140224", "name": "灵丘县" }, + { "code": "140225", "name": "浑源县" }, + { "code": "140226", "name": "左云县" }, + { "code": "140271", "name": "山西大同经济开发区" } + ] + }, + { + "code": "1403", + "name": "阳泉市", + "children": [ + { "code": "140302", "name": "城区" }, + { "code": "140303", "name": "矿区" }, + { "code": "140311", "name": "郊区" }, + { "code": "140321", "name": "平定县" }, + { "code": "140322", "name": "盂县" } + ] + }, + { + "code": "1404", + "name": "长治市", + "children": [ + { "code": "140403", "name": "潞州区" }, + { "code": "140404", "name": "上党区" }, + { "code": "140405", "name": "屯留区" }, + { "code": "140406", "name": "潞城区" }, + { "code": "140423", "name": "襄垣县" }, + { "code": "140425", "name": "平顺县" }, + { "code": "140426", "name": "黎城县" }, + { "code": "140427", "name": "壶关县" }, + { "code": "140428", "name": "长子县" }, + { "code": "140429", "name": "武乡县" }, + { "code": "140430", "name": "沁县" }, + { "code": "140431", "name": "沁源县" }, + { "code": "140471", "name": "山西长治高新技术产业园区" } + ] + }, + { + "code": "1405", + "name": "晋城市", + "children": [ + { "code": "140502", "name": "城区" }, + { "code": "140521", "name": "沁水县" }, + { "code": "140522", "name": "阳城县" }, + { "code": "140524", "name": "陵川县" }, + { "code": "140525", "name": "泽州县" }, + { "code": "140581", "name": "高平市" } + ] + }, + { + "code": "1406", + "name": "朔州市", + "children": [ + { "code": "140602", "name": "朔城区" }, + { "code": "140603", "name": "平鲁区" }, + { "code": "140621", "name": "山阴县" }, + { "code": "140622", "name": "应县" }, + { "code": "140623", "name": "右玉县" }, + { "code": "140671", "name": "山西朔州经济开发区" }, + { "code": "140681", "name": "怀仁市" } + ] + }, + { + "code": "1407", + "name": "晋中市", + "children": [ + { "code": "140702", "name": "榆次区" }, + { "code": "140703", "name": "太谷区" }, + { "code": "140721", "name": "榆社县" }, + { "code": "140722", "name": "左权县" }, + { "code": "140723", "name": "和顺县" }, + { "code": "140724", "name": "昔阳县" }, + { "code": "140725", "name": "寿阳县" }, + { "code": "140727", "name": "祁县" }, + { "code": "140728", "name": "平遥县" }, + { "code": "140729", "name": "灵石县" }, + { "code": "140781", "name": "介休市" } + ] + }, + { + "code": "1408", + "name": "运城市", + "children": [ + { "code": "140802", "name": "盐湖区" }, + { "code": "140821", "name": "临猗县" }, + { "code": "140822", "name": "万荣县" }, + { "code": "140823", "name": "闻喜县" }, + { "code": "140824", "name": "稷山县" }, + { "code": "140825", "name": "新绛县" }, + { "code": "140826", "name": "绛县" }, + { "code": "140827", "name": "垣曲县" }, + { "code": "140828", "name": "夏县" }, + { "code": "140829", "name": "平陆县" }, + { "code": "140830", "name": "芮城县" }, + { "code": "140881", "name": "永济市" }, + { "code": "140882", "name": "河津市" } + ] + }, + { + "code": "1409", + "name": "忻州市", + "children": [ + { "code": "140902", "name": "忻府区" }, + { "code": "140921", "name": "定襄县" }, + { "code": "140922", "name": "五台县" }, + { "code": "140923", "name": "代县" }, + { "code": "140924", "name": "繁峙县" }, + { "code": "140925", "name": "宁武县" }, + { "code": "140926", "name": "静乐县" }, + { "code": "140927", "name": "神池县" }, + { "code": "140928", "name": "五寨县" }, + { "code": "140929", "name": "岢岚县" }, + { "code": "140930", "name": "河曲县" }, + { "code": "140931", "name": "保德县" }, + { "code": "140932", "name": "偏关县" }, + { "code": "140971", "name": "五台山风景名胜区" }, + { "code": "140981", "name": "原平市" } + ] + }, + { + "code": "1410", + "name": "临汾市", + "children": [ + { "code": "141002", "name": "尧都区" }, + { "code": "141021", "name": "曲沃县" }, + { "code": "141022", "name": "翼城县" }, + { "code": "141023", "name": "襄汾县" }, + { "code": "141024", "name": "洪洞县" }, + { "code": "141025", "name": "古县" }, + { "code": "141026", "name": "安泽县" }, + { "code": "141027", "name": "浮山县" }, + { "code": "141028", "name": "吉县" }, + { "code": "141029", "name": "乡宁县" }, + { "code": "141030", "name": "大宁县" }, + { "code": "141031", "name": "隰县" }, + { "code": "141032", "name": "永和县" }, + { "code": "141033", "name": "蒲县" }, + { "code": "141034", "name": "汾西县" }, + { "code": "141081", "name": "侯马市" }, + { "code": "141082", "name": "霍州市" } + ] + }, + { + "code": "1411", + "name": "吕梁市", + "children": [ + { "code": "141102", "name": "离石区" }, + { "code": "141121", "name": "文水县" }, + { "code": "141122", "name": "交城县" }, + { "code": "141123", "name": "兴县" }, + { "code": "141124", "name": "临县" }, + { "code": "141125", "name": "柳林县" }, + { "code": "141126", "name": "石楼县" }, + { "code": "141127", "name": "岚县" }, + { "code": "141128", "name": "方山县" }, + { "code": "141129", "name": "中阳县" }, + { "code": "141130", "name": "交口县" }, + { "code": "141181", "name": "孝义市" }, + { "code": "141182", "name": "汾阳市" } + ] + } + ] + }, + { + "code": "15", + "name": "内蒙古自治区", + "children": [ + { + "code": "1501", + "name": "呼和浩特市", + "children": [ + { "code": "150102", "name": "新城区" }, + { "code": "150103", "name": "回民区" }, + { "code": "150104", "name": "玉泉区" }, + { "code": "150105", "name": "赛罕区" }, + { "code": "150121", "name": "土默特左旗" }, + { "code": "150122", "name": "托克托县" }, + { "code": "150123", "name": "和林格尔县" }, + { "code": "150124", "name": "清水河县" }, + { "code": "150125", "name": "武川县" }, + { "code": "150172", "name": "呼和浩特经济技术开发区" } + ] + }, + { + "code": "1502", + "name": "包头市", + "children": [ + { "code": "150202", "name": "东河区" }, + { "code": "150203", "name": "昆都仑区" }, + { "code": "150204", "name": "青山区" }, + { "code": "150205", "name": "石拐区" }, + { "code": "150206", "name": "白云鄂博矿区" }, + { "code": "150207", "name": "九原区" }, + { "code": "150221", "name": "土默特右旗" }, + { "code": "150222", "name": "固阳县" }, + { "code": "150223", "name": "达尔罕茂明安联合旗" }, + { "code": "150271", "name": "包头稀土高新技术产业开发区" } + ] + }, + { + "code": "1503", + "name": "乌海市", + "children": [ + { "code": "150302", "name": "海勃湾区" }, + { "code": "150303", "name": "海南区" }, + { "code": "150304", "name": "乌达区" } + ] + }, + { + "code": "1504", + "name": "赤峰市", + "children": [ + { "code": "150402", "name": "红山区" }, + { "code": "150403", "name": "元宝山区" }, + { "code": "150404", "name": "松山区" }, + { "code": "150421", "name": "阿鲁科尔沁旗" }, + { "code": "150422", "name": "巴林左旗" }, + { "code": "150423", "name": "巴林右旗" }, + { "code": "150424", "name": "林西县" }, + { "code": "150425", "name": "克什克腾旗" }, + { "code": "150426", "name": "翁牛特旗" }, + { "code": "150428", "name": "喀喇沁旗" }, + { "code": "150429", "name": "宁城县" }, + { "code": "150430", "name": "敖汉旗" } + ] + }, + { + "code": "1505", + "name": "通辽市", + "children": [ + { "code": "150502", "name": "科尔沁区" }, + { "code": "150521", "name": "科尔沁左翼中旗" }, + { "code": "150522", "name": "科尔沁左翼后旗" }, + { "code": "150523", "name": "开鲁县" }, + { "code": "150524", "name": "库伦旗" }, + { "code": "150525", "name": "奈曼旗" }, + { "code": "150526", "name": "扎鲁特旗" }, + { "code": "150571", "name": "通辽经济技术开发区" }, + { "code": "150581", "name": "霍林郭勒市" } + ] + }, + { + "code": "1506", + "name": "鄂尔多斯市", + "children": [ + { "code": "150602", "name": "东胜区" }, + { "code": "150603", "name": "康巴什区" }, + { "code": "150621", "name": "达拉特旗" }, + { "code": "150622", "name": "准格尔旗" }, + { "code": "150623", "name": "鄂托克前旗" }, + { "code": "150624", "name": "鄂托克旗" }, + { "code": "150625", "name": "杭锦旗" }, + { "code": "150626", "name": "乌审旗" }, + { "code": "150627", "name": "伊金霍洛旗" } + ] + }, + { + "code": "1507", + "name": "呼伦贝尔市", + "children": [ + { "code": "150702", "name": "海拉尔区" }, + { "code": "150703", "name": "扎赉诺尔区" }, + { "code": "150721", "name": "阿荣旗" }, + { "code": "150722", "name": "莫力达瓦达斡尔族自治旗" }, + { "code": "150723", "name": "鄂伦春自治旗" }, + { "code": "150724", "name": "鄂温克族自治旗" }, + { "code": "150725", "name": "陈巴尔虎旗" }, + { "code": "150726", "name": "新巴尔虎左旗" }, + { "code": "150727", "name": "新巴尔虎右旗" }, + { "code": "150781", "name": "满洲里市" }, + { "code": "150782", "name": "牙克石市" }, + { "code": "150783", "name": "扎兰屯市" }, + { "code": "150784", "name": "额尔古纳市" }, + { "code": "150785", "name": "根河市" } + ] + }, + { + "code": "1508", + "name": "巴彦淖尔市", + "children": [ + { "code": "150802", "name": "临河区" }, + { "code": "150821", "name": "五原县" }, + { "code": "150822", "name": "磴口县" }, + { "code": "150823", "name": "乌拉特前旗" }, + { "code": "150824", "name": "乌拉特中旗" }, + { "code": "150825", "name": "乌拉特后旗" }, + { "code": "150826", "name": "杭锦后旗" } + ] + }, + { + "code": "1509", + "name": "乌兰察布市", + "children": [ + { "code": "150902", "name": "集宁区" }, + { "code": "150921", "name": "卓资县" }, + { "code": "150922", "name": "化德县" }, + { "code": "150923", "name": "商都县" }, + { "code": "150924", "name": "兴和县" }, + { "code": "150925", "name": "凉城县" }, + { "code": "150926", "name": "察哈尔右翼前旗" }, + { "code": "150927", "name": "察哈尔右翼中旗" }, + { "code": "150928", "name": "察哈尔右翼后旗" }, + { "code": "150929", "name": "四子王旗" }, + { "code": "150981", "name": "丰镇市" } + ] + }, + { + "code": "1522", + "name": "兴安盟", + "children": [ + { "code": "152201", "name": "乌兰浩特市" }, + { "code": "152202", "name": "阿尔山市" }, + { "code": "152221", "name": "科尔沁右翼前旗" }, + { "code": "152222", "name": "科尔沁右翼中旗" }, + { "code": "152223", "name": "扎赉特旗" }, + { "code": "152224", "name": "突泉县" } + ] + }, + { + "code": "1525", + "name": "锡林郭勒盟", + "children": [ + { "code": "152501", "name": "二连浩特市" }, + { "code": "152502", "name": "锡林浩特市" }, + { "code": "152522", "name": "阿巴嘎旗" }, + { "code": "152523", "name": "苏尼特左旗" }, + { "code": "152524", "name": "苏尼特右旗" }, + { "code": "152525", "name": "东乌珠穆沁旗" }, + { "code": "152526", "name": "西乌珠穆沁旗" }, + { "code": "152527", "name": "太仆寺旗" }, + { "code": "152528", "name": "镶黄旗" }, + { "code": "152529", "name": "正镶白旗" }, + { "code": "152530", "name": "正蓝旗" }, + { "code": "152531", "name": "多伦县" }, + { "code": "152571", "name": "乌拉盖管委会" } + ] + }, + { + "code": "1529", + "name": "阿拉善盟", + "children": [ + { "code": "152921", "name": "阿拉善左旗" }, + { "code": "152922", "name": "阿拉善右旗" }, + { "code": "152923", "name": "额济纳旗" }, + { "code": "152971", "name": "内蒙古阿拉善高新技术产业开发区" } + ] + } + ] + }, + { + "code": "21", + "name": "辽宁省", + "children": [ + { + "code": "2101", + "name": "沈阳市", + "children": [ + { "code": "210102", "name": "和平区" }, + { "code": "210103", "name": "沈河区" }, + { "code": "210104", "name": "大东区" }, + { "code": "210105", "name": "皇姑区" }, + { "code": "210106", "name": "铁西区" }, + { "code": "210111", "name": "苏家屯区" }, + { "code": "210112", "name": "浑南区" }, + { "code": "210113", "name": "沈北新区" }, + { "code": "210114", "name": "于洪区" }, + { "code": "210115", "name": "辽中区" }, + { "code": "210123", "name": "康平县" }, + { "code": "210124", "name": "法库县" }, + { "code": "210181", "name": "新民市" } + ] + }, + { + "code": "2102", + "name": "大连市", + "children": [ + { "code": "210202", "name": "中山区" }, + { "code": "210203", "name": "西岗区" }, + { "code": "210204", "name": "沙河口区" }, + { "code": "210211", "name": "甘井子区" }, + { "code": "210212", "name": "旅顺口区" }, + { "code": "210213", "name": "金州区" }, + { "code": "210214", "name": "普兰店区" }, + { "code": "210224", "name": "长海县" }, + { "code": "210281", "name": "瓦房店市" }, + { "code": "210283", "name": "庄河市" } + ] + }, + { + "code": "2103", + "name": "鞍山市", + "children": [ + { "code": "210302", "name": "铁东区" }, + { "code": "210303", "name": "铁西区" }, + { "code": "210304", "name": "立山区" }, + { "code": "210311", "name": "千山区" }, + { "code": "210321", "name": "台安县" }, + { "code": "210323", "name": "岫岩满族自治县" }, + { "code": "210381", "name": "海城市" } + ] + }, + { + "code": "2104", + "name": "抚顺市", + "children": [ + { "code": "210402", "name": "新抚区" }, + { "code": "210403", "name": "东洲区" }, + { "code": "210404", "name": "望花区" }, + { "code": "210411", "name": "顺城区" }, + { "code": "210421", "name": "抚顺县" }, + { "code": "210422", "name": "新宾满族自治县" }, + { "code": "210423", "name": "清原满族自治县" } + ] + }, + { + "code": "2105", + "name": "本溪市", + "children": [ + { "code": "210502", "name": "平山区" }, + { "code": "210503", "name": "溪湖区" }, + { "code": "210504", "name": "明山区" }, + { "code": "210505", "name": "南芬区" }, + { "code": "210521", "name": "本溪满族自治县" }, + { "code": "210522", "name": "桓仁满族自治县" } + ] + }, + { + "code": "2106", + "name": "丹东市", + "children": [ + { "code": "210602", "name": "元宝区" }, + { "code": "210603", "name": "振兴区" }, + { "code": "210604", "name": "振安区" }, + { "code": "210624", "name": "宽甸满族自治县" }, + { "code": "210681", "name": "东港市" }, + { "code": "210682", "name": "凤城市" } + ] + }, + { + "code": "2107", + "name": "锦州市", + "children": [ + { "code": "210702", "name": "古塔区" }, + { "code": "210703", "name": "凌河区" }, + { "code": "210711", "name": "太和区" }, + { "code": "210726", "name": "黑山县" }, + { "code": "210727", "name": "义县" }, + { "code": "210781", "name": "凌海市" }, + { "code": "210782", "name": "北镇市" } + ] + }, + { + "code": "2108", + "name": "营口市", + "children": [ + { "code": "210802", "name": "站前区" }, + { "code": "210803", "name": "西市区" }, + { "code": "210804", "name": "鲅鱼圈区" }, + { "code": "210811", "name": "老边区" }, + { "code": "210881", "name": "盖州市" }, + { "code": "210882", "name": "大石桥市" } + ] + }, + { + "code": "2109", + "name": "阜新市", + "children": [ + { "code": "210902", "name": "海州区" }, + { "code": "210903", "name": "新邱区" }, + { "code": "210904", "name": "太平区" }, + { "code": "210905", "name": "清河门区" }, + { "code": "210911", "name": "细河区" }, + { "code": "210921", "name": "阜新蒙古族自治县" }, + { "code": "210922", "name": "彰武县" } + ] + }, + { + "code": "2110", + "name": "辽阳市", + "children": [ + { "code": "211002", "name": "白塔区" }, + { "code": "211003", "name": "文圣区" }, + { "code": "211004", "name": "宏伟区" }, + { "code": "211005", "name": "弓长岭区" }, + { "code": "211011", "name": "太子河区" }, + { "code": "211021", "name": "辽阳县" }, + { "code": "211081", "name": "灯塔市" } + ] + }, + { + "code": "2111", + "name": "盘锦市", + "children": [ + { "code": "211102", "name": "双台子区" }, + { "code": "211103", "name": "兴隆台区" }, + { "code": "211104", "name": "大洼区" }, + { "code": "211122", "name": "盘山县" } + ] + }, + { + "code": "2112", + "name": "铁岭市", + "children": [ + { "code": "211202", "name": "银州区" }, + { "code": "211204", "name": "清河区" }, + { "code": "211221", "name": "铁岭县" }, + { "code": "211223", "name": "西丰县" }, + { "code": "211224", "name": "昌图县" }, + { "code": "211281", "name": "调兵山市" }, + { "code": "211282", "name": "开原市" } + ] + }, + { + "code": "2113", + "name": "朝阳市", + "children": [ + { "code": "211302", "name": "双塔区" }, + { "code": "211303", "name": "龙城区" }, + { "code": "211321", "name": "朝阳县" }, + { "code": "211322", "name": "建平县" }, + { "code": "211324", "name": "喀喇沁左翼蒙古族自治县" }, + { "code": "211381", "name": "北票市" }, + { "code": "211382", "name": "凌源市" } + ] + }, + { + "code": "2114", + "name": "葫芦岛市", + "children": [ + { "code": "211402", "name": "连山区" }, + { "code": "211403", "name": "龙港区" }, + { "code": "211404", "name": "南票区" }, + { "code": "211421", "name": "绥中县" }, + { "code": "211422", "name": "建昌县" }, + { "code": "211481", "name": "兴城市" } + ] + } + ] + }, + { + "code": "22", + "name": "吉林省", + "children": [ + { + "code": "2201", + "name": "长春市", + "children": [ + { "code": "220102", "name": "南关区" }, + { "code": "220103", "name": "宽城区" }, + { "code": "220104", "name": "朝阳区" }, + { "code": "220105", "name": "二道区" }, + { "code": "220106", "name": "绿园区" }, + { "code": "220112", "name": "双阳区" }, + { "code": "220113", "name": "九台区" }, + { "code": "220122", "name": "农安县" }, + { "code": "220171", "name": "长春经济技术开发区" }, + { "code": "220172", "name": "长春净月高新技术产业开发区" }, + { "code": "220173", "name": "长春高新技术产业开发区" }, + { "code": "220174", "name": "长春汽车经济技术开发区" }, + { "code": "220182", "name": "榆树市" }, + { "code": "220183", "name": "德惠市" }, + { "code": "220184", "name": "公主岭市" } + ] + }, + { + "code": "2202", + "name": "吉林市", + "children": [ + { "code": "220202", "name": "昌邑区" }, + { "code": "220203", "name": "龙潭区" }, + { "code": "220204", "name": "船营区" }, + { "code": "220211", "name": "丰满区" }, + { "code": "220221", "name": "永吉县" }, + { "code": "220271", "name": "吉林经济开发区" }, + { "code": "220272", "name": "吉林高新技术产业开发区" }, + { "code": "220273", "name": "吉林中国新加坡食品区" }, + { "code": "220281", "name": "蛟河市" }, + { "code": "220282", "name": "桦甸市" }, + { "code": "220283", "name": "舒兰市" }, + { "code": "220284", "name": "磐石市" } + ] + }, + { + "code": "2203", + "name": "四平市", + "children": [ + { "code": "220302", "name": "铁西区" }, + { "code": "220303", "name": "铁东区" }, + { "code": "220322", "name": "梨树县" }, + { "code": "220323", "name": "伊通满族自治县" }, + { "code": "220382", "name": "双辽市" } + ] + }, + { + "code": "2204", + "name": "辽源市", + "children": [ + { "code": "220402", "name": "龙山区" }, + { "code": "220403", "name": "西安区" }, + { "code": "220421", "name": "东丰县" }, + { "code": "220422", "name": "东辽县" } + ] + }, + { + "code": "2205", + "name": "通化市", + "children": [ + { "code": "220502", "name": "东昌区" }, + { "code": "220503", "name": "二道江区" }, + { "code": "220521", "name": "通化县" }, + { "code": "220523", "name": "辉南县" }, + { "code": "220524", "name": "柳河县" }, + { "code": "220581", "name": "梅河口市" }, + { "code": "220582", "name": "集安市" } + ] + }, + { + "code": "2206", + "name": "白山市", + "children": [ + { "code": "220602", "name": "浑江区" }, + { "code": "220605", "name": "江源区" }, + { "code": "220621", "name": "抚松县" }, + { "code": "220622", "name": "靖宇县" }, + { "code": "220623", "name": "长白朝鲜族自治县" }, + { "code": "220681", "name": "临江市" } + ] + }, + { + "code": "2207", + "name": "松原市", + "children": [ + { "code": "220702", "name": "宁江区" }, + { "code": "220721", "name": "前郭尔罗斯蒙古族自治县" }, + { "code": "220722", "name": "长岭县" }, + { "code": "220723", "name": "乾安县" }, + { "code": "220771", "name": "吉林松原经济开发区" }, + { "code": "220781", "name": "扶余市" } + ] + }, + { + "code": "2208", + "name": "白城市", + "children": [ + { "code": "220802", "name": "洮北区" }, + { "code": "220821", "name": "镇赉县" }, + { "code": "220822", "name": "通榆县" }, + { "code": "220871", "name": "吉林白城经济开发区" }, + { "code": "220881", "name": "洮南市" }, + { "code": "220882", "name": "大安市" } + ] + }, + { + "code": "2224", + "name": "延边朝鲜族自治州", + "children": [ + { "code": "222401", "name": "延吉市" }, + { "code": "222402", "name": "图们市" }, + { "code": "222403", "name": "敦化市" }, + { "code": "222404", "name": "珲春市" }, + { "code": "222405", "name": "龙井市" }, + { "code": "222406", "name": "和龙市" }, + { "code": "222424", "name": "汪清县" }, + { "code": "222426", "name": "安图县" } + ] + } + ] + }, + { + "code": "23", + "name": "黑龙江省", + "children": [ + { + "code": "2301", + "name": "哈尔滨市", + "children": [ + { "code": "230102", "name": "道里区" }, + { "code": "230103", "name": "南岗区" }, + { "code": "230104", "name": "道外区" }, + { "code": "230108", "name": "平房区" }, + { "code": "230109", "name": "松北区" }, + { "code": "230110", "name": "香坊区" }, + { "code": "230111", "name": "呼兰区" }, + { "code": "230112", "name": "阿城区" }, + { "code": "230113", "name": "双城区" }, + { "code": "230123", "name": "依兰县" }, + { "code": "230124", "name": "方正县" }, + { "code": "230125", "name": "宾县" }, + { "code": "230126", "name": "巴彦县" }, + { "code": "230127", "name": "木兰县" }, + { "code": "230128", "name": "通河县" }, + { "code": "230129", "name": "延寿县" }, + { "code": "230183", "name": "尚志市" }, + { "code": "230184", "name": "五常市" } + ] + }, + { + "code": "2302", + "name": "齐齐哈尔市", + "children": [ + { "code": "230202", "name": "龙沙区" }, + { "code": "230203", "name": "建华区" }, + { "code": "230204", "name": "铁锋区" }, + { "code": "230205", "name": "昂昂溪区" }, + { "code": "230206", "name": "富拉尔基区" }, + { "code": "230207", "name": "碾子山区" }, + { "code": "230208", "name": "梅里斯达斡尔族区" }, + { "code": "230221", "name": "龙江县" }, + { "code": "230223", "name": "依安县" }, + { "code": "230224", "name": "泰来县" }, + { "code": "230225", "name": "甘南县" }, + { "code": "230227", "name": "富裕县" }, + { "code": "230229", "name": "克山县" }, + { "code": "230230", "name": "克东县" }, + { "code": "230231", "name": "拜泉县" }, + { "code": "230281", "name": "讷河市" } + ] + }, + { + "code": "2303", + "name": "鸡西市", + "children": [ + { "code": "230302", "name": "鸡冠区" }, + { "code": "230303", "name": "恒山区" }, + { "code": "230304", "name": "滴道区" }, + { "code": "230305", "name": "梨树区" }, + { "code": "230306", "name": "城子河区" }, + { "code": "230307", "name": "麻山区" }, + { "code": "230321", "name": "鸡东县" }, + { "code": "230381", "name": "虎林市" }, + { "code": "230382", "name": "密山市" } + ] + }, + { + "code": "2304", + "name": "鹤岗市", + "children": [ + { "code": "230402", "name": "向阳区" }, + { "code": "230403", "name": "工农区" }, + { "code": "230404", "name": "南山区" }, + { "code": "230405", "name": "兴安区" }, + { "code": "230406", "name": "东山区" }, + { "code": "230407", "name": "兴山区" }, + { "code": "230421", "name": "萝北县" }, + { "code": "230422", "name": "绥滨县" } + ] + }, + { + "code": "2305", + "name": "双鸭山市", + "children": [ + { "code": "230502", "name": "尖山区" }, + { "code": "230503", "name": "岭东区" }, + { "code": "230505", "name": "四方台区" }, + { "code": "230506", "name": "宝山区" }, + { "code": "230521", "name": "集贤县" }, + { "code": "230522", "name": "友谊县" }, + { "code": "230523", "name": "宝清县" }, + { "code": "230524", "name": "饶河县" } + ] + }, + { + "code": "2306", + "name": "大庆市", + "children": [ + { "code": "230602", "name": "萨尔图区" }, + { "code": "230603", "name": "龙凤区" }, + { "code": "230604", "name": "让胡路区" }, + { "code": "230605", "name": "红岗区" }, + { "code": "230606", "name": "大同区" }, + { "code": "230621", "name": "肇州县" }, + { "code": "230622", "name": "肇源县" }, + { "code": "230623", "name": "林甸县" }, + { "code": "230624", "name": "杜尔伯特蒙古族自治县" }, + { "code": "230671", "name": "大庆高新技术产业开发区" } + ] + }, + { + "code": "2307", + "name": "伊春市", + "children": [ + { "code": "230717", "name": "伊美区" }, + { "code": "230718", "name": "乌翠区" }, + { "code": "230719", "name": "友好区" }, + { "code": "230722", "name": "嘉荫县" }, + { "code": "230723", "name": "汤旺县" }, + { "code": "230724", "name": "丰林县" }, + { "code": "230725", "name": "大箐山县" }, + { "code": "230726", "name": "南岔县" }, + { "code": "230751", "name": "金林区" }, + { "code": "230781", "name": "铁力市" } + ] + }, + { + "code": "2308", + "name": "佳木斯市", + "children": [ + { "code": "230803", "name": "向阳区" }, + { "code": "230804", "name": "前进区" }, + { "code": "230805", "name": "东风区" }, + { "code": "230811", "name": "郊区" }, + { "code": "230822", "name": "桦南县" }, + { "code": "230826", "name": "桦川县" }, + { "code": "230828", "name": "汤原县" }, + { "code": "230881", "name": "同江市" }, + { "code": "230882", "name": "富锦市" }, + { "code": "230883", "name": "抚远市" } + ] + }, + { + "code": "2309", + "name": "七台河市", + "children": [ + { "code": "230902", "name": "新兴区" }, + { "code": "230903", "name": "桃山区" }, + { "code": "230904", "name": "茄子河区" }, + { "code": "230921", "name": "勃利县" } + ] + }, + { + "code": "2310", + "name": "牡丹江市", + "children": [ + { "code": "231002", "name": "东安区" }, + { "code": "231003", "name": "阳明区" }, + { "code": "231004", "name": "爱民区" }, + { "code": "231005", "name": "西安区" }, + { "code": "231025", "name": "林口县" }, + { "code": "231071", "name": "牡丹江经济技术开发区" }, + { "code": "231081", "name": "绥芬河市" }, + { "code": "231083", "name": "海林市" }, + { "code": "231084", "name": "宁安市" }, + { "code": "231085", "name": "穆棱市" }, + { "code": "231086", "name": "东宁市" } + ] + }, + { + "code": "2311", + "name": "黑河市", + "children": [ + { "code": "231102", "name": "爱辉区" }, + { "code": "231123", "name": "逊克县" }, + { "code": "231124", "name": "孙吴县" }, + { "code": "231181", "name": "北安市" }, + { "code": "231182", "name": "五大连池市" }, + { "code": "231183", "name": "嫩江市" } + ] + }, + { + "code": "2312", + "name": "绥化市", + "children": [ + { "code": "231202", "name": "北林区" }, + { "code": "231221", "name": "望奎县" }, + { "code": "231222", "name": "兰西县" }, + { "code": "231223", "name": "青冈县" }, + { "code": "231224", "name": "庆安县" }, + { "code": "231225", "name": "明水县" }, + { "code": "231226", "name": "绥棱县" }, + { "code": "231281", "name": "安达市" }, + { "code": "231282", "name": "肇东市" }, + { "code": "231283", "name": "海伦市" } + ] + }, + { + "code": "2327", + "name": "大兴安岭地区", + "children": [ + { "code": "232701", "name": "漠河市" }, + { "code": "232721", "name": "呼玛县" }, + { "code": "232722", "name": "塔河县" }, + { "code": "232761", "name": "加格达奇区" }, + { "code": "232762", "name": "松岭区" }, + { "code": "232763", "name": "新林区" }, + { "code": "232764", "name": "呼中区" } + ] + } + ] + }, + { + "code": "31", + "name": "上海市", + "children": [ + { + "code": "3101", + "name": "市辖区", + "children": [ + { "code": "310101", "name": "黄浦区" }, + { "code": "310104", "name": "徐汇区" }, + { "code": "310105", "name": "长宁区" }, + { "code": "310106", "name": "静安区" }, + { "code": "310107", "name": "普陀区" }, + { "code": "310109", "name": "虹口区" }, + { "code": "310110", "name": "杨浦区" }, + { "code": "310112", "name": "闵行区" }, + { "code": "310113", "name": "宝山区" }, + { "code": "310114", "name": "嘉定区" }, + { "code": "310115", "name": "浦东新区" }, + { "code": "310116", "name": "金山区" }, + { "code": "310117", "name": "松江区" }, + { "code": "310118", "name": "青浦区" }, + { "code": "310120", "name": "奉贤区" }, + { "code": "310151", "name": "崇明区" } + ] + } + ] + }, + { + "code": "32", + "name": "江苏省", + "children": [ + { + "code": "3201", + "name": "南京市", + "children": [ + { "code": "320102", "name": "玄武区" }, + { "code": "320104", "name": "秦淮区" }, + { "code": "320105", "name": "建邺区" }, + { "code": "320106", "name": "鼓楼区" }, + { "code": "320111", "name": "浦口区" }, + { "code": "320113", "name": "栖霞区" }, + { "code": "320114", "name": "雨花台区" }, + { "code": "320115", "name": "江宁区" }, + { "code": "320116", "name": "六合区" }, + { "code": "320117", "name": "溧水区" }, + { "code": "320118", "name": "高淳区" } + ] + }, + { + "code": "3202", + "name": "无锡市", + "children": [ + { "code": "320205", "name": "锡山区" }, + { "code": "320206", "name": "惠山区" }, + { "code": "320211", "name": "滨湖区" }, + { "code": "320213", "name": "梁溪区" }, + { "code": "320214", "name": "新吴区" }, + { "code": "320281", "name": "江阴市" }, + { "code": "320282", "name": "宜兴市" } + ] + }, + { + "code": "3203", + "name": "徐州市", + "children": [ + { "code": "320302", "name": "鼓楼区" }, + { "code": "320303", "name": "云龙区" }, + { "code": "320305", "name": "贾汪区" }, + { "code": "320311", "name": "泉山区" }, + { "code": "320312", "name": "铜山区" }, + { "code": "320321", "name": "丰县" }, + { "code": "320322", "name": "沛县" }, + { "code": "320324", "name": "睢宁县" }, + { "code": "320371", "name": "徐州经济技术开发区" }, + { "code": "320381", "name": "新沂市" }, + { "code": "320382", "name": "邳州市" } + ] + }, + { + "code": "3204", + "name": "常州市", + "children": [ + { "code": "320402", "name": "天宁区" }, + { "code": "320404", "name": "钟楼区" }, + { "code": "320411", "name": "新北区" }, + { "code": "320412", "name": "武进区" }, + { "code": "320413", "name": "金坛区" }, + { "code": "320481", "name": "溧阳市" } + ] + }, + { + "code": "3205", + "name": "苏州市", + "children": [ + { "code": "320505", "name": "虎丘区" }, + { "code": "320506", "name": "吴中区" }, + { "code": "320507", "name": "相城区" }, + { "code": "320508", "name": "姑苏区" }, + { "code": "320509", "name": "吴江区" }, + { "code": "320571", "name": "苏州工业园区" }, + { "code": "320581", "name": "常熟市" }, + { "code": "320582", "name": "张家港市" }, + { "code": "320583", "name": "昆山市" }, + { "code": "320585", "name": "太仓市" } + ] + }, + { + "code": "3206", + "name": "南通市", + "children": [ + { "code": "320612", "name": "通州区" }, + { "code": "320613", "name": "崇川区" }, + { "code": "320614", "name": "海门区" }, + { "code": "320623", "name": "如东县" }, + { "code": "320671", "name": "南通经济技术开发区" }, + { "code": "320681", "name": "启东市" }, + { "code": "320682", "name": "如皋市" }, + { "code": "320685", "name": "海安市" } + ] + }, + { + "code": "3207", + "name": "连云港市", + "children": [ + { "code": "320703", "name": "连云区" }, + { "code": "320706", "name": "海州区" }, + { "code": "320707", "name": "赣榆区" }, + { "code": "320722", "name": "东海县" }, + { "code": "320723", "name": "灌云县" }, + { "code": "320724", "name": "灌南县" }, + { "code": "320771", "name": "连云港经济技术开发区" }, + { "code": "320772", "name": "连云港高新技术产业开发区" } + ] + }, + { + "code": "3208", + "name": "淮安市", + "children": [ + { "code": "320803", "name": "淮安区" }, + { "code": "320804", "name": "淮阴区" }, + { "code": "320812", "name": "清江浦区" }, + { "code": "320813", "name": "洪泽区" }, + { "code": "320826", "name": "涟水县" }, + { "code": "320830", "name": "盱眙县" }, + { "code": "320831", "name": "金湖县" }, + { "code": "320871", "name": "淮安经济技术开发区" } + ] + }, + { + "code": "3209", + "name": "盐城市", + "children": [ + { "code": "320902", "name": "亭湖区" }, + { "code": "320903", "name": "盐都区" }, + { "code": "320904", "name": "大丰区" }, + { "code": "320921", "name": "响水县" }, + { "code": "320922", "name": "滨海县" }, + { "code": "320923", "name": "阜宁县" }, + { "code": "320924", "name": "射阳县" }, + { "code": "320925", "name": "建湖县" }, + { "code": "320971", "name": "盐城经济技术开发区" }, + { "code": "320981", "name": "东台市" } + ] + }, + { + "code": "3210", + "name": "扬州市", + "children": [ + { "code": "321002", "name": "广陵区" }, + { "code": "321003", "name": "邗江区" }, + { "code": "321012", "name": "江都区" }, + { "code": "321023", "name": "宝应县" }, + { "code": "321071", "name": "扬州经济技术开发区" }, + { "code": "321081", "name": "仪征市" }, + { "code": "321084", "name": "高邮市" } + ] + }, + { + "code": "3211", + "name": "镇江市", + "children": [ + { "code": "321102", "name": "京口区" }, + { "code": "321111", "name": "润州区" }, + { "code": "321112", "name": "丹徒区" }, + { "code": "321171", "name": "镇江新区" }, + { "code": "321181", "name": "丹阳市" }, + { "code": "321182", "name": "扬中市" }, + { "code": "321183", "name": "句容市" } + ] + }, + { + "code": "3212", + "name": "泰州市", + "children": [ + { "code": "321202", "name": "海陵区" }, + { "code": "321203", "name": "高港区" }, + { "code": "321204", "name": "姜堰区" }, + { "code": "321271", "name": "泰州医药高新技术产业开发区" }, + { "code": "321281", "name": "兴化市" }, + { "code": "321282", "name": "靖江市" }, + { "code": "321283", "name": "泰兴市" } + ] + }, + { + "code": "3213", + "name": "宿迁市", + "children": [ + { "code": "321302", "name": "宿城区" }, + { "code": "321311", "name": "宿豫区" }, + { "code": "321322", "name": "沭阳县" }, + { "code": "321323", "name": "泗阳县" }, + { "code": "321324", "name": "泗洪县" }, + { "code": "321371", "name": "宿迁经济技术开发区" } + ] + } + ] + }, + { + "code": "33", + "name": "浙江省", + "children": [ + { + "code": "3301", + "name": "杭州市", + "children": [ + { "code": "330102", "name": "上城区" }, + { "code": "330105", "name": "拱墅区" }, + { "code": "330106", "name": "西湖区" }, + { "code": "330108", "name": "滨江区" }, + { "code": "330109", "name": "萧山区" }, + { "code": "330110", "name": "余杭区" }, + { "code": "330111", "name": "富阳区" }, + { "code": "330112", "name": "临安区" }, + { "code": "330113", "name": "临平区" }, + { "code": "330114", "name": "钱塘区" }, + { "code": "330122", "name": "桐庐县" }, + { "code": "330127", "name": "淳安县" }, + { "code": "330182", "name": "建德市" } + ] + }, + { + "code": "3302", + "name": "宁波市", + "children": [ + { "code": "330203", "name": "海曙区" }, + { "code": "330205", "name": "江北区" }, + { "code": "330206", "name": "北仑区" }, + { "code": "330211", "name": "镇海区" }, + { "code": "330212", "name": "鄞州区" }, + { "code": "330213", "name": "奉化区" }, + { "code": "330225", "name": "象山县" }, + { "code": "330226", "name": "宁海县" }, + { "code": "330281", "name": "余姚市" }, + { "code": "330282", "name": "慈溪市" } + ] + }, + { + "code": "3303", + "name": "温州市", + "children": [ + { "code": "330302", "name": "鹿城区" }, + { "code": "330303", "name": "龙湾区" }, + { "code": "330304", "name": "瓯海区" }, + { "code": "330305", "name": "洞头区" }, + { "code": "330324", "name": "永嘉县" }, + { "code": "330326", "name": "平阳县" }, + { "code": "330327", "name": "苍南县" }, + { "code": "330328", "name": "文成县" }, + { "code": "330329", "name": "泰顺县" }, + { "code": "330381", "name": "瑞安市" }, + { "code": "330382", "name": "乐清市" }, + { "code": "330383", "name": "龙港市" } + ] + }, + { + "code": "3304", + "name": "嘉兴市", + "children": [ + { "code": "330402", "name": "南湖区" }, + { "code": "330411", "name": "秀洲区" }, + { "code": "330421", "name": "嘉善县" }, + { "code": "330424", "name": "海盐县" }, + { "code": "330481", "name": "海宁市" }, + { "code": "330482", "name": "平湖市" }, + { "code": "330483", "name": "桐乡市" } + ] + }, + { + "code": "3305", + "name": "湖州市", + "children": [ + { "code": "330502", "name": "吴兴区" }, + { "code": "330503", "name": "南浔区" }, + { "code": "330521", "name": "德清县" }, + { "code": "330522", "name": "长兴县" }, + { "code": "330523", "name": "安吉县" } + ] + }, + { + "code": "3306", + "name": "绍兴市", + "children": [ + { "code": "330602", "name": "越城区" }, + { "code": "330603", "name": "柯桥区" }, + { "code": "330604", "name": "上虞区" }, + { "code": "330624", "name": "新昌县" }, + { "code": "330681", "name": "诸暨市" }, + { "code": "330683", "name": "嵊州市" } + ] + }, + { + "code": "3307", + "name": "金华市", + "children": [ + { "code": "330702", "name": "婺城区" }, + { "code": "330703", "name": "金东区" }, + { "code": "330723", "name": "武义县" }, + { "code": "330726", "name": "浦江县" }, + { "code": "330727", "name": "磐安县" }, + { "code": "330781", "name": "兰溪市" }, + { "code": "330782", "name": "义乌市" }, + { "code": "330783", "name": "东阳市" }, + { "code": "330784", "name": "永康市" } + ] + }, + { + "code": "3308", + "name": "衢州市", + "children": [ + { "code": "330802", "name": "柯城区" }, + { "code": "330803", "name": "衢江区" }, + { "code": "330822", "name": "常山县" }, + { "code": "330824", "name": "开化县" }, + { "code": "330825", "name": "龙游县" }, + { "code": "330881", "name": "江山市" } + ] + }, + { + "code": "3309", + "name": "舟山市", + "children": [ + { "code": "330902", "name": "定海区" }, + { "code": "330903", "name": "普陀区" }, + { "code": "330921", "name": "岱山县" }, + { "code": "330922", "name": "嵊泗县" } + ] + }, + { + "code": "3310", + "name": "台州市", + "children": [ + { "code": "331002", "name": "椒江区" }, + { "code": "331003", "name": "黄岩区" }, + { "code": "331004", "name": "路桥区" }, + { "code": "331022", "name": "三门县" }, + { "code": "331023", "name": "天台县" }, + { "code": "331024", "name": "仙居县" }, + { "code": "331081", "name": "温岭市" }, + { "code": "331082", "name": "临海市" }, + { "code": "331083", "name": "玉环市" } + ] + }, + { + "code": "3311", + "name": "丽水市", + "children": [ + { "code": "331102", "name": "莲都区" }, + { "code": "331121", "name": "青田县" }, + { "code": "331122", "name": "缙云县" }, + { "code": "331123", "name": "遂昌县" }, + { "code": "331124", "name": "松阳县" }, + { "code": "331125", "name": "云和县" }, + { "code": "331126", "name": "庆元县" }, + { "code": "331127", "name": "景宁畲族自治县" }, + { "code": "331181", "name": "龙泉市" } + ] + } + ] + }, + { + "code": "34", + "name": "安徽省", + "children": [ + { + "code": "3401", + "name": "合肥市", + "children": [ + { "code": "340102", "name": "瑶海区" }, + { "code": "340103", "name": "庐阳区" }, + { "code": "340104", "name": "蜀山区" }, + { "code": "340111", "name": "包河区" }, + { "code": "340121", "name": "长丰县" }, + { "code": "340122", "name": "肥东县" }, + { "code": "340123", "name": "肥西县" }, + { "code": "340124", "name": "庐江县" }, + { "code": "340171", "name": "合肥高新技术产业开发区" }, + { "code": "340172", "name": "合肥经济技术开发区" }, + { "code": "340173", "name": "合肥新站高新技术产业开发区" }, + { "code": "340181", "name": "巢湖市" } + ] + }, + { + "code": "3402", + "name": "芜湖市", + "children": [ + { "code": "340202", "name": "镜湖区" }, + { "code": "340207", "name": "鸠江区" }, + { "code": "340209", "name": "弋江区" }, + { "code": "340210", "name": "湾沚区" }, + { "code": "340212", "name": "繁昌区" }, + { "code": "340223", "name": "南陵县" }, + { "code": "340271", "name": "芜湖经济技术开发区" }, + { "code": "340272", "name": "安徽芜湖三山经济开发区" }, + { "code": "340281", "name": "无为市" } + ] + }, + { + "code": "3403", + "name": "蚌埠市", + "children": [ + { "code": "340302", "name": "龙子湖区" }, + { "code": "340303", "name": "蚌山区" }, + { "code": "340304", "name": "禹会区" }, + { "code": "340311", "name": "淮上区" }, + { "code": "340321", "name": "怀远县" }, + { "code": "340322", "name": "五河县" }, + { "code": "340323", "name": "固镇县" }, + { "code": "340371", "name": "蚌埠市高新技术开发区" }, + { "code": "340372", "name": "蚌埠市经济开发区" } + ] + }, + { + "code": "3404", + "name": "淮南市", + "children": [ + { "code": "340402", "name": "大通区" }, + { "code": "340403", "name": "田家庵区" }, + { "code": "340404", "name": "谢家集区" }, + { "code": "340405", "name": "八公山区" }, + { "code": "340406", "name": "潘集区" }, + { "code": "340421", "name": "凤台县" }, + { "code": "340422", "name": "寿县" } + ] + }, + { + "code": "3405", + "name": "马鞍山市", + "children": [ + { "code": "340503", "name": "花山区" }, + { "code": "340504", "name": "雨山区" }, + { "code": "340506", "name": "博望区" }, + { "code": "340521", "name": "当涂县" }, + { "code": "340522", "name": "含山县" }, + { "code": "340523", "name": "和县" } + ] + }, + { + "code": "3406", + "name": "淮北市", + "children": [ + { "code": "340602", "name": "杜集区" }, + { "code": "340603", "name": "相山区" }, + { "code": "340604", "name": "烈山区" }, + { "code": "340621", "name": "濉溪县" } + ] + }, + { + "code": "3407", + "name": "铜陵市", + "children": [ + { "code": "340705", "name": "铜官区" }, + { "code": "340706", "name": "义安区" }, + { "code": "340711", "name": "郊区" }, + { "code": "340722", "name": "枞阳县" } + ] + }, + { + "code": "3408", + "name": "安庆市", + "children": [ + { "code": "340802", "name": "迎江区" }, + { "code": "340803", "name": "大观区" }, + { "code": "340811", "name": "宜秀区" }, + { "code": "340822", "name": "怀宁县" }, + { "code": "340825", "name": "太湖县" }, + { "code": "340826", "name": "宿松县" }, + { "code": "340827", "name": "望江县" }, + { "code": "340828", "name": "岳西县" }, + { "code": "340871", "name": "安徽安庆经济开发区" }, + { "code": "340881", "name": "桐城市" }, + { "code": "340882", "name": "潜山市" } + ] + }, + { + "code": "3410", + "name": "黄山市", + "children": [ + { "code": "341002", "name": "屯溪区" }, + { "code": "341003", "name": "黄山区" }, + { "code": "341004", "name": "徽州区" }, + { "code": "341021", "name": "歙县" }, + { "code": "341022", "name": "休宁县" }, + { "code": "341023", "name": "黟县" }, + { "code": "341024", "name": "祁门县" } + ] + }, + { + "code": "3411", + "name": "滁州市", + "children": [ + { "code": "341102", "name": "琅琊区" }, + { "code": "341103", "name": "南谯区" }, + { "code": "341122", "name": "来安县" }, + { "code": "341124", "name": "全椒县" }, + { "code": "341125", "name": "定远县" }, + { "code": "341126", "name": "凤阳县" }, + { "code": "341171", "name": "中新苏滁高新技术产业开发区" }, + { "code": "341172", "name": "滁州经济技术开发区" }, + { "code": "341181", "name": "天长市" }, + { "code": "341182", "name": "明光市" } + ] + }, + { + "code": "3412", + "name": "阜阳市", + "children": [ + { "code": "341202", "name": "颍州区" }, + { "code": "341203", "name": "颍东区" }, + { "code": "341204", "name": "颍泉区" }, + { "code": "341221", "name": "临泉县" }, + { "code": "341222", "name": "太和县" }, + { "code": "341225", "name": "阜南县" }, + { "code": "341226", "name": "颍上县" }, + { "code": "341271", "name": "阜阳合肥现代产业园区" }, + { "code": "341272", "name": "阜阳经济技术开发区" }, + { "code": "341282", "name": "界首市" } + ] + }, + { + "code": "3413", + "name": "宿州市", + "children": [ + { "code": "341302", "name": "埇桥区" }, + { "code": "341321", "name": "砀山县" }, + { "code": "341322", "name": "萧县" }, + { "code": "341323", "name": "灵璧县" }, + { "code": "341324", "name": "泗县" }, + { "code": "341371", "name": "宿州马鞍山现代产业园区" }, + { "code": "341372", "name": "宿州经济技术开发区" } + ] + }, + { + "code": "3415", + "name": "六安市", + "children": [ + { "code": "341502", "name": "金安区" }, + { "code": "341503", "name": "裕安区" }, + { "code": "341504", "name": "叶集区" }, + { "code": "341522", "name": "霍邱县" }, + { "code": "341523", "name": "舒城县" }, + { "code": "341524", "name": "金寨县" }, + { "code": "341525", "name": "霍山县" } + ] + }, + { + "code": "3416", + "name": "亳州市", + "children": [ + { "code": "341602", "name": "谯城区" }, + { "code": "341621", "name": "涡阳县" }, + { "code": "341622", "name": "蒙城县" }, + { "code": "341623", "name": "利辛县" } + ] + }, + { + "code": "3417", + "name": "池州市", + "children": [ + { "code": "341702", "name": "贵池区" }, + { "code": "341721", "name": "东至县" }, + { "code": "341722", "name": "石台县" }, + { "code": "341723", "name": "青阳县" } + ] + }, + { + "code": "3418", + "name": "宣城市", + "children": [ + { "code": "341802", "name": "宣州区" }, + { "code": "341821", "name": "郎溪县" }, + { "code": "341823", "name": "泾县" }, + { "code": "341824", "name": "绩溪县" }, + { "code": "341825", "name": "旌德县" }, + { "code": "341871", "name": "宣城市经济开发区" }, + { "code": "341881", "name": "宁国市" }, + { "code": "341882", "name": "广德市" } + ] + } + ] + }, + { + "code": "35", + "name": "福建省", + "children": [ + { + "code": "3501", + "name": "福州市", + "children": [ + { "code": "350102", "name": "鼓楼区" }, + { "code": "350103", "name": "台江区" }, + { "code": "350104", "name": "仓山区" }, + { "code": "350105", "name": "马尾区" }, + { "code": "350111", "name": "晋安区" }, + { "code": "350112", "name": "长乐区" }, + { "code": "350121", "name": "闽侯县" }, + { "code": "350122", "name": "连江县" }, + { "code": "350123", "name": "罗源县" }, + { "code": "350124", "name": "闽清县" }, + { "code": "350125", "name": "永泰县" }, + { "code": "350128", "name": "平潭县" }, + { "code": "350181", "name": "福清市" } + ] + }, + { + "code": "3502", + "name": "厦门市", + "children": [ + { "code": "350203", "name": "思明区" }, + { "code": "350205", "name": "海沧区" }, + { "code": "350206", "name": "湖里区" }, + { "code": "350211", "name": "集美区" }, + { "code": "350212", "name": "同安区" }, + { "code": "350213", "name": "翔安区" } + ] + }, + { + "code": "3503", + "name": "莆田市", + "children": [ + { "code": "350302", "name": "城厢区" }, + { "code": "350303", "name": "涵江区" }, + { "code": "350304", "name": "荔城区" }, + { "code": "350305", "name": "秀屿区" }, + { "code": "350322", "name": "仙游县" } + ] + }, + { + "code": "3504", + "name": "三明市", + "children": [ + { "code": "350404", "name": "三元区" }, + { "code": "350405", "name": "沙县区" }, + { "code": "350421", "name": "明溪县" }, + { "code": "350423", "name": "清流县" }, + { "code": "350424", "name": "宁化县" }, + { "code": "350425", "name": "大田县" }, + { "code": "350426", "name": "尤溪县" }, + { "code": "350428", "name": "将乐县" }, + { "code": "350429", "name": "泰宁县" }, + { "code": "350430", "name": "建宁县" }, + { "code": "350481", "name": "永安市" } + ] + }, + { + "code": "3505", + "name": "泉州市", + "children": [ + { "code": "350502", "name": "鲤城区" }, + { "code": "350503", "name": "丰泽区" }, + { "code": "350504", "name": "洛江区" }, + { "code": "350505", "name": "泉港区" }, + { "code": "350521", "name": "惠安县" }, + { "code": "350524", "name": "安溪县" }, + { "code": "350525", "name": "永春县" }, + { "code": "350526", "name": "德化县" }, + { "code": "350527", "name": "金门县" }, + { "code": "350581", "name": "石狮市" }, + { "code": "350582", "name": "晋江市" }, + { "code": "350583", "name": "南安市" } + ] + }, + { + "code": "3506", + "name": "漳州市", + "children": [ + { "code": "350602", "name": "芗城区" }, + { "code": "350603", "name": "龙文区" }, + { "code": "350604", "name": "龙海区" }, + { "code": "350605", "name": "长泰区" }, + { "code": "350622", "name": "云霄县" }, + { "code": "350623", "name": "漳浦县" }, + { "code": "350624", "name": "诏安县" }, + { "code": "350626", "name": "东山县" }, + { "code": "350627", "name": "南靖县" }, + { "code": "350628", "name": "平和县" }, + { "code": "350629", "name": "华安县" } + ] + }, + { + "code": "3507", + "name": "南平市", + "children": [ + { "code": "350702", "name": "延平区" }, + { "code": "350703", "name": "建阳区" }, + { "code": "350721", "name": "顺昌县" }, + { "code": "350722", "name": "浦城县" }, + { "code": "350723", "name": "光泽县" }, + { "code": "350724", "name": "松溪县" }, + { "code": "350725", "name": "政和县" }, + { "code": "350781", "name": "邵武市" }, + { "code": "350782", "name": "武夷山市" }, + { "code": "350783", "name": "建瓯市" } + ] + }, + { + "code": "3508", + "name": "龙岩市", + "children": [ + { "code": "350802", "name": "新罗区" }, + { "code": "350803", "name": "永定区" }, + { "code": "350821", "name": "长汀县" }, + { "code": "350823", "name": "上杭县" }, + { "code": "350824", "name": "武平县" }, + { "code": "350825", "name": "连城县" }, + { "code": "350881", "name": "漳平市" } + ] + }, + { + "code": "3509", + "name": "宁德市", + "children": [ + { "code": "350902", "name": "蕉城区" }, + { "code": "350921", "name": "霞浦县" }, + { "code": "350922", "name": "古田县" }, + { "code": "350923", "name": "屏南县" }, + { "code": "350924", "name": "寿宁县" }, + { "code": "350925", "name": "周宁县" }, + { "code": "350926", "name": "柘荣县" }, + { "code": "350981", "name": "福安市" }, + { "code": "350982", "name": "福鼎市" } + ] + } + ] + }, + { + "code": "36", + "name": "江西省", + "children": [ + { + "code": "3601", + "name": "南昌市", + "children": [ + { "code": "360102", "name": "东湖区" }, + { "code": "360103", "name": "西湖区" }, + { "code": "360104", "name": "青云谱区" }, + { "code": "360111", "name": "青山湖区" }, + { "code": "360112", "name": "新建区" }, + { "code": "360113", "name": "红谷滩区" }, + { "code": "360121", "name": "南昌县" }, + { "code": "360123", "name": "安义县" }, + { "code": "360124", "name": "进贤县" } + ] + }, + { + "code": "3602", + "name": "景德镇市", + "children": [ + { "code": "360202", "name": "昌江区" }, + { "code": "360203", "name": "珠山区" }, + { "code": "360222", "name": "浮梁县" }, + { "code": "360281", "name": "乐平市" } + ] + }, + { + "code": "3603", + "name": "萍乡市", + "children": [ + { "code": "360302", "name": "安源区" }, + { "code": "360313", "name": "湘东区" }, + { "code": "360321", "name": "莲花县" }, + { "code": "360322", "name": "上栗县" }, + { "code": "360323", "name": "芦溪县" } + ] + }, + { + "code": "3604", + "name": "九江市", + "children": [ + { "code": "360402", "name": "濂溪区" }, + { "code": "360403", "name": "浔阳区" }, + { "code": "360404", "name": "柴桑区" }, + { "code": "360423", "name": "武宁县" }, + { "code": "360424", "name": "修水县" }, + { "code": "360425", "name": "永修县" }, + { "code": "360426", "name": "德安县" }, + { "code": "360428", "name": "都昌县" }, + { "code": "360429", "name": "湖口县" }, + { "code": "360430", "name": "彭泽县" }, + { "code": "360481", "name": "瑞昌市" }, + { "code": "360482", "name": "共青城市" }, + { "code": "360483", "name": "庐山市" } + ] + }, + { + "code": "3605", + "name": "新余市", + "children": [ + { "code": "360502", "name": "渝水区" }, + { "code": "360521", "name": "分宜县" } + ] + }, + { + "code": "3606", + "name": "鹰潭市", + "children": [ + { "code": "360602", "name": "月湖区" }, + { "code": "360603", "name": "余江区" }, + { "code": "360681", "name": "贵溪市" } + ] + }, + { + "code": "3607", + "name": "赣州市", + "children": [ + { "code": "360702", "name": "章贡区" }, + { "code": "360703", "name": "南康区" }, + { "code": "360704", "name": "赣县区" }, + { "code": "360722", "name": "信丰县" }, + { "code": "360723", "name": "大余县" }, + { "code": "360724", "name": "上犹县" }, + { "code": "360725", "name": "崇义县" }, + { "code": "360726", "name": "安远县" }, + { "code": "360728", "name": "定南县" }, + { "code": "360729", "name": "全南县" }, + { "code": "360730", "name": "宁都县" }, + { "code": "360731", "name": "于都县" }, + { "code": "360732", "name": "兴国县" }, + { "code": "360733", "name": "会昌县" }, + { "code": "360734", "name": "寻乌县" }, + { "code": "360735", "name": "石城县" }, + { "code": "360781", "name": "瑞金市" }, + { "code": "360783", "name": "龙南市" } + ] + }, + { + "code": "3608", + "name": "吉安市", + "children": [ + { "code": "360802", "name": "吉州区" }, + { "code": "360803", "name": "青原区" }, + { "code": "360821", "name": "吉安县" }, + { "code": "360822", "name": "吉水县" }, + { "code": "360823", "name": "峡江县" }, + { "code": "360824", "name": "新干县" }, + { "code": "360825", "name": "永丰县" }, + { "code": "360826", "name": "泰和县" }, + { "code": "360827", "name": "遂川县" }, + { "code": "360828", "name": "万安县" }, + { "code": "360829", "name": "安福县" }, + { "code": "360830", "name": "永新县" }, + { "code": "360881", "name": "井冈山市" } + ] + }, + { + "code": "3609", + "name": "宜春市", + "children": [ + { "code": "360902", "name": "袁州区" }, + { "code": "360921", "name": "奉新县" }, + { "code": "360922", "name": "万载县" }, + { "code": "360923", "name": "上高县" }, + { "code": "360924", "name": "宜丰县" }, + { "code": "360925", "name": "靖安县" }, + { "code": "360926", "name": "铜鼓县" }, + { "code": "360981", "name": "丰城市" }, + { "code": "360982", "name": "樟树市" }, + { "code": "360983", "name": "高安市" } + ] + }, + { + "code": "3610", + "name": "抚州市", + "children": [ + { "code": "361002", "name": "临川区" }, + { "code": "361003", "name": "东乡区" }, + { "code": "361021", "name": "南城县" }, + { "code": "361022", "name": "黎川县" }, + { "code": "361023", "name": "南丰县" }, + { "code": "361024", "name": "崇仁县" }, + { "code": "361025", "name": "乐安县" }, + { "code": "361026", "name": "宜黄县" }, + { "code": "361027", "name": "金溪县" }, + { "code": "361028", "name": "资溪县" }, + { "code": "361030", "name": "广昌县" } + ] + }, + { + "code": "3611", + "name": "上饶市", + "children": [ + { "code": "361102", "name": "信州区" }, + { "code": "361103", "name": "广丰区" }, + { "code": "361104", "name": "广信区" }, + { "code": "361123", "name": "玉山县" }, + { "code": "361124", "name": "铅山县" }, + { "code": "361125", "name": "横峰县" }, + { "code": "361126", "name": "弋阳县" }, + { "code": "361127", "name": "余干县" }, + { "code": "361128", "name": "鄱阳县" }, + { "code": "361129", "name": "万年县" }, + { "code": "361130", "name": "婺源县" }, + { "code": "361181", "name": "德兴市" } + ] + } + ] + }, + { + "code": "37", + "name": "山东省", + "children": [ + { + "code": "3701", + "name": "济南市", + "children": [ + { "code": "370102", "name": "历下区" }, + { "code": "370103", "name": "市中区" }, + { "code": "370104", "name": "槐荫区" }, + { "code": "370105", "name": "天桥区" }, + { "code": "370112", "name": "历城区" }, + { "code": "370113", "name": "长清区" }, + { "code": "370114", "name": "章丘区" }, + { "code": "370115", "name": "济阳区" }, + { "code": "370116", "name": "莱芜区" }, + { "code": "370117", "name": "钢城区" }, + { "code": "370124", "name": "平阴县" }, + { "code": "370126", "name": "商河县" }, + { "code": "370171", "name": "济南高新技术产业开发区" } + ] + }, + { + "code": "3702", + "name": "青岛市", + "children": [ + { "code": "370202", "name": "市南区" }, + { "code": "370203", "name": "市北区" }, + { "code": "370211", "name": "黄岛区" }, + { "code": "370212", "name": "崂山区" }, + { "code": "370213", "name": "李沧区" }, + { "code": "370214", "name": "城阳区" }, + { "code": "370215", "name": "即墨区" }, + { "code": "370271", "name": "青岛高新技术产业开发区" }, + { "code": "370281", "name": "胶州市" }, + { "code": "370283", "name": "平度市" }, + { "code": "370285", "name": "莱西市" } + ] + }, + { + "code": "3703", + "name": "淄博市", + "children": [ + { "code": "370302", "name": "淄川区" }, + { "code": "370303", "name": "张店区" }, + { "code": "370304", "name": "博山区" }, + { "code": "370305", "name": "临淄区" }, + { "code": "370306", "name": "周村区" }, + { "code": "370321", "name": "桓台县" }, + { "code": "370322", "name": "高青县" }, + { "code": "370323", "name": "沂源县" } + ] + }, + { + "code": "3704", + "name": "枣庄市", + "children": [ + { "code": "370402", "name": "市中区" }, + { "code": "370403", "name": "薛城区" }, + { "code": "370404", "name": "峄城区" }, + { "code": "370405", "name": "台儿庄区" }, + { "code": "370406", "name": "山亭区" }, + { "code": "370481", "name": "滕州市" } + ] + }, + { + "code": "3705", + "name": "东营市", + "children": [ + { "code": "370502", "name": "东营区" }, + { "code": "370503", "name": "河口区" }, + { "code": "370505", "name": "垦利区" }, + { "code": "370522", "name": "利津县" }, + { "code": "370523", "name": "广饶县" }, + { "code": "370571", "name": "东营经济技术开发区" }, + { "code": "370572", "name": "东营港经济开发区" } + ] + }, + { + "code": "3706", + "name": "烟台市", + "children": [ + { "code": "370602", "name": "芝罘区" }, + { "code": "370611", "name": "福山区" }, + { "code": "370612", "name": "牟平区" }, + { "code": "370613", "name": "莱山区" }, + { "code": "370614", "name": "蓬莱区" }, + { "code": "370671", "name": "烟台高新技术产业开发区" }, + { "code": "370672", "name": "烟台经济技术开发区" }, + { "code": "370681", "name": "龙口市" }, + { "code": "370682", "name": "莱阳市" }, + { "code": "370683", "name": "莱州市" }, + { "code": "370685", "name": "招远市" }, + { "code": "370686", "name": "栖霞市" }, + { "code": "370687", "name": "海阳市" } + ] + }, + { + "code": "3707", + "name": "潍坊市", + "children": [ + { "code": "370702", "name": "潍城区" }, + { "code": "370703", "name": "寒亭区" }, + { "code": "370704", "name": "坊子区" }, + { "code": "370705", "name": "奎文区" }, + { "code": "370724", "name": "临朐县" }, + { "code": "370725", "name": "昌乐县" }, + { "code": "370772", "name": "潍坊滨海经济技术开发区" }, + { "code": "370781", "name": "青州市" }, + { "code": "370782", "name": "诸城市" }, + { "code": "370783", "name": "寿光市" }, + { "code": "370784", "name": "安丘市" }, + { "code": "370785", "name": "高密市" }, + { "code": "370786", "name": "昌邑市" } + ] + }, + { + "code": "3708", + "name": "济宁市", + "children": [ + { "code": "370811", "name": "任城区" }, + { "code": "370812", "name": "兖州区" }, + { "code": "370826", "name": "微山县" }, + { "code": "370827", "name": "鱼台县" }, + { "code": "370828", "name": "金乡县" }, + { "code": "370829", "name": "嘉祥县" }, + { "code": "370830", "name": "汶上县" }, + { "code": "370831", "name": "泗水县" }, + { "code": "370832", "name": "梁山县" }, + { "code": "370871", "name": "济宁高新技术产业开发区" }, + { "code": "370881", "name": "曲阜市" }, + { "code": "370883", "name": "邹城市" } + ] + }, + { + "code": "3709", + "name": "泰安市", + "children": [ + { "code": "370902", "name": "泰山区" }, + { "code": "370911", "name": "岱岳区" }, + { "code": "370921", "name": "宁阳县" }, + { "code": "370923", "name": "东平县" }, + { "code": "370982", "name": "新泰市" }, + { "code": "370983", "name": "肥城市" } + ] + }, + { + "code": "3710", + "name": "威海市", + "children": [ + { "code": "371002", "name": "环翠区" }, + { "code": "371003", "name": "文登区" }, + { "code": "371071", "name": "威海火炬高技术产业开发区" }, + { "code": "371072", "name": "威海经济技术开发区" }, + { "code": "371073", "name": "威海临港经济技术开发区" }, + { "code": "371082", "name": "荣成市" }, + { "code": "371083", "name": "乳山市" } + ] + }, + { + "code": "3711", + "name": "日照市", + "children": [ + { "code": "371102", "name": "东港区" }, + { "code": "371103", "name": "岚山区" }, + { "code": "371121", "name": "五莲县" }, + { "code": "371122", "name": "莒县" }, + { "code": "371171", "name": "日照经济技术开发区" } + ] + }, + { + "code": "3713", + "name": "临沂市", + "children": [ + { "code": "371302", "name": "兰山区" }, + { "code": "371311", "name": "罗庄区" }, + { "code": "371312", "name": "河东区" }, + { "code": "371321", "name": "沂南县" }, + { "code": "371322", "name": "郯城县" }, + { "code": "371323", "name": "沂水县" }, + { "code": "371324", "name": "兰陵县" }, + { "code": "371325", "name": "费县" }, + { "code": "371326", "name": "平邑县" }, + { "code": "371327", "name": "莒南县" }, + { "code": "371328", "name": "蒙阴县" }, + { "code": "371329", "name": "临沭县" }, + { "code": "371371", "name": "临沂高新技术产业开发区" } + ] + }, + { + "code": "3714", + "name": "德州市", + "children": [ + { "code": "371402", "name": "德城区" }, + { "code": "371403", "name": "陵城区" }, + { "code": "371422", "name": "宁津县" }, + { "code": "371423", "name": "庆云县" }, + { "code": "371424", "name": "临邑县" }, + { "code": "371425", "name": "齐河县" }, + { "code": "371426", "name": "平原县" }, + { "code": "371427", "name": "夏津县" }, + { "code": "371428", "name": "武城县" }, + { "code": "371471", "name": "德州天衢新区" }, + { "code": "371481", "name": "乐陵市" }, + { "code": "371482", "name": "禹城市" } + ] + }, + { + "code": "3715", + "name": "聊城市", + "children": [ + { "code": "371502", "name": "东昌府区" }, + { "code": "371503", "name": "茌平区" }, + { "code": "371521", "name": "阳谷县" }, + { "code": "371522", "name": "莘县" }, + { "code": "371524", "name": "东阿县" }, + { "code": "371525", "name": "冠县" }, + { "code": "371526", "name": "高唐县" }, + { "code": "371581", "name": "临清市" } + ] + }, + { + "code": "3716", + "name": "滨州市", + "children": [ + { "code": "371602", "name": "滨城区" }, + { "code": "371603", "name": "沾化区" }, + { "code": "371621", "name": "惠民县" }, + { "code": "371622", "name": "阳信县" }, + { "code": "371623", "name": "无棣县" }, + { "code": "371625", "name": "博兴县" }, + { "code": "371681", "name": "邹平市" } + ] + }, + { + "code": "3717", + "name": "菏泽市", + "children": [ + { "code": "371702", "name": "牡丹区" }, + { "code": "371703", "name": "定陶区" }, + { "code": "371721", "name": "曹县" }, + { "code": "371722", "name": "单县" }, + { "code": "371723", "name": "成武县" }, + { "code": "371724", "name": "巨野县" }, + { "code": "371725", "name": "郓城县" }, + { "code": "371726", "name": "鄄城县" }, + { "code": "371728", "name": "东明县" }, + { "code": "371771", "name": "菏泽经济技术开发区" }, + { "code": "371772", "name": "菏泽高新技术开发区" } + ] + } + ] + }, + { + "code": "41", + "name": "河南省", + "children": [ + { + "code": "4101", + "name": "郑州市", + "children": [ + { "code": "410102", "name": "中原区" }, + { "code": "410103", "name": "二七区" }, + { "code": "410104", "name": "管城回族区" }, + { "code": "410105", "name": "金水区" }, + { "code": "410106", "name": "上街区" }, + { "code": "410108", "name": "惠济区" }, + { "code": "410122", "name": "中牟县" }, + { "code": "410171", "name": "郑州经济技术开发区" }, + { "code": "410172", "name": "郑州高新技术产业开发区" }, + { "code": "410173", "name": "郑州航空港经济综合实验区" }, + { "code": "410181", "name": "巩义市" }, + { "code": "410182", "name": "荥阳市" }, + { "code": "410183", "name": "新密市" }, + { "code": "410184", "name": "新郑市" }, + { "code": "410185", "name": "登封市" } + ] + }, + { + "code": "4102", + "name": "开封市", + "children": [ + { "code": "410202", "name": "龙亭区" }, + { "code": "410203", "name": "顺河回族区" }, + { "code": "410204", "name": "鼓楼区" }, + { "code": "410205", "name": "禹王台区" }, + { "code": "410212", "name": "祥符区" }, + { "code": "410221", "name": "杞县" }, + { "code": "410222", "name": "通许县" }, + { "code": "410223", "name": "尉氏县" }, + { "code": "410225", "name": "兰考县" } + ] + }, + { + "code": "4103", + "name": "洛阳市", + "children": [ + { "code": "410302", "name": "老城区" }, + { "code": "410303", "name": "西工区" }, + { "code": "410304", "name": "瀍河回族区" }, + { "code": "410305", "name": "涧西区" }, + { "code": "410307", "name": "偃师区" }, + { "code": "410308", "name": "孟津区" }, + { "code": "410311", "name": "洛龙区" }, + { "code": "410323", "name": "新安县" }, + { "code": "410324", "name": "栾川县" }, + { "code": "410325", "name": "嵩县" }, + { "code": "410326", "name": "汝阳县" }, + { "code": "410327", "name": "宜阳县" }, + { "code": "410328", "name": "洛宁县" }, + { "code": "410329", "name": "伊川县" }, + { "code": "410371", "name": "洛阳高新技术产业开发区" } + ] + }, + { + "code": "4104", + "name": "平顶山市", + "children": [ + { "code": "410402", "name": "新华区" }, + { "code": "410403", "name": "卫东区" }, + { "code": "410404", "name": "石龙区" }, + { "code": "410411", "name": "湛河区" }, + { "code": "410421", "name": "宝丰县" }, + { "code": "410422", "name": "叶县" }, + { "code": "410423", "name": "鲁山县" }, + { "code": "410425", "name": "郏县" }, + { "code": "410471", "name": "平顶山高新技术产业开发区" }, + { "code": "410472", "name": "平顶山市城乡一体化示范区" }, + { "code": "410481", "name": "舞钢市" }, + { "code": "410482", "name": "汝州市" } + ] + }, + { + "code": "4105", + "name": "安阳市", + "children": [ + { "code": "410502", "name": "文峰区" }, + { "code": "410503", "name": "北关区" }, + { "code": "410505", "name": "殷都区" }, + { "code": "410506", "name": "龙安区" }, + { "code": "410522", "name": "安阳县" }, + { "code": "410523", "name": "汤阴县" }, + { "code": "410526", "name": "滑县" }, + { "code": "410527", "name": "内黄县" }, + { "code": "410571", "name": "安阳高新技术产业开发区" }, + { "code": "410581", "name": "林州市" } + ] + }, + { + "code": "4106", + "name": "鹤壁市", + "children": [ + { "code": "410602", "name": "鹤山区" }, + { "code": "410603", "name": "山城区" }, + { "code": "410611", "name": "淇滨区" }, + { "code": "410621", "name": "浚县" }, + { "code": "410622", "name": "淇县" }, + { "code": "410671", "name": "鹤壁经济技术开发区" } + ] + }, + { + "code": "4107", + "name": "新乡市", + "children": [ + { "code": "410702", "name": "红旗区" }, + { "code": "410703", "name": "卫滨区" }, + { "code": "410704", "name": "凤泉区" }, + { "code": "410711", "name": "牧野区" }, + { "code": "410721", "name": "新乡县" }, + { "code": "410724", "name": "获嘉县" }, + { "code": "410725", "name": "原阳县" }, + { "code": "410726", "name": "延津县" }, + { "code": "410727", "name": "封丘县" }, + { "code": "410771", "name": "新乡高新技术产业开发区" }, + { "code": "410772", "name": "新乡经济技术开发区" }, + { "code": "410773", "name": "新乡市平原城乡一体化示范区" }, + { "code": "410781", "name": "卫辉市" }, + { "code": "410782", "name": "辉县市" }, + { "code": "410783", "name": "长垣市" } + ] + }, + { + "code": "4108", + "name": "焦作市", + "children": [ + { "code": "410802", "name": "解放区" }, + { "code": "410803", "name": "中站区" }, + { "code": "410804", "name": "马村区" }, + { "code": "410811", "name": "山阳区" }, + { "code": "410821", "name": "修武县" }, + { "code": "410822", "name": "博爱县" }, + { "code": "410823", "name": "武陟县" }, + { "code": "410825", "name": "温县" }, + { "code": "410871", "name": "焦作城乡一体化示范区" }, + { "code": "410882", "name": "沁阳市" }, + { "code": "410883", "name": "孟州市" } + ] + }, + { + "code": "4109", + "name": "濮阳市", + "children": [ + { "code": "410902", "name": "华龙区" }, + { "code": "410922", "name": "清丰县" }, + { "code": "410923", "name": "南乐县" }, + { "code": "410926", "name": "范县" }, + { "code": "410927", "name": "台前县" }, + { "code": "410928", "name": "濮阳县" }, + { "code": "410971", "name": "河南濮阳工业园区" }, + { "code": "410972", "name": "濮阳经济技术开发区" } + ] + }, + { + "code": "4110", + "name": "许昌市", + "children": [ + { "code": "411002", "name": "魏都区" }, + { "code": "411003", "name": "建安区" }, + { "code": "411024", "name": "鄢陵县" }, + { "code": "411025", "name": "襄城县" }, + { "code": "411071", "name": "许昌经济技术开发区" }, + { "code": "411081", "name": "禹州市" }, + { "code": "411082", "name": "长葛市" } + ] + }, + { + "code": "4111", + "name": "漯河市", + "children": [ + { "code": "411102", "name": "源汇区" }, + { "code": "411103", "name": "郾城区" }, + { "code": "411104", "name": "召陵区" }, + { "code": "411121", "name": "舞阳县" }, + { "code": "411122", "name": "临颍县" }, + { "code": "411171", "name": "漯河经济技术开发区" } + ] + }, + { + "code": "4112", + "name": "三门峡市", + "children": [ + { "code": "411202", "name": "湖滨区" }, + { "code": "411203", "name": "陕州区" }, + { "code": "411221", "name": "渑池县" }, + { "code": "411224", "name": "卢氏县" }, + { "code": "411271", "name": "河南三门峡经济开发区" }, + { "code": "411281", "name": "义马市" }, + { "code": "411282", "name": "灵宝市" } + ] + }, + { + "code": "4113", + "name": "南阳市", + "children": [ + { "code": "411302", "name": "宛城区" }, + { "code": "411303", "name": "卧龙区" }, + { "code": "411321", "name": "南召县" }, + { "code": "411322", "name": "方城县" }, + { "code": "411323", "name": "西峡县" }, + { "code": "411324", "name": "镇平县" }, + { "code": "411325", "name": "内乡县" }, + { "code": "411326", "name": "淅川县" }, + { "code": "411327", "name": "社旗县" }, + { "code": "411328", "name": "唐河县" }, + { "code": "411329", "name": "新野县" }, + { "code": "411330", "name": "桐柏县" }, + { "code": "411371", "name": "南阳高新技术产业开发区" }, + { "code": "411372", "name": "南阳市城乡一体化示范区" }, + { "code": "411381", "name": "邓州市" } + ] + }, + { + "code": "4114", + "name": "商丘市", + "children": [ + { "code": "411402", "name": "梁园区" }, + { "code": "411403", "name": "睢阳区" }, + { "code": "411421", "name": "民权县" }, + { "code": "411422", "name": "睢县" }, + { "code": "411423", "name": "宁陵县" }, + { "code": "411424", "name": "柘城县" }, + { "code": "411425", "name": "虞城县" }, + { "code": "411426", "name": "夏邑县" }, + { "code": "411471", "name": "豫东综合物流产业聚集区" }, + { "code": "411472", "name": "河南商丘经济开发区" }, + { "code": "411481", "name": "永城市" } + ] + }, + { + "code": "4115", + "name": "信阳市", + "children": [ + { "code": "411502", "name": "浉河区" }, + { "code": "411503", "name": "平桥区" }, + { "code": "411521", "name": "罗山县" }, + { "code": "411522", "name": "光山县" }, + { "code": "411523", "name": "新县" }, + { "code": "411524", "name": "商城县" }, + { "code": "411525", "name": "固始县" }, + { "code": "411526", "name": "潢川县" }, + { "code": "411527", "name": "淮滨县" }, + { "code": "411528", "name": "息县" }, + { "code": "411571", "name": "信阳高新技术产业开发区" } + ] + }, + { + "code": "4116", + "name": "周口市", + "children": [ + { "code": "411602", "name": "川汇区" }, + { "code": "411603", "name": "淮阳区" }, + { "code": "411621", "name": "扶沟县" }, + { "code": "411622", "name": "西华县" }, + { "code": "411623", "name": "商水县" }, + { "code": "411624", "name": "沈丘县" }, + { "code": "411625", "name": "郸城县" }, + { "code": "411627", "name": "太康县" }, + { "code": "411628", "name": "鹿邑县" }, + { "code": "411671", "name": "河南周口经济开发区" }, + { "code": "411681", "name": "项城市" } + ] + }, + { + "code": "4117", + "name": "驻马店市", + "children": [ + { "code": "411702", "name": "驿城区" }, + { "code": "411721", "name": "西平县" }, + { "code": "411722", "name": "上蔡县" }, + { "code": "411723", "name": "平舆县" }, + { "code": "411724", "name": "正阳县" }, + { "code": "411725", "name": "确山县" }, + { "code": "411726", "name": "泌阳县" }, + { "code": "411727", "name": "汝南县" }, + { "code": "411728", "name": "遂平县" }, + { "code": "411729", "name": "新蔡县" }, + { "code": "411771", "name": "河南驻马店经济开发区" } + ] + }, + { + "code": "4190", + "name": "省直辖县级行政区划", + "children": [{ "code": "419001", "name": "济源市" }] + } + ] + }, + { + "code": "42", + "name": "湖北省", + "children": [ + { + "code": "4201", + "name": "武汉市", + "children": [ + { "code": "420102", "name": "江岸区" }, + { "code": "420103", "name": "江汉区" }, + { "code": "420104", "name": "硚口区" }, + { "code": "420105", "name": "汉阳区" }, + { "code": "420106", "name": "武昌区" }, + { "code": "420107", "name": "青山区" }, + { "code": "420111", "name": "洪山区" }, + { "code": "420112", "name": "东西湖区" }, + { "code": "420113", "name": "汉南区" }, + { "code": "420114", "name": "蔡甸区" }, + { "code": "420115", "name": "江夏区" }, + { "code": "420116", "name": "黄陂区" }, + { "code": "420117", "name": "新洲区" } + ] + }, + { + "code": "4202", + "name": "黄石市", + "children": [ + { "code": "420202", "name": "黄石港区" }, + { "code": "420203", "name": "西塞山区" }, + { "code": "420204", "name": "下陆区" }, + { "code": "420205", "name": "铁山区" }, + { "code": "420222", "name": "阳新县" }, + { "code": "420281", "name": "大冶市" } + ] + }, + { + "code": "4203", + "name": "十堰市", + "children": [ + { "code": "420302", "name": "茅箭区" }, + { "code": "420303", "name": "张湾区" }, + { "code": "420304", "name": "郧阳区" }, + { "code": "420322", "name": "郧西县" }, + { "code": "420323", "name": "竹山县" }, + { "code": "420324", "name": "竹溪县" }, + { "code": "420325", "name": "房县" }, + { "code": "420381", "name": "丹江口市" } + ] + }, + { + "code": "4205", + "name": "宜昌市", + "children": [ + { "code": "420502", "name": "西陵区" }, + { "code": "420503", "name": "伍家岗区" }, + { "code": "420504", "name": "点军区" }, + { "code": "420505", "name": "猇亭区" }, + { "code": "420506", "name": "夷陵区" }, + { "code": "420525", "name": "远安县" }, + { "code": "420526", "name": "兴山县" }, + { "code": "420527", "name": "秭归县" }, + { "code": "420528", "name": "长阳土家族自治县" }, + { "code": "420529", "name": "五峰土家族自治县" }, + { "code": "420581", "name": "宜都市" }, + { "code": "420582", "name": "当阳市" }, + { "code": "420583", "name": "枝江市" } + ] + }, + { + "code": "4206", + "name": "襄阳市", + "children": [ + { "code": "420602", "name": "襄城区" }, + { "code": "420606", "name": "樊城区" }, + { "code": "420607", "name": "襄州区" }, + { "code": "420624", "name": "南漳县" }, + { "code": "420625", "name": "谷城县" }, + { "code": "420626", "name": "保康县" }, + { "code": "420682", "name": "老河口市" }, + { "code": "420683", "name": "枣阳市" }, + { "code": "420684", "name": "宜城市" } + ] + }, + { + "code": "4207", + "name": "鄂州市", + "children": [ + { "code": "420702", "name": "梁子湖区" }, + { "code": "420703", "name": "华容区" }, + { "code": "420704", "name": "鄂城区" } + ] + }, + { + "code": "4208", + "name": "荆门市", + "children": [ + { "code": "420802", "name": "东宝区" }, + { "code": "420804", "name": "掇刀区" }, + { "code": "420822", "name": "沙洋县" }, + { "code": "420881", "name": "钟祥市" }, + { "code": "420882", "name": "京山市" } + ] + }, + { + "code": "4209", + "name": "孝感市", + "children": [ + { "code": "420902", "name": "孝南区" }, + { "code": "420921", "name": "孝昌县" }, + { "code": "420922", "name": "大悟县" }, + { "code": "420923", "name": "云梦县" }, + { "code": "420981", "name": "应城市" }, + { "code": "420982", "name": "安陆市" }, + { "code": "420984", "name": "汉川市" } + ] + }, + { + "code": "4210", + "name": "荆州市", + "children": [ + { "code": "421002", "name": "沙市区" }, + { "code": "421003", "name": "荆州区" }, + { "code": "421022", "name": "公安县" }, + { "code": "421024", "name": "江陵县" }, + { "code": "421071", "name": "荆州经济技术开发区" }, + { "code": "421081", "name": "石首市" }, + { "code": "421083", "name": "洪湖市" }, + { "code": "421087", "name": "松滋市" }, + { "code": "421088", "name": "监利市" } + ] + }, + { + "code": "4211", + "name": "黄冈市", + "children": [ + { "code": "421102", "name": "黄州区" }, + { "code": "421121", "name": "团风县" }, + { "code": "421122", "name": "红安县" }, + { "code": "421123", "name": "罗田县" }, + { "code": "421124", "name": "英山县" }, + { "code": "421125", "name": "浠水县" }, + { "code": "421126", "name": "蕲春县" }, + { "code": "421127", "name": "黄梅县" }, + { "code": "421171", "name": "龙感湖管理区" }, + { "code": "421181", "name": "麻城市" }, + { "code": "421182", "name": "武穴市" } + ] + }, + { + "code": "4212", + "name": "咸宁市", + "children": [ + { "code": "421202", "name": "咸安区" }, + { "code": "421221", "name": "嘉鱼县" }, + { "code": "421222", "name": "通城县" }, + { "code": "421223", "name": "崇阳县" }, + { "code": "421224", "name": "通山县" }, + { "code": "421281", "name": "赤壁市" } + ] + }, + { + "code": "4213", + "name": "随州市", + "children": [ + { "code": "421303", "name": "曾都区" }, + { "code": "421321", "name": "随县" }, + { "code": "421381", "name": "广水市" } + ] + }, + { + "code": "4228", + "name": "恩施土家族苗族自治州", + "children": [ + { "code": "422801", "name": "恩施市" }, + { "code": "422802", "name": "利川市" }, + { "code": "422822", "name": "建始县" }, + { "code": "422823", "name": "巴东县" }, + { "code": "422825", "name": "宣恩县" }, + { "code": "422826", "name": "咸丰县" }, + { "code": "422827", "name": "来凤县" }, + { "code": "422828", "name": "鹤峰县" } + ] + }, + { + "code": "4290", + "name": "省直辖县级行政区划", + "children": [ + { "code": "429004", "name": "仙桃市" }, + { "code": "429005", "name": "潜江市" }, + { "code": "429006", "name": "天门市" }, + { "code": "429021", "name": "神农架林区" } + ] + } + ] + }, + { + "code": "43", + "name": "湖南省", + "children": [ + { + "code": "4301", + "name": "长沙市", + "children": [ + { "code": "430102", "name": "芙蓉区" }, + { "code": "430103", "name": "天心区" }, + { "code": "430104", "name": "岳麓区" }, + { "code": "430105", "name": "开福区" }, + { "code": "430111", "name": "雨花区" }, + { "code": "430112", "name": "望城区" }, + { "code": "430121", "name": "长沙县" }, + { "code": "430181", "name": "浏阳市" }, + { "code": "430182", "name": "宁乡市" } + ] + }, + { + "code": "4302", + "name": "株洲市", + "children": [ + { "code": "430202", "name": "荷塘区" }, + { "code": "430203", "name": "芦淞区" }, + { "code": "430204", "name": "石峰区" }, + { "code": "430211", "name": "天元区" }, + { "code": "430212", "name": "渌口区" }, + { "code": "430223", "name": "攸县" }, + { "code": "430224", "name": "茶陵县" }, + { "code": "430225", "name": "炎陵县" }, + { "code": "430281", "name": "醴陵市" } + ] + }, + { + "code": "4303", + "name": "湘潭市", + "children": [ + { "code": "430302", "name": "雨湖区" }, + { "code": "430304", "name": "岳塘区" }, + { "code": "430321", "name": "湘潭县" }, + { "code": "430371", "name": "湖南湘潭高新技术产业园区" }, + { "code": "430372", "name": "湘潭昭山示范区" }, + { "code": "430373", "name": "湘潭九华示范区" }, + { "code": "430381", "name": "湘乡市" }, + { "code": "430382", "name": "韶山市" } + ] + }, + { + "code": "4304", + "name": "衡阳市", + "children": [ + { "code": "430405", "name": "珠晖区" }, + { "code": "430406", "name": "雁峰区" }, + { "code": "430407", "name": "石鼓区" }, + { "code": "430408", "name": "蒸湘区" }, + { "code": "430412", "name": "南岳区" }, + { "code": "430421", "name": "衡阳县" }, + { "code": "430422", "name": "衡南县" }, + { "code": "430423", "name": "衡山县" }, + { "code": "430424", "name": "衡东县" }, + { "code": "430426", "name": "祁东县" }, + { "code": "430471", "name": "衡阳综合保税区" }, + { "code": "430472", "name": "湖南衡阳高新技术产业园区" }, + { "code": "430473", "name": "湖南衡阳松木经济开发区" }, + { "code": "430481", "name": "耒阳市" }, + { "code": "430482", "name": "常宁市" } + ] + }, + { + "code": "4305", + "name": "邵阳市", + "children": [ + { "code": "430502", "name": "双清区" }, + { "code": "430503", "name": "大祥区" }, + { "code": "430511", "name": "北塔区" }, + { "code": "430522", "name": "新邵县" }, + { "code": "430523", "name": "邵阳县" }, + { "code": "430524", "name": "隆回县" }, + { "code": "430525", "name": "洞口县" }, + { "code": "430527", "name": "绥宁县" }, + { "code": "430528", "name": "新宁县" }, + { "code": "430529", "name": "城步苗族自治县" }, + { "code": "430581", "name": "武冈市" }, + { "code": "430582", "name": "邵东市" } + ] + }, + { + "code": "4306", + "name": "岳阳市", + "children": [ + { "code": "430602", "name": "岳阳楼区" }, + { "code": "430603", "name": "云溪区" }, + { "code": "430611", "name": "君山区" }, + { "code": "430621", "name": "岳阳县" }, + { "code": "430623", "name": "华容县" }, + { "code": "430624", "name": "湘阴县" }, + { "code": "430626", "name": "平江县" }, + { "code": "430671", "name": "岳阳市屈原管理区" }, + { "code": "430681", "name": "汨罗市" }, + { "code": "430682", "name": "临湘市" } + ] + }, + { + "code": "4307", + "name": "常德市", + "children": [ + { "code": "430702", "name": "武陵区" }, + { "code": "430703", "name": "鼎城区" }, + { "code": "430721", "name": "安乡县" }, + { "code": "430722", "name": "汉寿县" }, + { "code": "430723", "name": "澧县" }, + { "code": "430724", "name": "临澧县" }, + { "code": "430725", "name": "桃源县" }, + { "code": "430726", "name": "石门县" }, + { "code": "430771", "name": "常德市西洞庭管理区" }, + { "code": "430781", "name": "津市市" } + ] + }, + { + "code": "4308", + "name": "张家界市", + "children": [ + { "code": "430802", "name": "永定区" }, + { "code": "430811", "name": "武陵源区" }, + { "code": "430821", "name": "慈利县" }, + { "code": "430822", "name": "桑植县" } + ] + }, + { + "code": "4309", + "name": "益阳市", + "children": [ + { "code": "430902", "name": "资阳区" }, + { "code": "430903", "name": "赫山区" }, + { "code": "430921", "name": "南县" }, + { "code": "430922", "name": "桃江县" }, + { "code": "430923", "name": "安化县" }, + { "code": "430971", "name": "益阳市大通湖管理区" }, + { "code": "430972", "name": "湖南益阳高新技术产业园区" }, + { "code": "430981", "name": "沅江市" } + ] + }, + { + "code": "4310", + "name": "郴州市", + "children": [ + { "code": "431002", "name": "北湖区" }, + { "code": "431003", "name": "苏仙区" }, + { "code": "431021", "name": "桂阳县" }, + { "code": "431022", "name": "宜章县" }, + { "code": "431023", "name": "永兴县" }, + { "code": "431024", "name": "嘉禾县" }, + { "code": "431025", "name": "临武县" }, + { "code": "431026", "name": "汝城县" }, + { "code": "431027", "name": "桂东县" }, + { "code": "431028", "name": "安仁县" }, + { "code": "431081", "name": "资兴市" } + ] + }, + { + "code": "4311", + "name": "永州市", + "children": [ + { "code": "431102", "name": "零陵区" }, + { "code": "431103", "name": "冷水滩区" }, + { "code": "431122", "name": "东安县" }, + { "code": "431123", "name": "双牌县" }, + { "code": "431124", "name": "道县" }, + { "code": "431125", "name": "江永县" }, + { "code": "431126", "name": "宁远县" }, + { "code": "431127", "name": "蓝山县" }, + { "code": "431128", "name": "新田县" }, + { "code": "431129", "name": "江华瑶族自治县" }, + { "code": "431171", "name": "永州经济技术开发区" }, + { "code": "431173", "name": "永州市回龙圩管理区" }, + { "code": "431181", "name": "祁阳市" } + ] + }, + { + "code": "4312", + "name": "怀化市", + "children": [ + { "code": "431202", "name": "鹤城区" }, + { "code": "431221", "name": "中方县" }, + { "code": "431222", "name": "沅陵县" }, + { "code": "431223", "name": "辰溪县" }, + { "code": "431224", "name": "溆浦县" }, + { "code": "431225", "name": "会同县" }, + { "code": "431226", "name": "麻阳苗族自治县" }, + { "code": "431227", "name": "新晃侗族自治县" }, + { "code": "431228", "name": "芷江侗族自治县" }, + { "code": "431229", "name": "靖州苗族侗族自治县" }, + { "code": "431230", "name": "通道侗族自治县" }, + { "code": "431271", "name": "怀化市洪江管理区" }, + { "code": "431281", "name": "洪江市" } + ] + }, + { + "code": "4313", + "name": "娄底市", + "children": [ + { "code": "431302", "name": "娄星区" }, + { "code": "431321", "name": "双峰县" }, + { "code": "431322", "name": "新化县" }, + { "code": "431381", "name": "冷水江市" }, + { "code": "431382", "name": "涟源市" } + ] + }, + { + "code": "4331", + "name": "湘西土家族苗族自治州", + "children": [ + { "code": "433101", "name": "吉首市" }, + { "code": "433122", "name": "泸溪县" }, + { "code": "433123", "name": "凤凰县" }, + { "code": "433124", "name": "花垣县" }, + { "code": "433125", "name": "保靖县" }, + { "code": "433126", "name": "古丈县" }, + { "code": "433127", "name": "永顺县" }, + { "code": "433130", "name": "龙山县" } + ] + } + ] + }, + { + "code": "44", + "name": "广东省", + "children": [ + { + "code": "4401", + "name": "广州市", + "children": [ + { "code": "440103", "name": "荔湾区" }, + { "code": "440104", "name": "越秀区" }, + { "code": "440105", "name": "海珠区" }, + { "code": "440106", "name": "天河区" }, + { "code": "440111", "name": "白云区" }, + { "code": "440112", "name": "黄埔区" }, + { "code": "440113", "name": "番禺区" }, + { "code": "440114", "name": "花都区" }, + { "code": "440115", "name": "南沙区" }, + { "code": "440117", "name": "从化区" }, + { "code": "440118", "name": "增城区" } + ] + }, + { + "code": "4402", + "name": "韶关市", + "children": [ + { "code": "440203", "name": "武江区" }, + { "code": "440204", "name": "浈江区" }, + { "code": "440205", "name": "曲江区" }, + { "code": "440222", "name": "始兴县" }, + { "code": "440224", "name": "仁化县" }, + { "code": "440229", "name": "翁源县" }, + { "code": "440232", "name": "乳源瑶族自治县" }, + { "code": "440233", "name": "新丰县" }, + { "code": "440281", "name": "乐昌市" }, + { "code": "440282", "name": "南雄市" } + ] + }, + { + "code": "4403", + "name": "深圳市", + "children": [ + { "code": "440303", "name": "罗湖区" }, + { "code": "440304", "name": "福田区" }, + { "code": "440305", "name": "南山区" }, + { "code": "440306", "name": "宝安区" }, + { "code": "440307", "name": "龙岗区" }, + { "code": "440308", "name": "盐田区" }, + { "code": "440309", "name": "龙华区" }, + { "code": "440310", "name": "坪山区" }, + { "code": "440311", "name": "光明区" } + ] + }, + { + "code": "4404", + "name": "珠海市", + "children": [ + { "code": "440402", "name": "香洲区" }, + { "code": "440403", "name": "斗门区" }, + { "code": "440404", "name": "金湾区" } + ] + }, + { + "code": "4405", + "name": "汕头市", + "children": [ + { "code": "440507", "name": "龙湖区" }, + { "code": "440511", "name": "金平区" }, + { "code": "440512", "name": "濠江区" }, + { "code": "440513", "name": "潮阳区" }, + { "code": "440514", "name": "潮南区" }, + { "code": "440515", "name": "澄海区" }, + { "code": "440523", "name": "南澳县" } + ] + }, + { + "code": "4406", + "name": "佛山市", + "children": [ + { "code": "440604", "name": "禅城区" }, + { "code": "440605", "name": "南海区" }, + { "code": "440606", "name": "顺德区" }, + { "code": "440607", "name": "三水区" }, + { "code": "440608", "name": "高明区" } + ] + }, + { + "code": "4407", + "name": "江门市", + "children": [ + { "code": "440703", "name": "蓬江区" }, + { "code": "440704", "name": "江海区" }, + { "code": "440705", "name": "新会区" }, + { "code": "440781", "name": "台山市" }, + { "code": "440783", "name": "开平市" }, + { "code": "440784", "name": "鹤山市" }, + { "code": "440785", "name": "恩平市" } + ] + }, + { + "code": "4408", + "name": "湛江市", + "children": [ + { "code": "440802", "name": "赤坎区" }, + { "code": "440803", "name": "霞山区" }, + { "code": "440804", "name": "坡头区" }, + { "code": "440811", "name": "麻章区" }, + { "code": "440823", "name": "遂溪县" }, + { "code": "440825", "name": "徐闻县" }, + { "code": "440881", "name": "廉江市" }, + { "code": "440882", "name": "雷州市" }, + { "code": "440883", "name": "吴川市" } + ] + }, + { + "code": "4409", + "name": "茂名市", + "children": [ + { "code": "440902", "name": "茂南区" }, + { "code": "440904", "name": "电白区" }, + { "code": "440981", "name": "高州市" }, + { "code": "440982", "name": "化州市" }, + { "code": "440983", "name": "信宜市" } + ] + }, + { + "code": "4412", + "name": "肇庆市", + "children": [ + { "code": "441202", "name": "端州区" }, + { "code": "441203", "name": "鼎湖区" }, + { "code": "441204", "name": "高要区" }, + { "code": "441223", "name": "广宁县" }, + { "code": "441224", "name": "怀集县" }, + { "code": "441225", "name": "封开县" }, + { "code": "441226", "name": "德庆县" }, + { "code": "441284", "name": "四会市" } + ] + }, + { + "code": "4413", + "name": "惠州市", + "children": [ + { "code": "441302", "name": "惠城区" }, + { "code": "441303", "name": "惠阳区" }, + { "code": "441322", "name": "博罗县" }, + { "code": "441323", "name": "惠东县" }, + { "code": "441324", "name": "龙门县" } + ] + }, + { + "code": "4414", + "name": "梅州市", + "children": [ + { "code": "441402", "name": "梅江区" }, + { "code": "441403", "name": "梅县区" }, + { "code": "441422", "name": "大埔县" }, + { "code": "441423", "name": "丰顺县" }, + { "code": "441424", "name": "五华县" }, + { "code": "441426", "name": "平远县" }, + { "code": "441427", "name": "蕉岭县" }, + { "code": "441481", "name": "兴宁市" } + ] + }, + { + "code": "4415", + "name": "汕尾市", + "children": [ + { "code": "441502", "name": "城区" }, + { "code": "441521", "name": "海丰县" }, + { "code": "441523", "name": "陆河县" }, + { "code": "441581", "name": "陆丰市" } + ] + }, + { + "code": "4416", + "name": "河源市", + "children": [ + { "code": "441602", "name": "源城区" }, + { "code": "441621", "name": "紫金县" }, + { "code": "441622", "name": "龙川县" }, + { "code": "441623", "name": "连平县" }, + { "code": "441624", "name": "和平县" }, + { "code": "441625", "name": "东源县" } + ] + }, + { + "code": "4417", + "name": "阳江市", + "children": [ + { "code": "441702", "name": "江城区" }, + { "code": "441704", "name": "阳东区" }, + { "code": "441721", "name": "阳西县" }, + { "code": "441781", "name": "阳春市" } + ] + }, + { + "code": "4418", + "name": "清远市", + "children": [ + { "code": "441802", "name": "清城区" }, + { "code": "441803", "name": "清新区" }, + { "code": "441821", "name": "佛冈县" }, + { "code": "441823", "name": "阳山县" }, + { "code": "441825", "name": "连山壮族瑶族自治县" }, + { "code": "441826", "name": "连南瑶族自治县" }, + { "code": "441881", "name": "英德市" }, + { "code": "441882", "name": "连州市" } + ] + }, + { + "code": "4419", + "name": "东莞市", + "children": [ + { "code": "441900003", "name": "东城街道" }, + { "code": "441900004", "name": "南城街道" }, + { "code": "441900005", "name": "万江街道" }, + { "code": "441900006", "name": "莞城街道" }, + { "code": "441900101", "name": "石碣镇" }, + { "code": "441900102", "name": "石龙镇" }, + { "code": "441900103", "name": "茶山镇" }, + { "code": "441900104", "name": "石排镇" }, + { "code": "441900105", "name": "企石镇" }, + { "code": "441900106", "name": "横沥镇" }, + { "code": "441900107", "name": "桥头镇" }, + { "code": "441900108", "name": "谢岗镇" }, + { "code": "441900109", "name": "东坑镇" }, + { "code": "441900110", "name": "常平镇" }, + { "code": "441900111", "name": "寮步镇" }, + { "code": "441900112", "name": "樟木头镇" }, + { "code": "441900113", "name": "大朗镇" }, + { "code": "441900114", "name": "黄江镇" }, + { "code": "441900115", "name": "清溪镇" }, + { "code": "441900116", "name": "塘厦镇" }, + { "code": "441900117", "name": "凤岗镇" }, + { "code": "441900118", "name": "大岭山镇" }, + { "code": "441900119", "name": "长安镇" }, + { "code": "441900121", "name": "虎门镇" }, + { "code": "441900122", "name": "厚街镇" }, + { "code": "441900123", "name": "沙田镇" }, + { "code": "441900124", "name": "道滘镇" }, + { "code": "441900125", "name": "洪梅镇" }, + { "code": "441900126", "name": "麻涌镇" }, + { "code": "441900127", "name": "望牛墩镇" }, + { "code": "441900128", "name": "中堂镇" }, + { "code": "441900129", "name": "高埗镇" }, + { "code": "441900401", "name": "松山湖" }, + { "code": "441900402", "name": "东莞港" }, + { "code": "441900403", "name": "东莞生态园" }, + { "code": "441900404", "name": "东莞滨海湾新区" } + ] + }, + { + "code": "4420", + "name": "中山市", + "children": [ + { "code": "442000001", "name": "石岐街道" }, + { "code": "442000002", "name": "东区街道" }, + { "code": "442000003", "name": "中山港街道" }, + { "code": "442000004", "name": "西区街道" }, + { "code": "442000005", "name": "南区街道" }, + { "code": "442000006", "name": "五桂山街道" }, + { "code": "442000007", "name": "民众街道" }, + { "code": "442000008", "name": "南朗街道" }, + { "code": "442000101", "name": "黄圃镇" }, + { "code": "442000103", "name": "东凤镇" }, + { "code": "442000105", "name": "古镇镇" }, + { "code": "442000106", "name": "沙溪镇" }, + { "code": "442000107", "name": "坦洲镇" }, + { "code": "442000108", "name": "港口镇" }, + { "code": "442000109", "name": "三角镇" }, + { "code": "442000110", "name": "横栏镇" }, + { "code": "442000111", "name": "南头镇" }, + { "code": "442000112", "name": "阜沙镇" }, + { "code": "442000114", "name": "三乡镇" }, + { "code": "442000115", "name": "板芙镇" }, + { "code": "442000116", "name": "大涌镇" }, + { "code": "442000117", "name": "神湾镇" }, + { "code": "442000118", "name": "小榄镇" } + ] + }, + { + "code": "4451", + "name": "潮州市", + "children": [ + { "code": "445102", "name": "湘桥区" }, + { "code": "445103", "name": "潮安区" }, + { "code": "445122", "name": "饶平县" } + ] + }, + { + "code": "4452", + "name": "揭阳市", + "children": [ + { "code": "445202", "name": "榕城区" }, + { "code": "445203", "name": "揭东区" }, + { "code": "445222", "name": "揭西县" }, + { "code": "445224", "name": "惠来县" }, + { "code": "445281", "name": "普宁市" } + ] + }, + { + "code": "4453", + "name": "云浮市", + "children": [ + { "code": "445302", "name": "云城区" }, + { "code": "445303", "name": "云安区" }, + { "code": "445321", "name": "新兴县" }, + { "code": "445322", "name": "郁南县" }, + { "code": "445381", "name": "罗定市" } + ] + } + ] + }, + { + "code": "45", + "name": "广西壮族自治区", + "children": [ + { + "code": "4501", + "name": "南宁市", + "children": [ + { "code": "450102", "name": "兴宁区" }, + { "code": "450103", "name": "青秀区" }, + { "code": "450105", "name": "江南区" }, + { "code": "450107", "name": "西乡塘区" }, + { "code": "450108", "name": "良庆区" }, + { "code": "450109", "name": "邕宁区" }, + { "code": "450110", "name": "武鸣区" }, + { "code": "450123", "name": "隆安县" }, + { "code": "450124", "name": "马山县" }, + { "code": "450125", "name": "上林县" }, + { "code": "450126", "name": "宾阳县" }, + { "code": "450181", "name": "横州市" } + ] + }, + { + "code": "4502", + "name": "柳州市", + "children": [ + { "code": "450202", "name": "城中区" }, + { "code": "450203", "name": "鱼峰区" }, + { "code": "450204", "name": "柳南区" }, + { "code": "450205", "name": "柳北区" }, + { "code": "450206", "name": "柳江区" }, + { "code": "450222", "name": "柳城县" }, + { "code": "450223", "name": "鹿寨县" }, + { "code": "450224", "name": "融安县" }, + { "code": "450225", "name": "融水苗族自治县" }, + { "code": "450226", "name": "三江侗族自治县" } + ] + }, + { + "code": "4503", + "name": "桂林市", + "children": [ + { "code": "450302", "name": "秀峰区" }, + { "code": "450303", "name": "叠彩区" }, + { "code": "450304", "name": "象山区" }, + { "code": "450305", "name": "七星区" }, + { "code": "450311", "name": "雁山区" }, + { "code": "450312", "name": "临桂区" }, + { "code": "450321", "name": "阳朔县" }, + { "code": "450323", "name": "灵川县" }, + { "code": "450324", "name": "全州县" }, + { "code": "450325", "name": "兴安县" }, + { "code": "450326", "name": "永福县" }, + { "code": "450327", "name": "灌阳县" }, + { "code": "450328", "name": "龙胜各族自治县" }, + { "code": "450329", "name": "资源县" }, + { "code": "450330", "name": "平乐县" }, + { "code": "450332", "name": "恭城瑶族自治县" }, + { "code": "450381", "name": "荔浦市" } + ] + }, + { + "code": "4504", + "name": "梧州市", + "children": [ + { "code": "450403", "name": "万秀区" }, + { "code": "450405", "name": "长洲区" }, + { "code": "450406", "name": "龙圩区" }, + { "code": "450421", "name": "苍梧县" }, + { "code": "450422", "name": "藤县" }, + { "code": "450423", "name": "蒙山县" }, + { "code": "450481", "name": "岑溪市" } + ] + }, + { + "code": "4505", + "name": "北海市", + "children": [ + { "code": "450502", "name": "海城区" }, + { "code": "450503", "name": "银海区" }, + { "code": "450512", "name": "铁山港区" }, + { "code": "450521", "name": "合浦县" } + ] + }, + { + "code": "4506", + "name": "防城港市", + "children": [ + { "code": "450602", "name": "港口区" }, + { "code": "450603", "name": "防城区" }, + { "code": "450621", "name": "上思县" }, + { "code": "450681", "name": "东兴市" } + ] + }, + { + "code": "4507", + "name": "钦州市", + "children": [ + { "code": "450702", "name": "钦南区" }, + { "code": "450703", "name": "钦北区" }, + { "code": "450721", "name": "灵山县" }, + { "code": "450722", "name": "浦北县" } + ] + }, + { + "code": "4508", + "name": "贵港市", + "children": [ + { "code": "450802", "name": "港北区" }, + { "code": "450803", "name": "港南区" }, + { "code": "450804", "name": "覃塘区" }, + { "code": "450821", "name": "平南县" }, + { "code": "450881", "name": "桂平市" } + ] + }, + { + "code": "4509", + "name": "玉林市", + "children": [ + { "code": "450902", "name": "玉州区" }, + { "code": "450903", "name": "福绵区" }, + { "code": "450921", "name": "容县" }, + { "code": "450922", "name": "陆川县" }, + { "code": "450923", "name": "博白县" }, + { "code": "450924", "name": "兴业县" }, + { "code": "450981", "name": "北流市" } + ] + }, + { + "code": "4510", + "name": "百色市", + "children": [ + { "code": "451002", "name": "右江区" }, + { "code": "451003", "name": "田阳区" }, + { "code": "451022", "name": "田东县" }, + { "code": "451024", "name": "德保县" }, + { "code": "451026", "name": "那坡县" }, + { "code": "451027", "name": "凌云县" }, + { "code": "451028", "name": "乐业县" }, + { "code": "451029", "name": "田林县" }, + { "code": "451030", "name": "西林县" }, + { "code": "451031", "name": "隆林各族自治县" }, + { "code": "451081", "name": "靖西市" }, + { "code": "451082", "name": "平果市" } + ] + }, + { + "code": "4511", + "name": "贺州市", + "children": [ + { "code": "451102", "name": "八步区" }, + { "code": "451103", "name": "平桂区" }, + { "code": "451121", "name": "昭平县" }, + { "code": "451122", "name": "钟山县" }, + { "code": "451123", "name": "富川瑶族自治县" } + ] + }, + { + "code": "4512", + "name": "河池市", + "children": [ + { "code": "451202", "name": "金城江区" }, + { "code": "451203", "name": "宜州区" }, + { "code": "451221", "name": "南丹县" }, + { "code": "451222", "name": "天峨县" }, + { "code": "451223", "name": "凤山县" }, + { "code": "451224", "name": "东兰县" }, + { "code": "451225", "name": "罗城仫佬族自治县" }, + { "code": "451226", "name": "环江毛南族自治县" }, + { "code": "451227", "name": "巴马瑶族自治县" }, + { "code": "451228", "name": "都安瑶族自治县" }, + { "code": "451229", "name": "大化瑶族自治县" } + ] + }, + { + "code": "4513", + "name": "来宾市", + "children": [ + { "code": "451302", "name": "兴宾区" }, + { "code": "451321", "name": "忻城县" }, + { "code": "451322", "name": "象州县" }, + { "code": "451323", "name": "武宣县" }, + { "code": "451324", "name": "金秀瑶族自治县" }, + { "code": "451381", "name": "合山市" } + ] + }, + { + "code": "4514", + "name": "崇左市", + "children": [ + { "code": "451402", "name": "江州区" }, + { "code": "451421", "name": "扶绥县" }, + { "code": "451422", "name": "宁明县" }, + { "code": "451423", "name": "龙州县" }, + { "code": "451424", "name": "大新县" }, + { "code": "451425", "name": "天等县" }, + { "code": "451481", "name": "凭祥市" } + ] + } + ] + }, + { + "code": "46", + "name": "海南省", + "children": [ + { + "code": "4601", + "name": "海口市", + "children": [ + { "code": "460105", "name": "秀英区" }, + { "code": "460106", "name": "龙华区" }, + { "code": "460107", "name": "琼山区" }, + { "code": "460108", "name": "美兰区" } + ] + }, + { + "code": "4602", + "name": "三亚市", + "children": [ + { "code": "460202", "name": "海棠区" }, + { "code": "460203", "name": "吉阳区" }, + { "code": "460204", "name": "天涯区" }, + { "code": "460205", "name": "崖州区" } + ] + }, + { + "code": "4603", + "name": "三沙市", + "children": [ + { "code": "460321", "name": "西沙群岛" }, + { "code": "460322", "name": "南沙群岛" }, + { "code": "460323", "name": "中沙群岛的岛礁及其海域" } + ] + }, + { + "code": "4604", + "name": "儋州市", + "children": [ + { "code": "460400100", "name": "那大镇" }, + { "code": "460400101", "name": "和庆镇" }, + { "code": "460400102", "name": "南丰镇" }, + { "code": "460400103", "name": "大成镇" }, + { "code": "460400104", "name": "雅星镇" }, + { "code": "460400105", "name": "兰洋镇" }, + { "code": "460400106", "name": "光村镇" }, + { "code": "460400107", "name": "木棠镇" }, + { "code": "460400108", "name": "海头镇" }, + { "code": "460400109", "name": "峨蔓镇" }, + { "code": "460400111", "name": "王五镇" }, + { "code": "460400112", "name": "白马井镇" }, + { "code": "460400113", "name": "中和镇" }, + { "code": "460400114", "name": "排浦镇" }, + { "code": "460400115", "name": "东成镇" }, + { "code": "460400116", "name": "新州镇" }, + { "code": "460400499", "name": "洋浦经济开发区" }, + { "code": "460400500", "name": "华南热作学院" } + ] + }, + { + "code": "4690", + "name": "省直辖县级行政区划", + "children": [ + { "code": "469001", "name": "五指山市" }, + { "code": "469002", "name": "琼海市" }, + { "code": "469005", "name": "文昌市" }, + { "code": "469006", "name": "万宁市" }, + { "code": "469007", "name": "东方市" }, + { "code": "469021", "name": "定安县" }, + { "code": "469022", "name": "屯昌县" }, + { "code": "469023", "name": "澄迈县" }, + { "code": "469024", "name": "临高县" }, + { "code": "469025", "name": "白沙黎族自治县" }, + { "code": "469026", "name": "昌江黎族自治县" }, + { "code": "469027", "name": "乐东黎族自治县" }, + { "code": "469028", "name": "陵水黎族自治县" }, + { "code": "469029", "name": "保亭黎族苗族自治县" }, + { "code": "469030", "name": "琼中黎族苗族自治县" } + ] + } + ] + }, + { + "code": "50", + "name": "重庆市", + "children": [ + { + "code": "5001", + "name": "市辖区", + "children": [ + { "code": "500101", "name": "万州区" }, + { "code": "500102", "name": "涪陵区" }, + { "code": "500103", "name": "渝中区" }, + { "code": "500104", "name": "大渡口区" }, + { "code": "500105", "name": "江北区" }, + { "code": "500106", "name": "沙坪坝区" }, + { "code": "500107", "name": "九龙坡区" }, + { "code": "500108", "name": "南岸区" }, + { "code": "500109", "name": "北碚区" }, + { "code": "500110", "name": "綦江区" }, + { "code": "500111", "name": "大足区" }, + { "code": "500112", "name": "渝北区" }, + { "code": "500113", "name": "巴南区" }, + { "code": "500114", "name": "黔江区" }, + { "code": "500115", "name": "长寿区" }, + { "code": "500116", "name": "江津区" }, + { "code": "500117", "name": "合川区" }, + { "code": "500118", "name": "永川区" }, + { "code": "500119", "name": "南川区" }, + { "code": "500120", "name": "璧山区" }, + { "code": "500151", "name": "铜梁区" }, + { "code": "500152", "name": "潼南区" }, + { "code": "500153", "name": "荣昌区" }, + { "code": "500154", "name": "开州区" }, + { "code": "500155", "name": "梁平区" }, + { "code": "500156", "name": "武隆区" } + ] + }, + { + "code": "5002", + "name": "县", + "children": [ + { "code": "500229", "name": "城口县" }, + { "code": "500230", "name": "丰都县" }, + { "code": "500231", "name": "垫江县" }, + { "code": "500233", "name": "忠县" }, + { "code": "500235", "name": "云阳县" }, + { "code": "500236", "name": "奉节县" }, + { "code": "500237", "name": "巫山县" }, + { "code": "500238", "name": "巫溪县" }, + { "code": "500240", "name": "石柱土家族自治县" }, + { "code": "500241", "name": "秀山土家族苗族自治县" }, + { "code": "500242", "name": "酉阳土家族苗族自治县" }, + { "code": "500243", "name": "彭水苗族土家族自治县" } + ] + } + ] + }, + { + "code": "51", + "name": "四川省", + "children": [ + { + "code": "5101", + "name": "成都市", + "children": [ + { "code": "510104", "name": "锦江区" }, + { "code": "510105", "name": "青羊区" }, + { "code": "510106", "name": "金牛区" }, + { "code": "510107", "name": "武侯区" }, + { "code": "510108", "name": "成华区" }, + { "code": "510112", "name": "龙泉驿区" }, + { "code": "510113", "name": "青白江区" }, + { "code": "510114", "name": "新都区" }, + { "code": "510115", "name": "温江区" }, + { "code": "510116", "name": "双流区" }, + { "code": "510117", "name": "郫都区" }, + { "code": "510118", "name": "新津区" }, + { "code": "510121", "name": "金堂县" }, + { "code": "510129", "name": "大邑县" }, + { "code": "510131", "name": "蒲江县" }, + { "code": "510181", "name": "都江堰市" }, + { "code": "510182", "name": "彭州市" }, + { "code": "510183", "name": "邛崃市" }, + { "code": "510184", "name": "崇州市" }, + { "code": "510185", "name": "简阳市" } + ] + }, + { + "code": "5103", + "name": "自贡市", + "children": [ + { "code": "510302", "name": "自流井区" }, + { "code": "510303", "name": "贡井区" }, + { "code": "510304", "name": "大安区" }, + { "code": "510311", "name": "沿滩区" }, + { "code": "510321", "name": "荣县" }, + { "code": "510322", "name": "富顺县" } + ] + }, + { + "code": "5104", + "name": "攀枝花市", + "children": [ + { "code": "510402", "name": "东区" }, + { "code": "510403", "name": "西区" }, + { "code": "510411", "name": "仁和区" }, + { "code": "510421", "name": "米易县" }, + { "code": "510422", "name": "盐边县" } + ] + }, + { + "code": "5105", + "name": "泸州市", + "children": [ + { "code": "510502", "name": "江阳区" }, + { "code": "510503", "name": "纳溪区" }, + { "code": "510504", "name": "龙马潭区" }, + { "code": "510521", "name": "泸县" }, + { "code": "510522", "name": "合江县" }, + { "code": "510524", "name": "叙永县" }, + { "code": "510525", "name": "古蔺县" } + ] + }, + { + "code": "5106", + "name": "德阳市", + "children": [ + { "code": "510603", "name": "旌阳区" }, + { "code": "510604", "name": "罗江区" }, + { "code": "510623", "name": "中江县" }, + { "code": "510681", "name": "广汉市" }, + { "code": "510682", "name": "什邡市" }, + { "code": "510683", "name": "绵竹市" } + ] + }, + { + "code": "5107", + "name": "绵阳市", + "children": [ + { "code": "510703", "name": "涪城区" }, + { "code": "510704", "name": "游仙区" }, + { "code": "510705", "name": "安州区" }, + { "code": "510722", "name": "三台县" }, + { "code": "510723", "name": "盐亭县" }, + { "code": "510725", "name": "梓潼县" }, + { "code": "510726", "name": "北川羌族自治县" }, + { "code": "510727", "name": "平武县" }, + { "code": "510781", "name": "江油市" } + ] + }, + { + "code": "5108", + "name": "广元市", + "children": [ + { "code": "510802", "name": "利州区" }, + { "code": "510811", "name": "昭化区" }, + { "code": "510812", "name": "朝天区" }, + { "code": "510821", "name": "旺苍县" }, + { "code": "510822", "name": "青川县" }, + { "code": "510823", "name": "剑阁县" }, + { "code": "510824", "name": "苍溪县" } + ] + }, + { + "code": "5109", + "name": "遂宁市", + "children": [ + { "code": "510903", "name": "船山区" }, + { "code": "510904", "name": "安居区" }, + { "code": "510921", "name": "蓬溪县" }, + { "code": "510923", "name": "大英县" }, + { "code": "510981", "name": "射洪市" } + ] + }, + { + "code": "5110", + "name": "内江市", + "children": [ + { "code": "511002", "name": "市中区" }, + { "code": "511011", "name": "东兴区" }, + { "code": "511024", "name": "威远县" }, + { "code": "511025", "name": "资中县" }, + { "code": "511083", "name": "隆昌市" } + ] + }, + { + "code": "5111", + "name": "乐山市", + "children": [ + { "code": "511102", "name": "市中区" }, + { "code": "511111", "name": "沙湾区" }, + { "code": "511112", "name": "五通桥区" }, + { "code": "511113", "name": "金口河区" }, + { "code": "511123", "name": "犍为县" }, + { "code": "511124", "name": "井研县" }, + { "code": "511126", "name": "夹江县" }, + { "code": "511129", "name": "沐川县" }, + { "code": "511132", "name": "峨边彝族自治县" }, + { "code": "511133", "name": "马边彝族自治县" }, + { "code": "511181", "name": "峨眉山市" } + ] + }, + { + "code": "5113", + "name": "南充市", + "children": [ + { "code": "511302", "name": "顺庆区" }, + { "code": "511303", "name": "高坪区" }, + { "code": "511304", "name": "嘉陵区" }, + { "code": "511321", "name": "南部县" }, + { "code": "511322", "name": "营山县" }, + { "code": "511323", "name": "蓬安县" }, + { "code": "511324", "name": "仪陇县" }, + { "code": "511325", "name": "西充县" }, + { "code": "511381", "name": "阆中市" } + ] + }, + { + "code": "5114", + "name": "眉山市", + "children": [ + { "code": "511402", "name": "东坡区" }, + { "code": "511403", "name": "彭山区" }, + { "code": "511421", "name": "仁寿县" }, + { "code": "511423", "name": "洪雅县" }, + { "code": "511424", "name": "丹棱县" }, + { "code": "511425", "name": "青神县" } + ] + }, + { + "code": "5115", + "name": "宜宾市", + "children": [ + { "code": "511502", "name": "翠屏区" }, + { "code": "511503", "name": "南溪区" }, + { "code": "511504", "name": "叙州区" }, + { "code": "511523", "name": "江安县" }, + { "code": "511524", "name": "长宁县" }, + { "code": "511525", "name": "高县" }, + { "code": "511526", "name": "珙县" }, + { "code": "511527", "name": "筠连县" }, + { "code": "511528", "name": "兴文县" }, + { "code": "511529", "name": "屏山县" } + ] + }, + { + "code": "5116", + "name": "广安市", + "children": [ + { "code": "511602", "name": "广安区" }, + { "code": "511603", "name": "前锋区" }, + { "code": "511621", "name": "岳池县" }, + { "code": "511622", "name": "武胜县" }, + { "code": "511623", "name": "邻水县" }, + { "code": "511681", "name": "华蓥市" } + ] + }, + { + "code": "5117", + "name": "达州市", + "children": [ + { "code": "511702", "name": "通川区" }, + { "code": "511703", "name": "达川区" }, + { "code": "511722", "name": "宣汉县" }, + { "code": "511723", "name": "开江县" }, + { "code": "511724", "name": "大竹县" }, + { "code": "511725", "name": "渠县" }, + { "code": "511781", "name": "万源市" } + ] + }, + { + "code": "5118", + "name": "雅安市", + "children": [ + { "code": "511802", "name": "雨城区" }, + { "code": "511803", "name": "名山区" }, + { "code": "511822", "name": "荥经县" }, + { "code": "511823", "name": "汉源县" }, + { "code": "511824", "name": "石棉县" }, + { "code": "511825", "name": "天全县" }, + { "code": "511826", "name": "芦山县" }, + { "code": "511827", "name": "宝兴县" } + ] + }, + { + "code": "5119", + "name": "巴中市", + "children": [ + { "code": "511902", "name": "巴州区" }, + { "code": "511903", "name": "恩阳区" }, + { "code": "511921", "name": "通江县" }, + { "code": "511922", "name": "南江县" }, + { "code": "511923", "name": "平昌县" } + ] + }, + { + "code": "5120", + "name": "资阳市", + "children": [ + { "code": "512002", "name": "雁江区" }, + { "code": "512021", "name": "安岳县" }, + { "code": "512022", "name": "乐至县" } + ] + }, + { + "code": "5132", + "name": "阿坝藏族羌族自治州", + "children": [ + { "code": "513201", "name": "马尔康市" }, + { "code": "513221", "name": "汶川县" }, + { "code": "513222", "name": "理县" }, + { "code": "513223", "name": "茂县" }, + { "code": "513224", "name": "松潘县" }, + { "code": "513225", "name": "九寨沟县" }, + { "code": "513226", "name": "金川县" }, + { "code": "513227", "name": "小金县" }, + { "code": "513228", "name": "黑水县" }, + { "code": "513230", "name": "壤塘县" }, + { "code": "513231", "name": "阿坝县" }, + { "code": "513232", "name": "若尔盖县" }, + { "code": "513233", "name": "红原县" } + ] + }, + { + "code": "5133", + "name": "甘孜藏族自治州", + "children": [ + { "code": "513301", "name": "康定市" }, + { "code": "513322", "name": "泸定县" }, + { "code": "513323", "name": "丹巴县" }, + { "code": "513324", "name": "九龙县" }, + { "code": "513325", "name": "雅江县" }, + { "code": "513326", "name": "道孚县" }, + { "code": "513327", "name": "炉霍县" }, + { "code": "513328", "name": "甘孜县" }, + { "code": "513329", "name": "新龙县" }, + { "code": "513330", "name": "德格县" }, + { "code": "513331", "name": "白玉县" }, + { "code": "513332", "name": "石渠县" }, + { "code": "513333", "name": "色达县" }, + { "code": "513334", "name": "理塘县" }, + { "code": "513335", "name": "巴塘县" }, + { "code": "513336", "name": "乡城县" }, + { "code": "513337", "name": "稻城县" }, + { "code": "513338", "name": "得荣县" } + ] + }, + { + "code": "5134", + "name": "凉山彝族自治州", + "children": [ + { "code": "513401", "name": "西昌市" }, + { "code": "513402", "name": "会理市" }, + { "code": "513422", "name": "木里藏族自治县" }, + { "code": "513423", "name": "盐源县" }, + { "code": "513424", "name": "德昌县" }, + { "code": "513426", "name": "会东县" }, + { "code": "513427", "name": "宁南县" }, + { "code": "513428", "name": "普格县" }, + { "code": "513429", "name": "布拖县" }, + { "code": "513430", "name": "金阳县" }, + { "code": "513431", "name": "昭觉县" }, + { "code": "513432", "name": "喜德县" }, + { "code": "513433", "name": "冕宁县" }, + { "code": "513434", "name": "越西县" }, + { "code": "513435", "name": "甘洛县" }, + { "code": "513436", "name": "美姑县" }, + { "code": "513437", "name": "雷波县" } + ] + } + ] + }, + { + "code": "52", + "name": "贵州省", + "children": [ + { + "code": "5201", + "name": "贵阳市", + "children": [ + { "code": "520102", "name": "南明区" }, + { "code": "520103", "name": "云岩区" }, + { "code": "520111", "name": "花溪区" }, + { "code": "520112", "name": "乌当区" }, + { "code": "520113", "name": "白云区" }, + { "code": "520115", "name": "观山湖区" }, + { "code": "520121", "name": "开阳县" }, + { "code": "520122", "name": "息烽县" }, + { "code": "520123", "name": "修文县" }, + { "code": "520181", "name": "清镇市" } + ] + }, + { + "code": "5202", + "name": "六盘水市", + "children": [ + { "code": "520201", "name": "钟山区" }, + { "code": "520203", "name": "六枝特区" }, + { "code": "520204", "name": "水城区" }, + { "code": "520281", "name": "盘州市" } + ] + }, + { + "code": "5203", + "name": "遵义市", + "children": [ + { "code": "520302", "name": "红花岗区" }, + { "code": "520303", "name": "汇川区" }, + { "code": "520304", "name": "播州区" }, + { "code": "520322", "name": "桐梓县" }, + { "code": "520323", "name": "绥阳县" }, + { "code": "520324", "name": "正安县" }, + { "code": "520325", "name": "道真仡佬族苗族自治县" }, + { "code": "520326", "name": "务川仡佬族苗族自治县" }, + { "code": "520327", "name": "凤冈县" }, + { "code": "520328", "name": "湄潭县" }, + { "code": "520329", "name": "余庆县" }, + { "code": "520330", "name": "习水县" }, + { "code": "520381", "name": "赤水市" }, + { "code": "520382", "name": "仁怀市" } + ] + }, + { + "code": "5204", + "name": "安顺市", + "children": [ + { "code": "520402", "name": "西秀区" }, + { "code": "520403", "name": "平坝区" }, + { "code": "520422", "name": "普定县" }, + { "code": "520423", "name": "镇宁布依族苗族自治县" }, + { "code": "520424", "name": "关岭布依族苗族自治县" }, + { "code": "520425", "name": "紫云苗族布依族自治县" } + ] + }, + { + "code": "5205", + "name": "毕节市", + "children": [ + { "code": "520502", "name": "七星关区" }, + { "code": "520521", "name": "大方县" }, + { "code": "520523", "name": "金沙县" }, + { "code": "520524", "name": "织金县" }, + { "code": "520525", "name": "纳雍县" }, + { "code": "520526", "name": "威宁彝族回族苗族自治县" }, + { "code": "520527", "name": "赫章县" }, + { "code": "520581", "name": "黔西市" } + ] + }, + { + "code": "5206", + "name": "铜仁市", + "children": [ + { "code": "520602", "name": "碧江区" }, + { "code": "520603", "name": "万山区" }, + { "code": "520621", "name": "江口县" }, + { "code": "520622", "name": "玉屏侗族自治县" }, + { "code": "520623", "name": "石阡县" }, + { "code": "520624", "name": "思南县" }, + { "code": "520625", "name": "印江土家族苗族自治县" }, + { "code": "520626", "name": "德江县" }, + { "code": "520627", "name": "沿河土家族自治县" }, + { "code": "520628", "name": "松桃苗族自治县" } + ] + }, + { + "code": "5223", + "name": "黔西南布依族苗族自治州", + "children": [ + { "code": "522301", "name": "兴义市" }, + { "code": "522302", "name": "兴仁市" }, + { "code": "522323", "name": "普安县" }, + { "code": "522324", "name": "晴隆县" }, + { "code": "522325", "name": "贞丰县" }, + { "code": "522326", "name": "望谟县" }, + { "code": "522327", "name": "册亨县" }, + { "code": "522328", "name": "安龙县" } + ] + }, + { + "code": "5226", + "name": "黔东南苗族侗族自治州", + "children": [ + { "code": "522601", "name": "凯里市" }, + { "code": "522622", "name": "黄平县" }, + { "code": "522623", "name": "施秉县" }, + { "code": "522624", "name": "三穗县" }, + { "code": "522625", "name": "镇远县" }, + { "code": "522626", "name": "岑巩县" }, + { "code": "522627", "name": "天柱县" }, + { "code": "522628", "name": "锦屏县" }, + { "code": "522629", "name": "剑河县" }, + { "code": "522630", "name": "台江县" }, + { "code": "522631", "name": "黎平县" }, + { "code": "522632", "name": "榕江县" }, + { "code": "522633", "name": "从江县" }, + { "code": "522634", "name": "雷山县" }, + { "code": "522635", "name": "麻江县" }, + { "code": "522636", "name": "丹寨县" } + ] + }, + { + "code": "5227", + "name": "黔南布依族苗族自治州", + "children": [ + { "code": "522701", "name": "都匀市" }, + { "code": "522702", "name": "福泉市" }, + { "code": "522722", "name": "荔波县" }, + { "code": "522723", "name": "贵定县" }, + { "code": "522725", "name": "瓮安县" }, + { "code": "522726", "name": "独山县" }, + { "code": "522727", "name": "平塘县" }, + { "code": "522728", "name": "罗甸县" }, + { "code": "522729", "name": "长顺县" }, + { "code": "522730", "name": "龙里县" }, + { "code": "522731", "name": "惠水县" }, + { "code": "522732", "name": "三都水族自治县" } + ] + } + ] + }, + { + "code": "53", + "name": "云南省", + "children": [ + { + "code": "5301", + "name": "昆明市", + "children": [ + { "code": "530102", "name": "五华区" }, + { "code": "530103", "name": "盘龙区" }, + { "code": "530111", "name": "官渡区" }, + { "code": "530112", "name": "西山区" }, + { "code": "530113", "name": "东川区" }, + { "code": "530114", "name": "呈贡区" }, + { "code": "530115", "name": "晋宁区" }, + { "code": "530124", "name": "富民县" }, + { "code": "530125", "name": "宜良县" }, + { "code": "530126", "name": "石林彝族自治县" }, + { "code": "530127", "name": "嵩明县" }, + { "code": "530128", "name": "禄劝彝族苗族自治县" }, + { "code": "530129", "name": "寻甸回族彝族自治县" }, + { "code": "530181", "name": "安宁市" } + ] + }, + { + "code": "5303", + "name": "曲靖市", + "children": [ + { "code": "530302", "name": "麒麟区" }, + { "code": "530303", "name": "沾益区" }, + { "code": "530304", "name": "马龙区" }, + { "code": "530322", "name": "陆良县" }, + { "code": "530323", "name": "师宗县" }, + { "code": "530324", "name": "罗平县" }, + { "code": "530325", "name": "富源县" }, + { "code": "530326", "name": "会泽县" }, + { "code": "530381", "name": "宣威市" } + ] + }, + { + "code": "5304", + "name": "玉溪市", + "children": [ + { "code": "530402", "name": "红塔区" }, + { "code": "530403", "name": "江川区" }, + { "code": "530423", "name": "通海县" }, + { "code": "530424", "name": "华宁县" }, + { "code": "530425", "name": "易门县" }, + { "code": "530426", "name": "峨山彝族自治县" }, + { "code": "530427", "name": "新平彝族傣族自治县" }, + { "code": "530428", "name": "元江哈尼族彝族傣族自治县" }, + { "code": "530481", "name": "澄江市" } + ] + }, + { + "code": "5305", + "name": "保山市", + "children": [ + { "code": "530502", "name": "隆阳区" }, + { "code": "530521", "name": "施甸县" }, + { "code": "530523", "name": "龙陵县" }, + { "code": "530524", "name": "昌宁县" }, + { "code": "530581", "name": "腾冲市" } + ] + }, + { + "code": "5306", + "name": "昭通市", + "children": [ + { "code": "530602", "name": "昭阳区" }, + { "code": "530621", "name": "鲁甸县" }, + { "code": "530622", "name": "巧家县" }, + { "code": "530623", "name": "盐津县" }, + { "code": "530624", "name": "大关县" }, + { "code": "530625", "name": "永善县" }, + { "code": "530626", "name": "绥江县" }, + { "code": "530627", "name": "镇雄县" }, + { "code": "530628", "name": "彝良县" }, + { "code": "530629", "name": "威信县" }, + { "code": "530681", "name": "水富市" } + ] + }, + { + "code": "5307", + "name": "丽江市", + "children": [ + { "code": "530702", "name": "古城区" }, + { "code": "530721", "name": "玉龙纳西族自治县" }, + { "code": "530722", "name": "永胜县" }, + { "code": "530723", "name": "华坪县" }, + { "code": "530724", "name": "宁蒗彝族自治县" } + ] + }, + { + "code": "5308", + "name": "普洱市", + "children": [ + { "code": "530802", "name": "思茅区" }, + { "code": "530821", "name": "宁洱哈尼族彝族自治县" }, + { "code": "530822", "name": "墨江哈尼族自治县" }, + { "code": "530823", "name": "景东彝族自治县" }, + { "code": "530824", "name": "景谷傣族彝族自治县" }, + { "code": "530825", "name": "镇沅彝族哈尼族拉祜族自治县" }, + { "code": "530826", "name": "江城哈尼族彝族自治县" }, + { "code": "530827", "name": "孟连傣族拉祜族佤族自治县" }, + { "code": "530828", "name": "澜沧拉祜族自治县" }, + { "code": "530829", "name": "西盟佤族自治县" } + ] + }, + { + "code": "5309", + "name": "临沧市", + "children": [ + { "code": "530902", "name": "临翔区" }, + { "code": "530921", "name": "凤庆县" }, + { "code": "530922", "name": "云县" }, + { "code": "530923", "name": "永德县" }, + { "code": "530924", "name": "镇康县" }, + { "code": "530925", "name": "双江拉祜族佤族布朗族傣族自治县" }, + { "code": "530926", "name": "耿马傣族佤族自治县" }, + { "code": "530927", "name": "沧源佤族自治县" } + ] + }, + { + "code": "5323", + "name": "楚雄彝族自治州", + "children": [ + { "code": "532301", "name": "楚雄市" }, + { "code": "532302", "name": "禄丰市" }, + { "code": "532322", "name": "双柏县" }, + { "code": "532323", "name": "牟定县" }, + { "code": "532324", "name": "南华县" }, + { "code": "532325", "name": "姚安县" }, + { "code": "532326", "name": "大姚县" }, + { "code": "532327", "name": "永仁县" }, + { "code": "532328", "name": "元谋县" }, + { "code": "532329", "name": "武定县" } + ] + }, + { + "code": "5325", + "name": "红河哈尼族彝族自治州", + "children": [ + { "code": "532501", "name": "个旧市" }, + { "code": "532502", "name": "开远市" }, + { "code": "532503", "name": "蒙自市" }, + { "code": "532504", "name": "弥勒市" }, + { "code": "532523", "name": "屏边苗族自治县" }, + { "code": "532524", "name": "建水县" }, + { "code": "532525", "name": "石屏县" }, + { "code": "532527", "name": "泸西县" }, + { "code": "532528", "name": "元阳县" }, + { "code": "532529", "name": "红河县" }, + { "code": "532530", "name": "金平苗族瑶族傣族自治县" }, + { "code": "532531", "name": "绿春县" }, + { "code": "532532", "name": "河口瑶族自治县" } + ] + }, + { + "code": "5326", + "name": "文山壮族苗族自治州", + "children": [ + { "code": "532601", "name": "文山市" }, + { "code": "532622", "name": "砚山县" }, + { "code": "532623", "name": "西畴县" }, + { "code": "532624", "name": "麻栗坡县" }, + { "code": "532625", "name": "马关县" }, + { "code": "532626", "name": "丘北县" }, + { "code": "532627", "name": "广南县" }, + { "code": "532628", "name": "富宁县" } + ] + }, + { + "code": "5328", + "name": "西双版纳傣族自治州", + "children": [ + { "code": "532801", "name": "景洪市" }, + { "code": "532822", "name": "勐海县" }, + { "code": "532823", "name": "勐腊县" } + ] + }, + { + "code": "5329", + "name": "大理白族自治州", + "children": [ + { "code": "532901", "name": "大理市" }, + { "code": "532922", "name": "漾濞彝族自治县" }, + { "code": "532923", "name": "祥云县" }, + { "code": "532924", "name": "宾川县" }, + { "code": "532925", "name": "弥渡县" }, + { "code": "532926", "name": "南涧彝族自治县" }, + { "code": "532927", "name": "巍山彝族回族自治县" }, + { "code": "532928", "name": "永平县" }, + { "code": "532929", "name": "云龙县" }, + { "code": "532930", "name": "洱源县" }, + { "code": "532931", "name": "剑川县" }, + { "code": "532932", "name": "鹤庆县" } + ] + }, + { + "code": "5331", + "name": "德宏傣族景颇族自治州", + "children": [ + { "code": "533102", "name": "瑞丽市" }, + { "code": "533103", "name": "芒市" }, + { "code": "533122", "name": "梁河县" }, + { "code": "533123", "name": "盈江县" }, + { "code": "533124", "name": "陇川县" } + ] + }, + { + "code": "5333", + "name": "怒江傈僳族自治州", + "children": [ + { "code": "533301", "name": "泸水市" }, + { "code": "533323", "name": "福贡县" }, + { "code": "533324", "name": "贡山独龙族怒族自治县" }, + { "code": "533325", "name": "兰坪白族普米族自治县" } + ] + }, + { + "code": "5334", + "name": "迪庆藏族自治州", + "children": [ + { "code": "533401", "name": "香格里拉市" }, + { "code": "533422", "name": "德钦县" }, + { "code": "533423", "name": "维西傈僳族自治县" } + ] + } + ] + }, + { + "code": "54", + "name": "西藏自治区", + "children": [ + { + "code": "5401", + "name": "拉萨市", + "children": [ + { "code": "540102", "name": "城关区" }, + { "code": "540103", "name": "堆龙德庆区" }, + { "code": "540104", "name": "达孜区" }, + { "code": "540121", "name": "林周县" }, + { "code": "540122", "name": "当雄县" }, + { "code": "540123", "name": "尼木县" }, + { "code": "540124", "name": "曲水县" }, + { "code": "540127", "name": "墨竹工卡县" }, + { "code": "540171", "name": "格尔木藏青工业园区" }, + { "code": "540172", "name": "拉萨经济技术开发区" }, + { "code": "540173", "name": "西藏文化旅游创意园区" }, + { "code": "540174", "name": "达孜工业园区" } + ] + }, + { + "code": "5402", + "name": "日喀则市", + "children": [ + { "code": "540202", "name": "桑珠孜区" }, + { "code": "540221", "name": "南木林县" }, + { "code": "540222", "name": "江孜县" }, + { "code": "540223", "name": "定日县" }, + { "code": "540224", "name": "萨迦县" }, + { "code": "540225", "name": "拉孜县" }, + { "code": "540226", "name": "昂仁县" }, + { "code": "540227", "name": "谢通门县" }, + { "code": "540228", "name": "白朗县" }, + { "code": "540229", "name": "仁布县" }, + { "code": "540230", "name": "康马县" }, + { "code": "540231", "name": "定结县" }, + { "code": "540232", "name": "仲巴县" }, + { "code": "540233", "name": "亚东县" }, + { "code": "540234", "name": "吉隆县" }, + { "code": "540235", "name": "聂拉木县" }, + { "code": "540236", "name": "萨嘎县" }, + { "code": "540237", "name": "岗巴县" } + ] + }, + { + "code": "5403", + "name": "昌都市", + "children": [ + { "code": "540302", "name": "卡若区" }, + { "code": "540321", "name": "江达县" }, + { "code": "540322", "name": "贡觉县" }, + { "code": "540323", "name": "类乌齐县" }, + { "code": "540324", "name": "丁青县" }, + { "code": "540325", "name": "察雅县" }, + { "code": "540326", "name": "八宿县" }, + { "code": "540327", "name": "左贡县" }, + { "code": "540328", "name": "芒康县" }, + { "code": "540329", "name": "洛隆县" }, + { "code": "540330", "name": "边坝县" } + ] + }, + { + "code": "5404", + "name": "林芝市", + "children": [ + { "code": "540402", "name": "巴宜区" }, + { "code": "540421", "name": "工布江达县" }, + { "code": "540422", "name": "米林县" }, + { "code": "540423", "name": "墨脱县" }, + { "code": "540424", "name": "波密县" }, + { "code": "540425", "name": "察隅县" }, + { "code": "540426", "name": "朗县" } + ] + }, + { + "code": "5405", + "name": "山南市", + "children": [ + { "code": "540502", "name": "乃东区" }, + { "code": "540521", "name": "扎囊县" }, + { "code": "540522", "name": "贡嘎县" }, + { "code": "540523", "name": "桑日县" }, + { "code": "540524", "name": "琼结县" }, + { "code": "540525", "name": "曲松县" }, + { "code": "540526", "name": "措美县" }, + { "code": "540527", "name": "洛扎县" }, + { "code": "540528", "name": "加查县" }, + { "code": "540529", "name": "隆子县" }, + { "code": "540530", "name": "错那县" }, + { "code": "540531", "name": "浪卡子县" } + ] + }, + { + "code": "5406", + "name": "那曲市", + "children": [ + { "code": "540602", "name": "色尼区" }, + { "code": "540621", "name": "嘉黎县" }, + { "code": "540622", "name": "比如县" }, + { "code": "540623", "name": "聂荣县" }, + { "code": "540624", "name": "安多县" }, + { "code": "540625", "name": "申扎县" }, + { "code": "540626", "name": "索县" }, + { "code": "540627", "name": "班戈县" }, + { "code": "540628", "name": "巴青县" }, + { "code": "540629", "name": "尼玛县" }, + { "code": "540630", "name": "双湖县" } + ] + }, + { + "code": "5425", + "name": "阿里地区", + "children": [ + { "code": "542521", "name": "普兰县" }, + { "code": "542522", "name": "札达县" }, + { "code": "542523", "name": "噶尔县" }, + { "code": "542524", "name": "日土县" }, + { "code": "542525", "name": "革吉县" }, + { "code": "542526", "name": "改则县" }, + { "code": "542527", "name": "措勤县" } + ] + } + ] + }, + { + "code": "61", + "name": "陕西省", + "children": [ + { + "code": "6101", + "name": "西安市", + "children": [ + { "code": "610102", "name": "新城区" }, + { "code": "610103", "name": "碑林区" }, + { "code": "610104", "name": "莲湖区" }, + { "code": "610111", "name": "灞桥区" }, + { "code": "610112", "name": "未央区" }, + { "code": "610113", "name": "雁塔区" }, + { "code": "610114", "name": "阎良区" }, + { "code": "610115", "name": "临潼区" }, + { "code": "610116", "name": "长安区" }, + { "code": "610117", "name": "高陵区" }, + { "code": "610118", "name": "鄠邑区" }, + { "code": "610122", "name": "蓝田县" }, + { "code": "610124", "name": "周至县" } + ] + }, + { + "code": "6102", + "name": "铜川市", + "children": [ + { "code": "610202", "name": "王益区" }, + { "code": "610203", "name": "印台区" }, + { "code": "610204", "name": "耀州区" }, + { "code": "610222", "name": "宜君县" } + ] + }, + { + "code": "6103", + "name": "宝鸡市", + "children": [ + { "code": "610302", "name": "渭滨区" }, + { "code": "610303", "name": "金台区" }, + { "code": "610304", "name": "陈仓区" }, + { "code": "610305", "name": "凤翔区" }, + { "code": "610323", "name": "岐山县" }, + { "code": "610324", "name": "扶风县" }, + { "code": "610326", "name": "眉县" }, + { "code": "610327", "name": "陇县" }, + { "code": "610328", "name": "千阳县" }, + { "code": "610329", "name": "麟游县" }, + { "code": "610330", "name": "凤县" }, + { "code": "610331", "name": "太白县" } + ] + }, + { + "code": "6104", + "name": "咸阳市", + "children": [ + { "code": "610402", "name": "秦都区" }, + { "code": "610403", "name": "杨陵区" }, + { "code": "610404", "name": "渭城区" }, + { "code": "610422", "name": "三原县" }, + { "code": "610423", "name": "泾阳县" }, + { "code": "610424", "name": "乾县" }, + { "code": "610425", "name": "礼泉县" }, + { "code": "610426", "name": "永寿县" }, + { "code": "610428", "name": "长武县" }, + { "code": "610429", "name": "旬邑县" }, + { "code": "610430", "name": "淳化县" }, + { "code": "610431", "name": "武功县" }, + { "code": "610481", "name": "兴平市" }, + { "code": "610482", "name": "彬州市" } + ] + }, + { + "code": "6105", + "name": "渭南市", + "children": [ + { "code": "610502", "name": "临渭区" }, + { "code": "610503", "name": "华州区" }, + { "code": "610522", "name": "潼关县" }, + { "code": "610523", "name": "大荔县" }, + { "code": "610524", "name": "合阳县" }, + { "code": "610525", "name": "澄城县" }, + { "code": "610526", "name": "蒲城县" }, + { "code": "610527", "name": "白水县" }, + { "code": "610528", "name": "富平县" }, + { "code": "610581", "name": "韩城市" }, + { "code": "610582", "name": "华阴市" } + ] + }, + { + "code": "6106", + "name": "延安市", + "children": [ + { "code": "610602", "name": "宝塔区" }, + { "code": "610603", "name": "安塞区" }, + { "code": "610621", "name": "延长县" }, + { "code": "610622", "name": "延川县" }, + { "code": "610625", "name": "志丹县" }, + { "code": "610626", "name": "吴起县" }, + { "code": "610627", "name": "甘泉县" }, + { "code": "610628", "name": "富县" }, + { "code": "610629", "name": "洛川县" }, + { "code": "610630", "name": "宜川县" }, + { "code": "610631", "name": "黄龙县" }, + { "code": "610632", "name": "黄陵县" }, + { "code": "610681", "name": "子长市" } + ] + }, + { + "code": "6107", + "name": "汉中市", + "children": [ + { "code": "610702", "name": "汉台区" }, + { "code": "610703", "name": "南郑区" }, + { "code": "610722", "name": "城固县" }, + { "code": "610723", "name": "洋县" }, + { "code": "610724", "name": "西乡县" }, + { "code": "610725", "name": "勉县" }, + { "code": "610726", "name": "宁强县" }, + { "code": "610727", "name": "略阳县" }, + { "code": "610728", "name": "镇巴县" }, + { "code": "610729", "name": "留坝县" }, + { "code": "610730", "name": "佛坪县" } + ] + }, + { + "code": "6108", + "name": "榆林市", + "children": [ + { "code": "610802", "name": "榆阳区" }, + { "code": "610803", "name": "横山区" }, + { "code": "610822", "name": "府谷县" }, + { "code": "610824", "name": "靖边县" }, + { "code": "610825", "name": "定边县" }, + { "code": "610826", "name": "绥德县" }, + { "code": "610827", "name": "米脂县" }, + { "code": "610828", "name": "佳县" }, + { "code": "610829", "name": "吴堡县" }, + { "code": "610830", "name": "清涧县" }, + { "code": "610831", "name": "子洲县" }, + { "code": "610881", "name": "神木市" } + ] + }, + { + "code": "6109", + "name": "安康市", + "children": [ + { "code": "610902", "name": "汉滨区" }, + { "code": "610921", "name": "汉阴县" }, + { "code": "610922", "name": "石泉县" }, + { "code": "610923", "name": "宁陕县" }, + { "code": "610924", "name": "紫阳县" }, + { "code": "610925", "name": "岚皋县" }, + { "code": "610926", "name": "平利县" }, + { "code": "610927", "name": "镇坪县" }, + { "code": "610929", "name": "白河县" }, + { "code": "610981", "name": "旬阳市" } + ] + }, + { + "code": "6110", + "name": "商洛市", + "children": [ + { "code": "611002", "name": "商州区" }, + { "code": "611021", "name": "洛南县" }, + { "code": "611022", "name": "丹凤县" }, + { "code": "611023", "name": "商南县" }, + { "code": "611024", "name": "山阳县" }, + { "code": "611025", "name": "镇安县" }, + { "code": "611026", "name": "柞水县" } + ] + } + ] + }, + { + "code": "62", + "name": "甘肃省", + "children": [ + { + "code": "6201", + "name": "兰州市", + "children": [ + { "code": "620102", "name": "城关区" }, + { "code": "620103", "name": "七里河区" }, + { "code": "620104", "name": "西固区" }, + { "code": "620105", "name": "安宁区" }, + { "code": "620111", "name": "红古区" }, + { "code": "620121", "name": "永登县" }, + { "code": "620122", "name": "皋兰县" }, + { "code": "620123", "name": "榆中县" }, + { "code": "620171", "name": "兰州新区" } + ] + }, + { + "code": "6202", + "name": "嘉峪关市", + "children": [ + { "code": "620201001", "name": "雄关街道" }, + { "code": "620201002", "name": "钢城街道" }, + { "code": "620201100", "name": "新城镇" }, + { "code": "620201101", "name": "峪泉镇" }, + { "code": "620201102", "name": "文殊镇" } + ] + }, + { + "code": "6203", + "name": "金昌市", + "children": [ + { "code": "620302", "name": "金川区" }, + { "code": "620321", "name": "永昌县" } + ] + }, + { + "code": "6204", + "name": "白银市", + "children": [ + { "code": "620402", "name": "白银区" }, + { "code": "620403", "name": "平川区" }, + { "code": "620421", "name": "靖远县" }, + { "code": "620422", "name": "会宁县" }, + { "code": "620423", "name": "景泰县" } + ] + }, + { + "code": "6205", + "name": "天水市", + "children": [ + { "code": "620502", "name": "秦州区" }, + { "code": "620503", "name": "麦积区" }, + { "code": "620521", "name": "清水县" }, + { "code": "620522", "name": "秦安县" }, + { "code": "620523", "name": "甘谷县" }, + { "code": "620524", "name": "武山县" }, + { "code": "620525", "name": "张家川回族自治县" } + ] + }, + { + "code": "6206", + "name": "武威市", + "children": [ + { "code": "620602", "name": "凉州区" }, + { "code": "620621", "name": "民勤县" }, + { "code": "620622", "name": "古浪县" }, + { "code": "620623", "name": "天祝藏族自治县" } + ] + }, + { + "code": "6207", + "name": "张掖市", + "children": [ + { "code": "620702", "name": "甘州区" }, + { "code": "620721", "name": "肃南裕固族自治县" }, + { "code": "620722", "name": "民乐县" }, + { "code": "620723", "name": "临泽县" }, + { "code": "620724", "name": "高台县" }, + { "code": "620725", "name": "山丹县" } + ] + }, + { + "code": "6208", + "name": "平凉市", + "children": [ + { "code": "620802", "name": "崆峒区" }, + { "code": "620821", "name": "泾川县" }, + { "code": "620822", "name": "灵台县" }, + { "code": "620823", "name": "崇信县" }, + { "code": "620825", "name": "庄浪县" }, + { "code": "620826", "name": "静宁县" }, + { "code": "620881", "name": "华亭市" } + ] + }, + { + "code": "6209", + "name": "酒泉市", + "children": [ + { "code": "620902", "name": "肃州区" }, + { "code": "620921", "name": "金塔县" }, + { "code": "620922", "name": "瓜州县" }, + { "code": "620923", "name": "肃北蒙古族自治县" }, + { "code": "620924", "name": "阿克塞哈萨克族自治县" }, + { "code": "620981", "name": "玉门市" }, + { "code": "620982", "name": "敦煌市" } + ] + }, + { + "code": "6210", + "name": "庆阳市", + "children": [ + { "code": "621002", "name": "西峰区" }, + { "code": "621021", "name": "庆城县" }, + { "code": "621022", "name": "环县" }, + { "code": "621023", "name": "华池县" }, + { "code": "621024", "name": "合水县" }, + { "code": "621025", "name": "正宁县" }, + { "code": "621026", "name": "宁县" }, + { "code": "621027", "name": "镇原县" } + ] + }, + { + "code": "6211", + "name": "定西市", + "children": [ + { "code": "621102", "name": "安定区" }, + { "code": "621121", "name": "通渭县" }, + { "code": "621122", "name": "陇西县" }, + { "code": "621123", "name": "渭源县" }, + { "code": "621124", "name": "临洮县" }, + { "code": "621125", "name": "漳县" }, + { "code": "621126", "name": "岷县" } + ] + }, + { + "code": "6212", + "name": "陇南市", + "children": [ + { "code": "621202", "name": "武都区" }, + { "code": "621221", "name": "成县" }, + { "code": "621222", "name": "文县" }, + { "code": "621223", "name": "宕昌县" }, + { "code": "621224", "name": "康县" }, + { "code": "621225", "name": "西和县" }, + { "code": "621226", "name": "礼县" }, + { "code": "621227", "name": "徽县" }, + { "code": "621228", "name": "两当县" } + ] + }, + { + "code": "6229", + "name": "临夏回族自治州", + "children": [ + { "code": "622901", "name": "临夏市" }, + { "code": "622921", "name": "临夏县" }, + { "code": "622922", "name": "康乐县" }, + { "code": "622923", "name": "永靖县" }, + { "code": "622924", "name": "广河县" }, + { "code": "622925", "name": "和政县" }, + { "code": "622926", "name": "东乡族自治县" }, + { "code": "622927", "name": "积石山保安族东乡族撒拉族自治县" } + ] + }, + { + "code": "6230", + "name": "甘南藏族自治州", + "children": [ + { "code": "623001", "name": "合作市" }, + { "code": "623021", "name": "临潭县" }, + { "code": "623022", "name": "卓尼县" }, + { "code": "623023", "name": "舟曲县" }, + { "code": "623024", "name": "迭部县" }, + { "code": "623025", "name": "玛曲县" }, + { "code": "623026", "name": "碌曲县" }, + { "code": "623027", "name": "夏河县" } + ] + } + ] + }, + { + "code": "63", + "name": "青海省", + "children": [ + { + "code": "6301", + "name": "西宁市", + "children": [ + { "code": "630102", "name": "城东区" }, + { "code": "630103", "name": "城中区" }, + { "code": "630104", "name": "城西区" }, + { "code": "630105", "name": "城北区" }, + { "code": "630106", "name": "湟中区" }, + { "code": "630121", "name": "大通回族土族自治县" }, + { "code": "630123", "name": "湟源县" } + ] + }, + { + "code": "6302", + "name": "海东市", + "children": [ + { "code": "630202", "name": "乐都区" }, + { "code": "630203", "name": "平安区" }, + { "code": "630222", "name": "民和回族土族自治县" }, + { "code": "630223", "name": "互助土族自治县" }, + { "code": "630224", "name": "化隆回族自治县" }, + { "code": "630225", "name": "循化撒拉族自治县" } + ] + }, + { + "code": "6322", + "name": "海北藏族自治州", + "children": [ + { "code": "632221", "name": "门源回族自治县" }, + { "code": "632222", "name": "祁连县" }, + { "code": "632223", "name": "海晏县" }, + { "code": "632224", "name": "刚察县" } + ] + }, + { + "code": "6323", + "name": "黄南藏族自治州", + "children": [ + { "code": "632301", "name": "同仁市" }, + { "code": "632322", "name": "尖扎县" }, + { "code": "632323", "name": "泽库县" }, + { "code": "632324", "name": "河南蒙古族自治县" } + ] + }, + { + "code": "6325", + "name": "海南藏族自治州", + "children": [ + { "code": "632521", "name": "共和县" }, + { "code": "632522", "name": "同德县" }, + { "code": "632523", "name": "贵德县" }, + { "code": "632524", "name": "兴海县" }, + { "code": "632525", "name": "贵南县" } + ] + }, + { + "code": "6326", + "name": "果洛藏族自治州", + "children": [ + { "code": "632621", "name": "玛沁县" }, + { "code": "632622", "name": "班玛县" }, + { "code": "632623", "name": "甘德县" }, + { "code": "632624", "name": "达日县" }, + { "code": "632625", "name": "久治县" }, + { "code": "632626", "name": "玛多县" } + ] + }, + { + "code": "6327", + "name": "玉树藏族自治州", + "children": [ + { "code": "632701", "name": "玉树市" }, + { "code": "632722", "name": "杂多县" }, + { "code": "632723", "name": "称多县" }, + { "code": "632724", "name": "治多县" }, + { "code": "632725", "name": "囊谦县" }, + { "code": "632726", "name": "曲麻莱县" } + ] + }, + { + "code": "6328", + "name": "海西蒙古族藏族自治州", + "children": [ + { "code": "632801", "name": "格尔木市" }, + { "code": "632802", "name": "德令哈市" }, + { "code": "632803", "name": "茫崖市" }, + { "code": "632821", "name": "乌兰县" }, + { "code": "632822", "name": "都兰县" }, + { "code": "632823", "name": "天峻县" }, + { "code": "632857", "name": "大柴旦行政委员会" } + ] + } + ] + }, + { + "code": "64", + "name": "宁夏回族自治区", + "children": [ + { + "code": "6401", + "name": "银川市", + "children": [ + { "code": "640104", "name": "兴庆区" }, + { "code": "640105", "name": "西夏区" }, + { "code": "640106", "name": "金凤区" }, + { "code": "640121", "name": "永宁县" }, + { "code": "640122", "name": "贺兰县" }, + { "code": "640181", "name": "灵武市" } + ] + }, + { + "code": "6402", + "name": "石嘴山市", + "children": [ + { "code": "640202", "name": "大武口区" }, + { "code": "640205", "name": "惠农区" }, + { "code": "640221", "name": "平罗县" } + ] + }, + { + "code": "6403", + "name": "吴忠市", + "children": [ + { "code": "640302", "name": "利通区" }, + { "code": "640303", "name": "红寺堡区" }, + { "code": "640323", "name": "盐池县" }, + { "code": "640324", "name": "同心县" }, + { "code": "640381", "name": "青铜峡市" } + ] + }, + { + "code": "6404", + "name": "固原市", + "children": [ + { "code": "640402", "name": "原州区" }, + { "code": "640422", "name": "西吉县" }, + { "code": "640423", "name": "隆德县" }, + { "code": "640424", "name": "泾源县" }, + { "code": "640425", "name": "彭阳县" } + ] + }, + { + "code": "6405", + "name": "中卫市", + "children": [ + { "code": "640502", "name": "沙坡头区" }, + { "code": "640521", "name": "中宁县" }, + { "code": "640522", "name": "海原县" } + ] + } + ] + }, + { + "code": "65", + "name": "新疆维吾尔自治区", + "children": [ + { + "code": "6501", + "name": "乌鲁木齐市", + "children": [ + { "code": "650102", "name": "天山区" }, + { "code": "650103", "name": "沙依巴克区" }, + { "code": "650104", "name": "新市区" }, + { "code": "650105", "name": "水磨沟区" }, + { "code": "650106", "name": "头屯河区" }, + { "code": "650107", "name": "达坂城区" }, + { "code": "650109", "name": "米东区" }, + { "code": "650121", "name": "乌鲁木齐县" } + ] + }, + { + "code": "6502", + "name": "克拉玛依市", + "children": [ + { "code": "650202", "name": "独山子区" }, + { "code": "650203", "name": "克拉玛依区" }, + { "code": "650204", "name": "白碱滩区" }, + { "code": "650205", "name": "乌尔禾区" } + ] + }, + { + "code": "6504", + "name": "吐鲁番市", + "children": [ + { "code": "650402", "name": "高昌区" }, + { "code": "650421", "name": "鄯善县" }, + { "code": "650422", "name": "托克逊县" } + ] + }, + { + "code": "6505", + "name": "哈密市", + "children": [ + { "code": "650502", "name": "伊州区" }, + { "code": "650521", "name": "巴里坤哈萨克自治县" }, + { "code": "650522", "name": "伊吾县" } + ] + }, + { + "code": "6523", + "name": "昌吉回族自治州", + "children": [ + { "code": "652301", "name": "昌吉市" }, + { "code": "652302", "name": "阜康市" }, + { "code": "652323", "name": "呼图壁县" }, + { "code": "652324", "name": "玛纳斯县" }, + { "code": "652325", "name": "奇台县" }, + { "code": "652327", "name": "吉木萨尔县" }, + { "code": "652328", "name": "木垒哈萨克自治县" } + ] + }, + { + "code": "6527", + "name": "博尔塔拉蒙古自治州", + "children": [ + { "code": "652701", "name": "博乐市" }, + { "code": "652702", "name": "阿拉山口市" }, + { "code": "652722", "name": "精河县" }, + { "code": "652723", "name": "温泉县" } + ] + }, + { + "code": "6528", + "name": "巴音郭楞蒙古自治州", + "children": [ + { "code": "652801", "name": "库尔勒市" }, + { "code": "652822", "name": "轮台县" }, + { "code": "652823", "name": "尉犁县" }, + { "code": "652824", "name": "若羌县" }, + { "code": "652825", "name": "且末县" }, + { "code": "652826", "name": "焉耆回族自治县" }, + { "code": "652827", "name": "和静县" }, + { "code": "652828", "name": "和硕县" }, + { "code": "652829", "name": "博湖县" }, + { "code": "652871", "name": "库尔勒经济技术开发区" } + ] + }, + { + "code": "6529", + "name": "阿克苏地区", + "children": [ + { "code": "652901", "name": "阿克苏市" }, + { "code": "652902", "name": "库车市" }, + { "code": "652922", "name": "温宿县" }, + { "code": "652924", "name": "沙雅县" }, + { "code": "652925", "name": "新和县" }, + { "code": "652926", "name": "拜城县" }, + { "code": "652927", "name": "乌什县" }, + { "code": "652928", "name": "阿瓦提县" }, + { "code": "652929", "name": "柯坪县" } + ] + }, + { + "code": "6530", + "name": "克孜勒苏柯尔克孜自治州", + "children": [ + { "code": "653001", "name": "阿图什市" }, + { "code": "653022", "name": "阿克陶县" }, + { "code": "653023", "name": "阿合奇县" }, + { "code": "653024", "name": "乌恰县" } + ] + }, + { + "code": "6531", + "name": "喀什地区", + "children": [ + { "code": "653101", "name": "喀什市" }, + { "code": "653121", "name": "疏附县" }, + { "code": "653122", "name": "疏勒县" }, + { "code": "653123", "name": "英吉沙县" }, + { "code": "653124", "name": "泽普县" }, + { "code": "653125", "name": "莎车县" }, + { "code": "653126", "name": "叶城县" }, + { "code": "653127", "name": "麦盖提县" }, + { "code": "653128", "name": "岳普湖县" }, + { "code": "653129", "name": "伽师县" }, + { "code": "653130", "name": "巴楚县" }, + { "code": "653131", "name": "塔什库尔干塔吉克自治县" } + ] + }, + { + "code": "6532", + "name": "和田地区", + "children": [ + { "code": "653201", "name": "和田市" }, + { "code": "653221", "name": "和田县" }, + { "code": "653222", "name": "墨玉县" }, + { "code": "653223", "name": "皮山县" }, + { "code": "653224", "name": "洛浦县" }, + { "code": "653225", "name": "策勒县" }, + { "code": "653226", "name": "于田县" }, + { "code": "653227", "name": "民丰县" } + ] + }, + { + "code": "6540", + "name": "伊犁哈萨克自治州", + "children": [ + { "code": "654002", "name": "伊宁市" }, + { "code": "654003", "name": "奎屯市" }, + { "code": "654004", "name": "霍尔果斯市" }, + { "code": "654021", "name": "伊宁县" }, + { "code": "654022", "name": "察布查尔锡伯自治县" }, + { "code": "654023", "name": "霍城县" }, + { "code": "654024", "name": "巩留县" }, + { "code": "654025", "name": "新源县" }, + { "code": "654026", "name": "昭苏县" }, + { "code": "654027", "name": "特克斯县" }, + { "code": "654028", "name": "尼勒克县" } + ] + }, + { + "code": "6542", + "name": "塔城地区", + "children": [ + { "code": "654201", "name": "塔城市" }, + { "code": "654202", "name": "乌苏市" }, + { "code": "654203", "name": "沙湾市" }, + { "code": "654221", "name": "额敏县" }, + { "code": "654224", "name": "托里县" }, + { "code": "654225", "name": "裕民县" }, + { "code": "654226", "name": "和布克赛尔蒙古自治县" } + ] + }, + { + "code": "6543", + "name": "阿勒泰地区", + "children": [ + { "code": "654301", "name": "阿勒泰市" }, + { "code": "654321", "name": "布尔津县" }, + { "code": "654322", "name": "富蕴县" }, + { "code": "654323", "name": "福海县" }, + { "code": "654324", "name": "哈巴河县" }, + { "code": "654325", "name": "青河县" }, + { "code": "654326", "name": "吉木乃县" } + ] + }, + { + "code": "6590", + "name": "自治区直辖县级行政区划", + "children": [ + { "code": "659001", "name": "石河子市" }, + { "code": "659002", "name": "阿拉尔市" }, + { "code": "659003", "name": "图木舒克市" }, + { "code": "659004", "name": "五家渠市" }, + { "code": "659005", "name": "北屯市" }, + { "code": "659006", "name": "铁门关市" }, + { "code": "659007", "name": "双河市" }, + { "code": "659008", "name": "可克达拉市" }, + { "code": "659009", "name": "昆玉市" }, + { "code": "659010", "name": "胡杨河市" }, + { "code": "659011", "name": "新星市" } + ] + } + ] + } +] diff --git a/src/plugins/distpicker/demo/base.vue b/src/plugins/distpicker/demo/base.vue new file mode 100644 index 0000000..75e18d1 --- /dev/null +++ b/src/plugins/distpicker/demo/base.vue @@ -0,0 +1,10 @@ + + + diff --git a/src/plugins/echarts/config.ts b/src/plugins/echarts/config.ts new file mode 100644 index 0000000..13ea798 --- /dev/null +++ b/src/plugins/echarts/config.ts @@ -0,0 +1,16 @@ +import type { ModuleConfig } from "/@/cool"; +import VueECharts from "vue-echarts"; + +export default (): ModuleConfig => { + return { + order: 100, + label: "ECharts 图表", + description: "echarts、vue-echarts 配置", + author: "COOL", + version: "1.0.0", + updateTime: "2024-07-22", + install(app) { + app.component("v-chart", VueECharts); + } + }; +}; diff --git a/src/plugins/editor-monaco/components/monaco.vue b/src/plugins/editor-monaco/components/monaco.vue new file mode 100644 index 0000000..dab02b6 --- /dev/null +++ b/src/plugins/editor-monaco/components/monaco.vue @@ -0,0 +1,216 @@ + + + + + diff --git a/src/plugins/editor-monaco/config.ts b/src/plugins/editor-monaco/config.ts new file mode 100644 index 0000000..d510f94 --- /dev/null +++ b/src/plugins/editor-monaco/config.ts @@ -0,0 +1,23 @@ +import { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return { + label: "代码编辑器", + description: "基于 monaco 封装的代码编辑器", + author: "COOL", + version: "1.1.0", + updateTime: "2024-06-01", + demo: [ + { + name: "基础用法", + component: () => import("./demo/base.vue") + } + ], + + components: [ + // 依赖偏大,现用于“ai编码页面”,如不需要请注释组件 + // 代码编辑器 https://www.npmjs.com/package/monaco-editor + () => import("./components/monaco.vue") + ] + }; +}; diff --git a/src/plugins/editor-monaco/demo/base.vue b/src/plugins/editor-monaco/demo/base.vue new file mode 100644 index 0000000..a6052b7 --- /dev/null +++ b/src/plugins/editor-monaco/demo/base.vue @@ -0,0 +1,28 @@ + + + diff --git a/src/plugins/editor-monaco/index.ts b/src/plugins/editor-monaco/index.ts new file mode 100644 index 0000000..b597b4d --- /dev/null +++ b/src/plugins/editor-monaco/index.ts @@ -0,0 +1,2 @@ +export * from "./utils/declare"; +export * from "./utils/theme"; diff --git a/src/plugins/editor-monaco/utils/config.ts b/src/plugins/editor-monaco/utils/config.ts new file mode 100644 index 0000000..dfc3d01 --- /dev/null +++ b/src/plugins/editor-monaco/utils/config.ts @@ -0,0 +1,28 @@ +import { languages } from "monaco-editor"; + +languages.typescript.typescriptDefaults.setEagerModelSync(true); + +languages.typescript.typescriptDefaults.setCompilerOptions({ + target: languages.typescript.ScriptTarget.ES2018, + allowNonTsExtensions: true, + experimentalDecorators: true, + jsx: languages.typescript.JsxEmit.Preserve, + esModuleInterop: true, + noEmit: true, + allowSyntheticDefaultImports: true, + noImplicitAny: false, + emitDecoratorMetadata: true, + inlineSourceMap: true, + noImplicitThis: true, + noUnusedLocals: false, + stripInternal: true, + skipLibCheck: true, + resolveJsonModule: true, + pretty: true, + declaration: true +}); + +languages.typescript.typescriptDefaults.setDiagnosticsOptions({ + noSemanticValidation: true, + noSyntaxValidation: true +}); diff --git a/src/plugins/editor-monaco/utils/declare.ts b/src/plugins/editor-monaco/utils/declare.ts new file mode 100644 index 0000000..08d437b --- /dev/null +++ b/src/plugins/editor-monaco/utils/declare.ts @@ -0,0 +1,32 @@ +import { languages } from "monaco-editor"; +import { keys } from "lodash-es"; + +export function addDeclare({ path, content }: { path: string; content: string }) { + const defaults = languages.typescript.typescriptDefaults; + + const filePath = `file:///node_modules/${path}`; + const loaded = defaults.getExtraLibs(); + const libs = keys(loaded).map((e) => { + return { + filePath: e, + content: loaded[e].content + }; + }); + + const item = libs.find((e) => e.filePath.includes(path)); + try { + if (item) { + item.content = content; + defaults.setExtraLibs(libs); + } else { + defaults.addExtraLib(content, filePath); + } + } catch (err) { + console.error(err); + } +} + +export function clearDeclare() { + const defaults = languages.typescript.typescriptDefaults; + defaults.setExtraLibs([]); +} diff --git a/src/plugins/editor-monaco/utils/format.ts b/src/plugins/editor-monaco/utils/format.ts new file mode 100644 index 0000000..f9791c8 --- /dev/null +++ b/src/plugins/editor-monaco/utils/format.ts @@ -0,0 +1,65 @@ +import { languages } from "monaco-editor"; +import prettier from "prettier/standalone"; +import pluginHtml from "prettier/plugins/html"; +import pluginTypescript from "prettier/plugins/typescript"; +import pluginPostcss from "prettier/plugins/postcss"; +import pluginEstree from "prettier/plugins/estree"; + +const options: { [key: string]: { parser: string; plugins: any[] } } = { + html: { + parser: "html", + plugins: [pluginHtml, pluginTypescript, pluginPostcss, pluginEstree] + }, + typescript: { + parser: "typescript", + plugins: [pluginTypescript, pluginEstree] + }, + css: { + parser: "css", + plugins: [pluginPostcss] + }, + scss: { + parser: "scss", + plugins: [pluginPostcss] + } +}; + +export function useFormat() { + function register() { + for (const i in options) { + languages.registerDocumentFormattingEditProvider(i, { + async provideDocumentFormattingEdits(model) { + let text = model.getValue(); + + const parser = options[i].parser; + + try { + text = await prettier.format(text, { + parser, + plugins: options[i].plugins, + semi: true, + printWidth: 100, + tabWidth: parser == "html" ? 4 : 2, + useTabs: parser == "html", + singleQuote: true, + trailingComma: "none" + }); + } catch (err) { + // ... + } + + return [ + { + range: model.getFullModelRange(), + text + } + ]; + } + }); + } + } + + return { + register + }; +} diff --git a/src/plugins/editor-monaco/utils/theme.ts b/src/plugins/editor-monaco/utils/theme.ts new file mode 100644 index 0000000..03b32de --- /dev/null +++ b/src/plugins/editor-monaco/utils/theme.ts @@ -0,0 +1,364 @@ +import { editor } from "monaco-editor"; + +defineTheme("default", { + base: "vs", + inherit: true, + rules: [ + { + background: "ffffff", + token: "" + }, + { + foreground: "6a737d", + token: "comment" + }, + { + foreground: "6a737d", + token: "punctuation.definition.comment" + }, + { + foreground: "6a737d", + token: "string.comment" + }, + { + foreground: "005cc5", + token: "constant" + }, + { + foreground: "005cc5", + token: "entity.name.constant" + }, + { + foreground: "005cc5", + token: "variable.other.constant" + }, + { + foreground: "005cc5", + token: "variable.language" + }, + { + foreground: "6f42c1", + token: "entity" + }, + { + foreground: "6f42c1", + token: "entity.name" + }, + { + foreground: "24292e", + token: "variable.parameter.function" + }, + { + foreground: "22863a", + token: "entity.name.tag" + }, + { + foreground: "d73a49", + token: "keyword" + }, + { + foreground: "d73a49", + token: "storage" + }, + { + foreground: "d73a49", + token: "storage.type" + }, + { + foreground: "24292e", + token: "storage.modifier.package" + }, + { + foreground: "24292e", + token: "storage.modifier.import" + }, + { + foreground: "24292e", + token: "storage.type.java" + }, + { + foreground: "032f62", + token: "string" + }, + { + foreground: "032f62", + token: "punctuation.definition.string" + }, + { + foreground: "032f62", + token: "string punctuation.section.embedded source" + }, + { + foreground: "005cc5", + token: "support" + }, + { + foreground: "005cc5", + token: "meta.property-name" + }, + { + foreground: "e36209", + token: "variable" + }, + { + foreground: "24292e", + token: "variable.other" + }, + { + foreground: "b31d28", + fontStyle: "bold italic underline", + token: "invalid.broken" + }, + { + foreground: "b31d28", + fontStyle: "bold italic underline", + token: "invalid.deprecated" + }, + { + foreground: "fafbfc", + background: "b31d28", + fontStyle: "italic underline", + token: "invalid.illegal" + }, + { + foreground: "fafbfc", + background: "d73a49", + fontStyle: "italic underline", + token: "carriage-return" + }, + { + foreground: "b31d28", + fontStyle: "bold italic underline", + token: "invalid.unimplemented" + }, + { + foreground: "b31d28", + token: "message.error" + }, + { + foreground: "24292e", + token: "string source" + }, + { + foreground: "005cc5", + token: "string variable" + }, + { + foreground: "032f62", + token: "source.regexp" + }, + { + foreground: "032f62", + token: "string.regexp" + }, + { + foreground: "032f62", + token: "string.regexp.character-class" + }, + { + foreground: "032f62", + token: "string.regexp constant.character.escape" + }, + { + foreground: "032f62", + token: "string.regexp source.ruby.embedded" + }, + { + foreground: "032f62", + token: "string.regexp string.regexp.arbitrary-repitition" + }, + { + foreground: "22863a", + fontStyle: "bold", + token: "string.regexp constant.character.escape" + }, + { + foreground: "005cc5", + token: "support.constant" + }, + { + foreground: "005cc5", + token: "support.variable" + }, + { + foreground: "005cc5", + token: "meta.module-reference" + }, + { + foreground: "735c0f", + token: "markup.list" + }, + { + foreground: "005cc5", + fontStyle: "bold", + token: "markup.heading" + }, + { + foreground: "005cc5", + fontStyle: "bold", + token: "markup.heading entity.name" + }, + { + foreground: "22863a", + token: "markup.quote" + }, + { + foreground: "24292e", + fontStyle: "italic", + token: "markup.italic" + }, + { + foreground: "24292e", + fontStyle: "bold", + token: "markup.bold" + }, + { + foreground: "005cc5", + token: "markup.raw" + }, + { + foreground: "b31d28", + background: "ffeef0", + token: "markup.deleted" + }, + { + foreground: "b31d28", + background: "ffeef0", + token: "meta.diff.header.from-file" + }, + { + foreground: "b31d28", + background: "ffeef0", + token: "punctuation.definition.deleted" + }, + { + foreground: "22863a", + background: "f0fff4", + token: "markup.inserted" + }, + { + foreground: "22863a", + background: "f0fff4", + token: "meta.diff.header.to-file" + }, + { + foreground: "22863a", + background: "f0fff4", + token: "punctuation.definition.inserted" + }, + { + foreground: "e36209", + background: "ffebda", + token: "markup.changed" + }, + { + foreground: "e36209", + background: "ffebda", + token: "punctuation.definition.changed" + }, + { + foreground: "f6f8fa", + background: "005cc5", + token: "markup.ignored" + }, + { + foreground: "f6f8fa", + background: "005cc5", + token: "markup.untracked" + }, + { + foreground: "6f42c1", + fontStyle: "bold", + token: "meta.diff.range" + }, + { + foreground: "005cc5", + token: "meta.diff.header" + }, + { + foreground: "005cc5", + fontStyle: "bold", + token: "meta.separator" + }, + { + foreground: "005cc5", + token: "meta.output" + }, + { + foreground: "586069", + token: "brackethighlighter.tag" + }, + { + foreground: "586069", + token: "brackethighlighter.curly" + }, + { + foreground: "586069", + token: "brackethighlighter.round" + }, + { + foreground: "586069", + token: "brackethighlighter.square" + }, + { + foreground: "586069", + token: "brackethighlighter.angle" + }, + { + foreground: "586069", + token: "brackethighlighter.quote" + }, + { + foreground: "b31d28", + token: "brackethighlighter.unmatched" + }, + { + foreground: "b31d28", + token: "sublimelinter.mark.error" + }, + { + foreground: "e36209", + token: "sublimelinter.mark.warning" + }, + { + foreground: "959da5", + token: "sublimelinter.gutter-mark" + }, + { + foreground: "032f62", + fontStyle: "underline", + token: "constant.other.reference.link" + }, + { + foreground: "032f62", + fontStyle: "underline", + token: "string.other.link" + } + ], + colors: { + "editor.foreground": "#24292e", + "editor.background": "#ffffff", + "editor.selectionBackground": "#c8c8fa", + "editor.inactiveSelectionBackground": "#fafbfc", + "editor.lineHighlightBackground": "#fafbfc", + "editorCursor.foreground": "#24292e", + "editorWhitespace.foreground": "#959da5", + "editorIndentGuide.background": "#959da5", + "editorIndentGuide.activeBackground": "#24292e", + "editor.selectionHighlightBorder": "#fafbfc" + } +}); + +defineTheme("ai-code--dark", { + base: "vs-dark", + inherit: true, + rules: [], + colors: { + "editor.background": "#0f151e", + "editor.inactiveSelectionBackground": "#0f151e" + } +}); + +export function defineTheme(name: string, data: editor.IStandaloneThemeData) { + editor.defineTheme(name, data); +} diff --git a/src/plugins/editor-monaco/utils/worker.ts b/src/plugins/editor-monaco/utils/worker.ts new file mode 100644 index 0000000..80a2da1 --- /dev/null +++ b/src/plugins/editor-monaco/utils/worker.ts @@ -0,0 +1,23 @@ +import editorWorker from "monaco-editor/esm/vs/editor/editor.worker?worker"; +import cssWorker from "monaco-editor/esm/vs/language/css/css.worker?worker"; +import htmlWorker from "monaco-editor/esm/vs/language/html/html.worker?worker"; +import jsonWorker from "monaco-editor/esm/vs/language/json/json.worker?worker"; +import tsWorker from "monaco-editor/esm/vs/language/typescript/ts.worker?worker"; + +self.MonacoEnvironment = { + getWorker(_: any, label: string) { + if (label === "json") { + return new jsonWorker(); + } + if (label === "css" || label === "scss" || label === "less") { + return new cssWorker(); + } + if (label === "html" || label === "handlebars" || label === "razor") { + return new htmlWorker(); + } + if (label === "typescript" || label === "javascript") { + return new tsWorker(); + } + return new editorWorker(); + } +}; diff --git a/src/plugins/editor-preview/components/preview.vue b/src/plugins/editor-preview/components/preview.vue new file mode 100644 index 0000000..41e1d3c --- /dev/null +++ b/src/plugins/editor-preview/components/preview.vue @@ -0,0 +1,215 @@ + + + + + diff --git a/src/plugins/editor-preview/config.ts b/src/plugins/editor-preview/config.ts new file mode 100644 index 0000000..ce37b67 --- /dev/null +++ b/src/plugins/editor-preview/config.ts @@ -0,0 +1,19 @@ +import type { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return { + label: "编辑器内容预览", + description: "基于 monaco、wang 等编辑器的内容预览组件", + author: "COOL", + version: "1.0.1", + updateTime: "2024-02-27", + demo: [ + { + name: "基础用法", + component: () => import("./demo/base.vue") + } + ], + + components: [() => import("./components/preview.vue")] + }; +}; diff --git a/src/plugins/editor-preview/demo/base.vue b/src/plugins/editor-preview/demo/base.vue new file mode 100644 index 0000000..57d3bc6 --- /dev/null +++ b/src/plugins/editor-preview/demo/base.vue @@ -0,0 +1,29 @@ + + + diff --git a/src/plugins/editor-wang/components/wang.vue b/src/plugins/editor-wang/components/wang.vue new file mode 100644 index 0000000..b46af3c --- /dev/null +++ b/src/plugins/editor-wang/components/wang.vue @@ -0,0 +1,255 @@ + + + + + diff --git a/src/plugins/editor-wang/config.ts b/src/plugins/editor-wang/config.ts new file mode 100644 index 0000000..03ba78b --- /dev/null +++ b/src/plugins/editor-wang/config.ts @@ -0,0 +1,19 @@ +import type { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return { + label: "富文本编辑器", + description: "基于 wangEditor 封装的富文本编辑器", // https://www.wangeditor.com + author: "COOL", + version: "1.0.0", + updateTime: "2024-02-01", + demo: [ + { + name: "基础用法", + component: () => import("./demo/base.vue") + } + ], + + components: [() => import("./components/wang.vue")] + }; +}; diff --git a/src/plugins/editor-wang/demo/base.vue b/src/plugins/editor-wang/demo/base.vue new file mode 100644 index 0000000..d252552 --- /dev/null +++ b/src/plugins/editor-wang/demo/base.vue @@ -0,0 +1,11 @@ + + + diff --git a/src/plugins/element-ui/config.ts b/src/plugins/element-ui/config.ts new file mode 100644 index 0000000..c7ca379 --- /dev/null +++ b/src/plugins/element-ui/config.ts @@ -0,0 +1,17 @@ +import type { ModuleConfig } from "/@/cool"; +import ElementPlus from "element-plus"; +import "./css/index.scss"; + +export default (): ModuleConfig => { + return { + order: 100, + label: "Element Ui", + description: "Element Plus 变量、样式配置", + author: "COOL", + version: "1.0.0", + updateTime: "2024-07-22", + install(app) { + app.use(ElementPlus); + } + }; +}; diff --git a/src/plugins/element-ui/css/index.scss b/src/plugins/element-ui/css/index.scss new file mode 100644 index 0000000..bc5fd60 --- /dev/null +++ b/src/plugins/element-ui/css/index.scss @@ -0,0 +1,38 @@ +@forward "element-plus/theme-chalk/src/common/var.scss" with ( + $scrollbar: () +); + +@use "element-plus/theme-chalk/src/index.scss" as *; + +// Element-plus +.el-input-number { + &__decrease, + &__increase { + border: 0 !important; + background-color: transparent; + } +} + +.el-message, +.el-overlay.is-message-box { + z-index: 10000 !important; +} + +.el-message { + &.el-message--success, + &.el-message--error, + &.el-message--info, + &.el-message--warning { + border-radius: 6px; + } + + &__icon { + font-size: 18px; + } +} + +@media only screen and (max-width: 768px) { + .el-message-box { + width: 90% !important; + } +} diff --git a/src/plugins/excel/components/export-btn.tsx b/src/plugins/excel/components/export-btn.tsx new file mode 100644 index 0000000..5bc7a15 --- /dev/null +++ b/src/plugins/excel/components/export-btn.tsx @@ -0,0 +1,206 @@ +import { defineComponent, type PropType, ref } from "vue"; +import { useCrud, useForm } from "@cool-vue/crud"; +import { ElMessage } from "element-plus"; +import dayjs from "dayjs"; +import { isEmpty, orderBy } from "lodash-es"; +import { export_json_to_excel } from "../utils"; +import { deepFind } from "/$/dict/utils"; + +export default defineComponent({ + name: "cl-export-btn", + + props: { + filename: [Function, String], + autoWidth: { + type: Boolean, + default: true + }, + bookType: { + type: String, + default: "xlsx" + }, + header: Array, + columns: { + type: Array as PropType + }, + data: [Function, Array], + maxExportLimit: Number // 最大导出条数,不传或者小于等于0为不限制 + }, + + setup(props, { slots }) { + // crud + const Crud = useCrud(); + + // 表单 + const Form = useForm(); + + // 加载状态 + const loading = ref(false); + + // 获取表头数据 + async function getHeader(columns: any[], fields: any[]) { + return columns.filter((e) => !e.hidden && fields.includes(e.prop)).map((e) => e.label); + } + + // 获取表格数据 + async function getData(): Promise { + const params = { + ...Crud.value?.paramsReplace(Crud.value.params), + maxExportLimit: props.maxExportLimit, + isExport: true + }; + + if (typeof props.data === "function") { + return props.data(params); + } else { + if (props.data) { + return props.data; + } else { + if (Crud.value?.service) { + return Crud.value?.service + .page(params) + .then((res) => { + return res.list.map((e, i) => { + for (const k in e) { + const col = props.columns?.find((c) => c.prop == k); + + if (col) { + // 格式化 + if (col.formatter) { + e[k] = col.formatter(e, col, e[k], i); + } + + // 字典 + if (col.dict) { + // @ts-ignore + e[k] = deepFind(e[k], col.dict)?.label || e[k]; + } + } + } + + return e; + }); + }) + .catch((err) => { + ElMessage.error(err.message); + return []; + }); + } else { + console.error("useCrud 中未设置 service 参数"); + return []; + } + } + } + } + + // 获取文件名 + async function getFileName() { + if (typeof props.filename === "function") { + return await props?.filename(); + } else { + return props.filename || `报表(${dayjs().format("YYYY-MM-DD HH_mm_ss")})`; + } + } + + // 导出 + async function toExport(columns: ClTable.Column[]) { + // 加载 + loading.value = true; + + // 字段 + const fields = columns.map((e) => e.prop).filter(Boolean); + + // 表头 + const header = await getHeader(columns, fields); + + // 数据 + let data = await getData(); + + if (!data) { + loading.value = false; + return ElMessage.error("导出数据异常"); + } + + // 文件名 + const filename = await getFileName(); + + // 过滤 + data = data.map((d) => fields.map((f) => d[f])); + + // 导出 excel + export_json_to_excel({ + header, + data, + filename, + autoWidth: props.autoWidth, + bookType: props.bookType + }); + + loading.value = false; + } + + function open() { + if (!props.columns) { + return console.error(" columns is required"); + } + + // 表格列 + const columns = orderBy(props.columns, "orderNum", "asc").filter( + (e) => + !( + e.hidden === true || + ["selection", "expand", "index", "op"].includes(e.type) || + e.filterExport || + e["filter-export"] + ) + ); + + Form.value?.open({ + title: "导出", + width: "600px", + props: { + labelPosition: "top" + }, + form: { + checked: columns.map((e) => e.prop) + }, + items: [ + { + label: "选择列", + prop: "checked", + component: { + name: "el-checkbox-group", + options: columns.map((e) => { + return { + label: String(e.label), + value: e.prop + }; + }) + } + } + ], + on: { + submit(data, { close, done }) { + if (isEmpty(data.checked)) { + ElMessage.warning("请先选择要导出的列"); + done(); + } else { + toExport(columns.filter((e) => data.checked.includes(e.prop))); + close(); + } + } + } + }); + } + + return () => { + return ( + + {slots.default ? slots.default() : "导出"} + + + + ); + }; + } +}); diff --git a/src/plugins/excel/components/import-btn.vue b/src/plugins/excel/components/import-btn.vue new file mode 100644 index 0000000..4896770 --- /dev/null +++ b/src/plugins/excel/components/import-btn.vue @@ -0,0 +1,385 @@ + + + + + diff --git a/src/plugins/excel/config.ts b/src/plugins/excel/config.ts new file mode 100644 index 0000000..93df13a --- /dev/null +++ b/src/plugins/excel/config.ts @@ -0,0 +1,22 @@ +import type { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return { + label: "Excel", + description: "表格的导入、导出组件", + author: "COOL", + version: "1.0.0", + updateTime: "2024-02-01", + demo: [ + { + name: "基础用法", + component: () => import("./demo/base.vue") + } + ], + + components: [ + () => import("./components/import-btn.vue"), + () => import("./components/export-btn") + ] + }; +}; diff --git a/src/plugins/excel/demo/base.vue b/src/plugins/excel/demo/base.vue new file mode 100644 index 0000000..b2af212 --- /dev/null +++ b/src/plugins/excel/demo/base.vue @@ -0,0 +1,80 @@ + + + diff --git a/src/plugins/excel/utils/index.ts b/src/plugins/excel/utils/index.ts new file mode 100644 index 0000000..b2f7adf --- /dev/null +++ b/src/plugins/excel/utils/index.ts @@ -0,0 +1,227 @@ +/* eslint-disable */ +// @ts-nocheck +import { saveAs } from "file-saver"; +import * as XLSX from "xlsx"; + +function generateArray(table) { + var out = []; + var rows = table.querySelectorAll("tr"); + var ranges = []; + for (var R = 0; R < rows.length; ++R) { + var outRow = []; + var row = rows[R]; + var columns = row.querySelectorAll("td"); + for (var C = 0; C < columns.length; ++C) { + var cell = columns[C]; + var colspan = cell.getAttribute("colspan"); + var rowspan = cell.getAttribute("rowspan"); + var cellValue = cell.innerText; + if (cellValue !== "" && cellValue == +cellValue) cellValue = +cellValue; + + //Skip ranges + ranges.forEach(function (range) { + if ( + R >= range.s.r && + R <= range.e.r && + outRow.length >= range.s.c && + outRow.length <= range.e.c + ) { + for (var i = 0; i <= range.e.c - range.s.c; ++i) outRow.push(null); + } + }); + + //Handle Row Span + if (rowspan || colspan) { + rowspan = rowspan || 1; + colspan = colspan || 1; + ranges.push({ + s: { + r: R, + c: outRow.length + }, + e: { + r: R + rowspan - 1, + c: outRow.length + colspan - 1 + } + }); + } + + //Handle Value + outRow.push(cellValue !== "" ? cellValue : null); + + //Handle Colspan + if (colspan) for (var k = 0; k < colspan - 1; ++k) outRow.push(null); + } + out.push(outRow); + } + return [out, ranges]; +} + +function datenum(v, date1904) { + if (date1904) v += 1462; + var epoch = Date.parse(v); + return (epoch - new Date(Date.UTC(1899, 11, 30))) / (24 * 60 * 60 * 1000); +} + +function sheet_from_array_of_arrays(data, opts) { + var ws = {}; + var range = { + s: { + c: 10000000, + r: 10000000 + }, + e: { + c: 0, + r: 0 + } + }; + for (var R = 0; R != data.length; ++R) { + for (var C = 0; C != data[R].length; ++C) { + if (range.s.r > R) range.s.r = R; + if (range.s.c > C) range.s.c = C; + if (range.e.r < R) range.e.r = R; + if (range.e.c < C) range.e.c = C; + var cell = { + v: data[R][C] + }; + if (cell.v == null) continue; + var cell_ref = XLSX.utils.encode_cell({ + c: C, + r: R + }); + + // 修改这里:无论原始类型是什么,都将单元格类型设置为 "s" + cell.t = "s"; + + ws[cell_ref] = cell; + } + } + if (range.s.c < 10000000) ws["!ref"] = XLSX.utils.encode_range(range); + return ws; +} + +function Workbook() { + if (!(this instanceof Workbook)) return new Workbook(); + this.SheetNames = []; + this.Sheets = {}; +} + +function s2ab(s) { + var buf = new ArrayBuffer(s.length); + var view = new Uint8Array(buf); + for (var i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xff; + return buf; +} + +export function export_table_to_excel(id) { + var theTable = document.getElementById(id); + var oo = generateArray(theTable); + var ranges = oo[1]; + + /* original data */ + var data = oo[0]; + var ws_name = "SheetJS"; + + var wb = new Workbook(), + ws = sheet_from_array_of_arrays(data); + + /* add ranges to worksheet */ + // ws['!cols'] = ['apple', 'banan']; + ws["!merges"] = ranges; + + /* add worksheet to workbook */ + wb.SheetNames.push(ws_name); + wb.Sheets[ws_name] = ws; + + var wbout = XLSX.write(wb, { + bookType: "xlsx", + bookSST: false, + type: "binary" + }); + + saveAs( + new Blob([s2ab(wbout)], { + type: "application/octet-stream" + }), + "test.xlsx" + ); +} + +export function export_json_to_excel({ + multiHeader = [], + header, + data, + filename, + merges = [], + autoWidth = true, + bookType = "xlsx" +} = {}) { + /* original data */ + filename = filename || "excel-list"; + data = [...data]; + data.unshift(header); + + for (let i = multiHeader.length - 1; i > -1; i--) { + data.unshift(multiHeader[i]); + } + + var ws_name = "SheetJS"; + var wb = new Workbook(), + ws = sheet_from_array_of_arrays(data); + + if (merges.length > 0) { + if (!ws["!merges"]) ws["!merges"] = []; + merges.forEach((item) => { + ws["!merges"].push(XLSX.utils.decode_range(item)); + }); + } + + if (autoWidth) { + /*设置worksheet每列的最大宽度*/ + const colWidth = data.map((row) => + row.map((val) => { + /*先判断是否为null/undefined*/ + if (val == null) { + return { + wch: 10 + }; + } else if (val.toString().charCodeAt(0) > 255) { + /*再判断是否为中文*/ + return { + wch: val.toString().length * 2 + }; + } else { + return { + wch: val.toString().length + }; + } + }) + ); + /*以第一行为初始值*/ + let result = colWidth[0]; + for (let i = 1; i < colWidth.length; i++) { + for (let j = 0; j < colWidth[i].length; j++) { + if (result[j]["wch"] < colWidth[i][j]["wch"]) { + result[j]["wch"] = colWidth[i][j]["wch"]; + } + } + } + ws["!cols"] = result; + } + + /* add worksheet to workbook */ + wb.SheetNames.push(ws_name); + wb.Sheets[ws_name] = ws; + + var wbout = XLSX.write(wb, { + bookType: bookType, + bookSST: false, + type: "binary" + }); + saveAs( + new Blob([s2ab(wbout)], { + type: "application/octet-stream" + }), + `${filename}.${bookType}` + ); +} diff --git a/src/plugins/theme/components/theme.vue b/src/plugins/theme/components/theme.vue new file mode 100644 index 0000000..a3df8d4 --- /dev/null +++ b/src/plugins/theme/components/theme.vue @@ -0,0 +1,204 @@ + + + + + diff --git a/src/plugins/theme/config.ts b/src/plugins/theme/config.ts new file mode 100644 index 0000000..1f2bc14 --- /dev/null +++ b/src/plugins/theme/config.ts @@ -0,0 +1,40 @@ +import { setTheme } from "./utils"; +import { config } from "/@/config"; +import { storage, type ModuleConfig } from "/@/cool"; +import "element-plus/theme-chalk/dark/css-vars.css"; +import "./static/css/index.scss"; + +export default (): ModuleConfig => { + return { + label: "主题", + description: "自定义主色、菜单分组、暗黑模式", + author: "COOL", + version: "1.0.0", + updateTime: "2024-07-22", + + toolbar: { + component: import("./components/theme.vue") + }, + + options: { + // 推荐主题:'jihei', 'guolv', 'jiangzi' + name: "default" + // 自定义主题色 + // color: "#4165d7", + }, + + install(_, options) { + const data = + storage.get("theme") || + Object.assign( + { + isGroup: config.app.menu.isGroup, + transition: config.app.router.transition + }, + options + ); + + setTheme(data); + } + }; +}; diff --git a/src/plugins/theme/static/css/index.scss b/src/plugins/theme/static/css/index.scss new file mode 100644 index 0000000..0696451 --- /dev/null +++ b/src/plugins/theme/static/css/index.scss @@ -0,0 +1,197 @@ +html.dark { + --el-bg-color: #2c3142; + + .cl-dialog { + .el-dialog__header, + .cl-form-tabs { + border-bottom: 1px solid #222; + } + + .cl-form-tabs * { + border-color: #222; + } + } + + .cl-crud { + background-color: transparent !important; + + .el-table { + &__header { + tr { + th { + background-color: #222222 !important; + color: #fff; + } + } + } + + &__body { + tr { + &.current-row { + td { + background-color: #18222c !important; + } + } + } + } + } + } + + .app-layout { + background-color: var(--el-bg-color) !important; + + .a-menu { + .el-menu-item { + border-radius: 6px !important; + } + } + + .app-topbar { + background-color: transparent; + color: #fff; + + span { + color: #fff; + } + + .el-breadcrumb { + span { + color: #ddd; + } + + &__item { + &:last-child { + span { + color: #fff; + } + } + } + } + } + + .app-views { + background-color: #2f3447; + } + } + + .el-overlay { + background-size: 4px 4px; + backdrop-filter: saturate(50%) blur(4px); + } +} + +body.theme { + &-jihei { + .app-layout { + background-color: rgba(47, 52, 71, 0.9); + + .a-menu { + .el-menu-item { + border-radius: 6px !important; + } + } + + .app-topbar { + background-color: transparent; + color: #fff; + + span { + color: #fff; + } + + .el-breadcrumb { + span { + color: #ddd; + } + + &__item { + &:last-child { + span { + color: #fff; + } + } + } + } + } + } + } + + &-guolv { + .app-layout { + .app-slider__logo { + background-color: var(--color-primary); + } + } + } + + &-jiangzi { + .app-layout { + .app-slider { + background-color: var(--color-primary); + + .el-sub-menu__title, + .el-menu-item { + color: #fff; + border-radius: 10px; + position: relative; + + &::after { + position: absolute; + left: 10px; + top: 0; + content: ""; + display: block; + height: 100%; + width: calc(100% - 20px); + border-radius: 8px; + } + + &:hover { + &::after { + background-color: rgba(255, 255, 255, 0.2); + } + } + + &.is-active { + &::after { + background-color: #fff; + } + + .cl-svg, + span { + color: var(--color-primary); + position: relative; + z-index: 2; + } + } + } + } + } + } + + &-cangqing { + .app-layout { + .app-slider { + background: linear-gradient(30deg, #7397ab, #75878a); + + .el-sub-menu__title, + .el-menu-item { + color: #fff; + + &:hover { + background-color: rgba(255, 255, 255, 0.2) !important; + } + + &.is-active { + background-color: #fff !important; + + .cl-svg, + span { + color: var(--color-primary); + } + } + } + } + } + } +} diff --git a/src/plugins/theme/types/index.d.ts b/src/plugins/theme/types/index.d.ts new file mode 100644 index 0000000..4d62828 --- /dev/null +++ b/src/plugins/theme/types/index.d.ts @@ -0,0 +1,6 @@ +export declare interface Theme { + color?: string; + name?: string; + isGroup?: boolean; + transition?: string; +} diff --git a/src/plugins/theme/utils/index.ts b/src/plugins/theme/utils/index.ts new file mode 100644 index 0000000..2034c6d --- /dev/null +++ b/src/plugins/theme/utils/index.ts @@ -0,0 +1,117 @@ +import { Theme } from "../types"; +import { useBase } from "/$/base"; +import { storage } from "/@/cool"; + +function mix(color1: string, color2: string, weight: number) { + weight = Math.max(Math.min(Number(weight), 1), 0); + const r1 = parseInt(color1.substring(1, 3), 16); + const g1 = parseInt(color1.substring(3, 5), 16); + const b1 = parseInt(color1.substring(5, 7), 16); + const r2 = parseInt(color2.substring(1, 3), 16); + const g2 = parseInt(color2.substring(3, 5), 16); + const b2 = parseInt(color2.substring(5, 7), 16); + let r = Math.round(r1 * (1 - weight) + r2 * weight).toString(16); + let g = Math.round(g1 * (1 - weight) + g2 * weight).toString(16); + let b = Math.round(b1 * (1 - weight) + b2 * weight).toString(16); + r = ("0" + (r || 0).toString(16)).slice(-2); + g = ("0" + (g || 0).toString(16)).slice(-2); + b = ("0" + (b || 0).toString(16)).slice(-2); + return "#" + r + g + b; +} + +export const themes = [ + { + label: "钴蓝", + name: "default", + color: "#4165d7" + }, + { + label: "极黑", + name: "jihei", + color: "#222222" + }, + { + label: "果绿", + name: "guolv", + color: "#51C21A" + }, + { + label: "酱紫", + name: "jiangzi", + color: "#d0378d" + }, + { + label: "苍青", + name: "cangqing", + color: "#7397ab" + } +]; + +export function setTheme({ color, name, isGroup, transition }: Theme) { + const { app } = useBase(); + + // 主题配置 + const theme = storage.get("theme") || {}; + + // 变量前缀 + const pre = "--el-color-primary"; + + // 白色混合色 + const mixWhite = "#ffffff"; + + // 黑色混合色 + const mixBlack = "#000000"; + + // 元素 + const el = document.documentElement; + + // 主题 + if (name) { + const item = themes.find((e) => e.name == name); + + if (item) { + if (!color) { + color = item.color; + } + document.body?.setAttribute("class", `theme-${name}`); + } + + theme.name = name; + } + + // 设置主色 + if (color) { + el.style.setProperty(pre, color); + el.style.setProperty("--color-primary", color); + + // 设置辅色 + for (let i = 1; i < 10; i += 1) { + el.style.setProperty(`${pre}-light-${i}`, mix(color, mixWhite, i * 0.1)); + el.style.setProperty(`${pre}-dark-${i}`, mix(color, mixBlack, i * 0.1)); + } + + theme.color = color; + } + + // 菜单分组显示 + if (isGroup !== undefined) { + theme.isGroup = isGroup; + app.set({ + menu: { + isGroup + } + }); + } + + // 转场动画 + if (transition !== undefined) { + theme.transition = transition; + app.set({ + router: { + transition + } + }); + } + + storage.set("theme", theme); +} diff --git a/src/plugins/upload/components/upload-item/index.vue b/src/plugins/upload/components/upload-item/index.vue new file mode 100644 index 0000000..f4b9701 --- /dev/null +++ b/src/plugins/upload/components/upload-item/index.vue @@ -0,0 +1,442 @@ + + + + + diff --git a/src/plugins/upload/components/upload-item/viewer.vue b/src/plugins/upload/components/upload-item/viewer.vue new file mode 100644 index 0000000..38a53e5 --- /dev/null +++ b/src/plugins/upload/components/upload-item/viewer.vue @@ -0,0 +1,103 @@ +

3ExPr5W{+xe@#{A zl~h-ASJ&C{FrIk}K$gUO>GdN_IcFs2!B?>zbq!KwO9~El7P1goV`FJ5A2;@9M880& z$;JwuVGzTk4=v3APK^4v5er8OZEVjOPf^jIqt@N{TY5Img%PBBTvVI)#&s`>lqE^S z`S!ET(%;gk`HH5tiOG*X(of5)nGe3GVZJ|Odf>Ga=!2{|?fsOVo?gASyZgLPP5ZYx z;rMCh+%rUro8V{OjrC`c-7YvyqM4{i!0l#V!NEXAhGR{;jg9(baQS_c>_DW{8ijbx zl(5J%9|j5K@uti<9P`SqO+B@vrdi-(!2Yl7(m`}3Iy72^=Ji{9SV>VTaelk{J;T(@ z(pQ`4=;*k|DTPILFxb6dsB9Ua?fnMokY;0pcQmIFx9fI$vw-?tz`Cjd=8+B(zu6AB zU@kHDqh(@h$a5%EF`Q&xX}@^Ca6|-@hIHJ-&)932-0h@Y9wSk6I;v%5Wjiyt!;WAJ z8kJu05JT>+vMN?{9WH!k+Cox|9Y@@=A8b`&RmIfDR#C1j8I`=fb)+W#sn)p{D*4{1oxR|I@#3wi!&6QKhM~C357D81N)%8zMy@TQ*_#`9 z>l<(b+gvA9n8cvz2$^Uv9}cZNdzz+0!eYJdxLLpBaJp;!f#?0G(6HTTCh<`{i1#JL zAo*$o;!8xO4~;f7)!FHETWN;tIe07UbG&930R_3PEL;belvZ%}D_*({$FP;nO1hg{ zm_fbTMIM^bj~dr@&Xw^q(vOmFF`3q^CKH3y*o4fSedDvZoLYQl+WN3ns2)}ia~}z5 znr8YnNIL)8!)|$&F=;US=>&DR^he!2jixP$ClrM<7n`0!k5sL2kHj2`V>VB2zK9Fb z)0S7*tXbVHak(E0LtnWImPuz3$I-Inu$gWm3htBOnNw=vbA8n zJ=&;RWb?i?X7$l7PtjL*`>30l^%?(%5u}pjkxWV6v?o}~dnL5l-(6t=M6(;~21lKC zge^7y%ffUEzl&-c5j)}!vs#)5F$@}`+~Bw3yi1mfDKf!X4_yeSpzJkjO z{9&)%mFW3X@@R|Cuo*roA|B5Wuk7=*Fst2h8^61pek^tk4ntUJ&2pHt+d~f}syBS^ zSH~{T=^DEsn6uO>1;khpJ=p*|*MsT1Ops@b{tn-UKyPPMo;hPzDV7+2phM^Q8(AyB{f?fG8Hn%k<5`~FLlx}o`!Lx3!@CKxscBfBDEAbd`NHlb}MwR*_c!A&UCW^3dSM`s_N^!hVUS3YP7Pl(RLy! zepAt<*`4SeT_JPz{p3RRo%IRFBu3AJlJu7k0@#I*dtLZ*f7?CW$RJQj%(WK-_vN^` mU%Sg;Jun}61RUJyaR9q;-_eCS;L^Wq2T)VeQmmG@`t(1%KlQu- literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/sleeping-face.png b/src/modules/cs/static/emoji/sleeping-face.png new file mode 100644 index 0000000000000000000000000000000000000000..749e4ee2ac9033f5cac7b8b62b53cbc962e367c6 GIT binary patch literal 3525 zcmV;$4Lb6PP)Px&08mU+MMrQ7`CANZC7$7%z{ zY6JYQ5y)u)$Z7=*K(^R*1M7_t7fQb5h7iPO0q}$X<$wj~f(0~Aq5G;5@Q4EEfdrLt zQUNo$8X#ime*yBE6!)MI13bS0GP5U8w=PDUI%do>X4OJpt{hd(^^^wqrW5s|7a}{E z4oSuXKEd~!2{vrn0X4leXVju)JgR|Iv4BoSQITR~lK;~(|Ijk{sSE$cGylFT|H?K0 z%rgJiG5^Cg>X-)ot`7g&G4`hm|LZvayC(nKE&sV6|KKeDun_d54FBdd@1F|fkp$j_ z0spfS|MzS9tqlME+yC}s|Ls5j}UD0G3w)|L91~Z2S5F|hyURQs0z@SX$z;71x!%ipUU|N5#0 zIlU!X&opP(-FpE4%?|(9P{N!bqjC-Z))N2bYUz#v9z3J}&@8=!5bLQZ|Dgf(-aY@{ zV*lxU;;c37xHj|SeN$$q^S3Ad)Hkn!71^H~{q&*y(Iflriq^O-|JpzQ&_abu0H9?A zsHBSZ%_HsFbndwv^Nau`O}SB6hWXkl^T8n0#d+`CjGM2}^Qs*6@U+>;i~iU}%yk9y z%s%wfHDMeM#>;2rVhq`Ae(($J*3x~gG$v|d?tbbXHV^VXfEsol$Ez`2d?(o?O7 zK+d>NjhCv9q{p{-Ep#J9xc~qFNOV$8Qvd=caT*5j0S-X@4*yn&`DaAAy2jrzN%EiC?v`{&--{qhy=nZM(pF?RXi$??q7w!qbi)uPYO)3w2ugC_kpyZ`_S z4@pEpR9M5r*9lZpXBr0JfC)*etb!97dAzV$PR)C$|9&Jqhej^a_YWUd)m(T{r4sW#Hn-6oHOrH#QR?!-tWK5 z0S4oL6BgtD6s@tUX#IDEIocCUG297E4ty!wa6Fwqj{Wd&&NQ9A_>+}$CzH%w`T1h= z37j$2be7NfFon%DUA%H~>Vfq1+SuUJ-HJ^q6lv9+gv zGT9A~#>Pj!!c{&#tAvjp0hgZQHF1tEbIh{pKY#W?ZKi+8O9AE4rfGy&8u5-V-nFwfUD;K=>>n(~XJa%nW3;Ij zPs~Y4Z#-NO?@gj+FbVt4zkPZ9_=0)noTbNy^4oU{YMGKcl?Cw$7;ieSo_vRS0*8J+ zdi14V^2FJHKRT3mAU4Y!>&VMu_w9>^5DT?e%Ra*V93TJP(T|@0a5W{TZGpwdBzXMy zBS+&*N2jtcHWs8IDZ7vReu<9Z)%xOY(uB-;K75%#l37F}Ik&kD}p+E1smqvyv%rKNuw`K+*JgJav#alWjm zsOT$*N+aG=ixZgF+wdE8LauancDy-JT6$usmKJ*$WV)y=C$YVwqob^>Z1d(KS|c3f z{N0D0Fhg2;lhRLTU*O?Rz*%?U0vPb_QUOk zc;0%o(@Zq;Fza@F` z;>8zBmeBW-miD}gw&;r&?}1ba(pJch!W|(Y@JBN{LUwE|#6>u#?)x*RPoF+>#;Ia( zHcNfo%waNT8<6f*R#xtXY^Abec&LMWx55dP_Xu?Q%T!eG_zdr-g$&)8{56;q6LCpDs6^;Pe^z`)iXJ>E1`1tsQ`h<>* zkI!DRnw{O>0}?nV$2F&!sYc^28ccTMhgsWNS@!e@va>162Qk#ZgAW)uMig)aJ+>t! z-4tZRp&^5`-N~s@{4fznmIA>}h)^h8HCo`f5S9f3J4-7ONVjhXH>XXbH;Ju>kV|4r zlBlIcZYNMv!qF9A1a@-E7Ew}3clYfz&1s9c^j=RwC?F~^Mig`PMvGj&di82oS6|Ha z%h!ced@o=36~Yl$S6E#wmnU7lAxcV&0%%;COqW64030KLybb`AbVFLMyi7~3uGeq0 zUQ!a!S9p2Pfbu#XB+j|3t%_bzOk&glFTdSsHB%{wIgLOriBPHGfg-o}>g99i&RxEt zY|VQRd4Ym_uS?Y{=U_$pAmeJ18q#B{pyneyYPJiC$jFSmf!y5O0cl2LWaL?)D~Yf0 zLI$ps!%Ey9I4QEeB_=V^<$zZ*LRurCq?Y$_JgGD<4?alq8XgL%-GZF~iC)Qr4`=VU;AHAX)XT^iA=o5m zfJ8kP8l-zL((?LKJ68{7r%+=Bofy8eODYc<$@J*Ano&(N>~^Y)Gan_p`xnkOG&G#8 zf4E5?(17d&!c(f1?3SdY7(PELAeDE--$P7aHP$3q#JFVKBz{;}l1OAF$UX(xB#^`1 ziT5WJJ3F}`dlRl?TZ%-YurNL!CUZK{>><%r-PvLfe=-}NlE|ZCX(5+wg}jo3n`?Bqs%K5&(Hso3kxKCFifnCdt%zkeEHq;aCsiOszOe%MKbl%&zvILh_gW@O0uD(A!`_8YXeX? zTrw!+9=tYCtfzV(u_j3Z*9MbHQT*_5YmjW{R#rWfW<`ZCv_lH5afk@gSHB?{2Kqq} z4r`!NGz=FWZjX|hvZDS4(bfhOV65%K@zQ%bLLr!JiuTY?A|_%FtmNexfRepEDTSDg zO^LF#wN+U{bs>(ueK<9kmlJGtpu}wGS1gv9gsS97Ng7mlEO`Rfg9WdJLI%Tr9Dz!KFq}INq-d8YRVn+{4Cz2zT-?B(j9h!l;p6q9 z9si9%!wW&BNfMdIT9rhVLaRYh9)~HvG!U+?6X5BF4h|+BQHa6j&V=D;$&HjE1GZkG zpn*%ds1PPv<-lMP^JKoqCKGwIDW(t>rZ9B^XnCZYXVY)N^*1~R#f*{ z@>0z0jtU?OFdKO8#0eBAogCKqsVUg35@mgBy|Tn58ny-)G8W~? zB*xkd_9zf*s*V^W*l{d{6j*Ma0f~u;0f?Xi42%^_Ia4OQ>kOBINh#ij4Tz}%|3k(d z7=Oy)!q>1Fqe+~pI=JHz{)D0}b$Bq8JE;=^PUia@s*{09rYz87<65MMNK@Fy4* z!a$uV2Au!kP@NboU1NQT3|kQkGtL1KR6LNGX-39+ze7*QU>X?fh*61%Ac{sJ_6DV7$8tM~7TDD%$(ng#_W zvu3f0N(zXvf=yv3YOIDG>MTwHDIpnAWo0%tWo1zjNj_dCsAGsT2WAvVF(U^HlNcuu zh%vFSvO~p*HU`NoaDssHSqYfJia*@Z0B)NBq;HuzSyx6!00000NkvXXu0mjf($2>0 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/sleepy-face.png b/src/modules/cs/static/emoji/sleepy-face.png new file mode 100644 index 0000000000000000000000000000000000000000..21b72c12dd5e612428781b34384d194b9d66df29 GIT binary patch literal 3477 zcmV;G4QldPx&08mU+MMrQ<&3hTcXamS;0K|D9$bKjMl>qFfAj641 zxqmsrYz;Tl|Luqe;fNUHh!oCn1;1<$&uswohycZC1M{F6(SIJvZ3fbE10#^_>s;r4RhA4#sE$Hqrkz%=ow5)5sJwhsO@NTY6OPKH`D*GUkwM5)|Fu~ z1A)T|d$&4uj{n6q|K2PA;w}EVA^x`+@uCX zH~;r&|N5i<_i@&F0RR2flvM%$f7_F@0;VgLKC%xwVcnFas(lK=EjqGJN~ zrv}n=0RN=|GRgm@X9Iyl0ss8R|Kw@^{@nlk!S}r{cRc`)Q2}!|0RQ{A|KBnH?ri_P z1{kjT|KCuXTLa^M0RQrL|LA%GpXdL!1NNN)-h=`Fza#(94zF|ydq4o@q966uK>zhy z?T-Qa#WDZyi2vtYo?rs@wI%MfBmeP}|J+^woB{vQM3i3#a5Vt``@8@9zW?%p@{0if z)li!5|Nq7YwSEuRx-_bN7OD3C|H2ac=u1b?|Nq(&{mVVtni>AuKc8+7y_6)k>;DkD z|JMBf$MOHT`v2y@Hj7sX$D=I?sq@~hEyn);0g~7M$1Bd17Uj%1^4&uRu>0lz|16ik z_rfEDM*+oz8>{L6j)`L1tUL9~Bme&GU?>2NHPk)FveVYH?9e`#+li2^AhE6!T@3*a8{n9cIbEi(I z#z(c_hr6w4zRr%McUd^;wg3PCKy*@0QvfZV^cWHT0R#zL{+~@+lu`Nf!k{C7q-i1X z*R}cQLW;CD#dE>l_N!i+%kkRL>#MS}jo~ok+TFvX=Jm2}Ecpk`S^xkE=Sf6CR9M5j z*m+b_R~`p&_OJ^gR>Y%Dt8KMb|7fS3eS$zBfGlAbq|~q|5U>;kL|F_FQ9{^-01`k& z!Y-SDB6~r$vW~LopyR%cyY_UJ-@Wg>5CYY9X3l&$z~SDT&-wlCeGgucll$L9P452| znko+MzcEym<+V1fUxBlJgOVKr@Yp>t<;YUo93pggsk3QVELRmqWrli&jaufv-=sOzlI^vdDa4t7)>${3l-mn@R+nkhLIHcu8C@^s3$gX4M9w4n= z|F%u1DQl`gTbbV+gB*buk(if+f$`d72NMg@BUCisvgb<3DM&n+l9F=vY%RuikpUKz z+S;=zDZwehCI0CVm233h>bT?t1ROkgFgSP~Lk3dLA|)6m0|Lq|k@OQap}ulma=wNH zNHC;92!lWZ&g=?Z$K_r~ zEH5vQj&^gy8d)2euPMYWIyxGb<+$X=E!SRrVES7tBho8MAm--gu$0m;hY8Y%hIw0US@M96%oj4n&g7nTpKJh6eh=T~b|E zUl~EK!ct~&e}8{*>;C;GaSljA{CskMD_($BHZ)YVGb`(*4o?45l{jX5RaI4I`*2hB z6u)p_ur(%T|Kh~N#Iz0$@cF#zrs4KZctN{yoQl-7qywE9)7z`shed@f7ORFe*V4KN z60O8JBOJXeODSX5I}SlvAi#W?Rkab6Jk84ZTCg5Emb zU0qmHBNE#oNOErS+v@WyjKw0BU|?`!M+lLyupK+T+98{;urOk}bz*Qpz!HfOwC#Or zi?*H0voM(!@LR@1cJB1>NKHjfSlCxGL1Q64A*rdU9y@pL3>k0X3#N@N@=hIBC!tlx z0WzK94;EYc@7;@~)a^cy1seLn$0rpr9v-%P_xf8FcNg-C5VVqnRwAgVdhpGtiwj7$ zwl+3KM%y7iGC^ZTMm9FK$Z>ITc{ALFp=C7kassdF5To&d8kS(HRMgZYiVYXASY4+_8Bh>{*yQoct>WgpN=Sjgyp=#6S-05b8{K{x6%pA)73Tl7N!NNCbqWvf-;Jx5P{fS65SC=xpu$FMm}9 zE|g??n3Cvpl9FJKfRtT*{rX>zZrr$x^XO%fgDEPA`?1t>gpMLE$w9g%io!XaRQCEG z|NQZlU+8y4CMSiLXFH-efo#d)6X%9X!p%ub6h@~fiLc(c|Dva_udk=4XJ)4H`h)oI zCWTMB1MRtb3!Wm>b=moFEkQ{ZG7=6HG(3Cx@vHlP@9FF9{R>L(_4rBQ(|dPxS1o+3 zfsz6_5=oY_OA;3i&R!l7K1NjE_q|B#y9hPi)026$)lnpxBP5ct1K`%JA|f5#vriw6 z2tn$(^gW~(q@j^7)ny<36b`ZHC4v%n+==xLpW^X&Q@rrxtlRfq64Ff1rAuJ-_V$Dd zfAeSAhe&xtE^HH$$f8VqZ^MH?jew7XZu`Ql*&iMXGhWQhoIg)ccZWv)B>Pk=rw+{p z&Pz<+EEaz%Jej&rlf!ue*1(g8BSZIJG&Y_;kF36#4B;QuWFfVcx}=0D*V{Ey-dUxw zR7y*;k_&5||1cs9y?ecppn7JYH-kkbDWh$YNT$h!-mwW%0V|pR{OO2r^5Wg=h-z%S zem6t-`(MjKG&!Y@vBZhOR01yx6_Q1yw6rv~)SGwj2|#`qAN&pF!9~Bxp)VFgG&y}F zh@#;a8XEeAf!q?3k^<~gejvvk zigLI$Hda(778|38#bviT#>P6PXCDs@W%$L%Ye3@tfc(TjZV5?Md8Hy3w}&ZL2NY8W zDyD(cvp;4GWx(rEpdi2a%}cyZ28pJoZiw1L6j;8{BNbm?xYzHzyy1r>6y$7Kx8!?T z6nO5lSRpWAoDQwo!L9H61-fAq!(fI2>#1K=z&6 z&u>6W87NSX#u7U;Uy~|LNgXIIvJgfjFhAd(%{_Gc4_|)vc|6gUhQ@E#z2z-Cj*^ao zAy`~CXhhg-F87e4;qP>mRM&0(d)8U#e(C=cmII6*)JMFd1j={UNi z1f)A8j;`P1{r~=dJG=Xxo!QyX%+Bo0Vhs(nD3Pp42n0f@jnXi>RMdZil;|?sN~?Qa z3f$qg{%r`PG?x6#hWN6EdKhV`L5h3XS0NB0h@qZ|rk1N-_j5>_3;BK_eRm*FOMqbK zGYBD{sXJP(Gk~|ppQXits2NSv=F8Ieisd{Vx`1Wae1m9mByDwsce#=6Vj+82=q`>0 zkA}{?f(*J76Oy2(?}@vdU@h*{T`v&JaqKNmVqfz zaFPyAzJdlmP{jp~!oaL37!n3U;>(HfKd&KRUhLurBRKK}?b41Cko>1mfENK}bfDon zI3EVbNuZm5As!BTL{18+j=zwD+RK4i>*)w^F$V^OL8mY{{RK{P&q}F!oFHKFIyiC% z=bfNS`m~s0JQxbfnL(!{_<4yA0>?3b-@p#@DMtMv;ItB~%76wDaMm^%3Og)792Qc7 zvp#Uv2&Qj7^gY7slc(cDu%3U6^90dil z)?n5Nj9Y+}0MM`b!JT${LccDCHN+6n{gwf&J_oDbfPhJ`A^|1Dpoam}(t_Ux=}(cM znQlGi5|rLicoiHxS#Q6!_LXnCR&hFugD@_)+#?3owZQ~kDVFEBUxF}mV=afV%!v$4 zF&+I7-l@5E^!v_HgD9BdDs&)|h-&k`Jd7SsBLic|u_<7n-i5+R#fq|6uAV>2f1K!W zK}h;@k$1jRC5O>()IC^C4oa;l=y0!D7{ZSplrA(ob>e%$|I;r1C;Wn*@p?7{g5=iL zxNY*F`PUekiSd0_64z|iF-_t|&dGHtT&dlsO-1^$epS^t{ka5&hZs60-fYRThdgEE zS4r<+O!`?uF``PwjToo0kg|e}x7k%LUW)%J2!BtfLP&Kc(iI7KyY^c7dcxnW+J&1r z+1vHlnkFD7^Nuz2&Z!H9tbKIYV;1H!sIB$(uo|WP4}cmNkUiRc%O7KBC(mPNcTKnM zXi)Y3^5G+4a!NBXZI++DUyC+65>q9zl&^{W^rkN{MUW>4lH?ZsY+6}sl(x5*sP}!f zuz^x|6i2K8d+Q z^*v$99odSE%J}2>SCbo~uCM}=+;rCR#}dom$cRb$TI4tWQXEsMUEOA5#nuGujyGaL zP3z5)I;C<$G16?s8-J=j94su@pT(mr*rWbUPP})^=Way*DaQ?3sSs6oxvyKHOc?B) z1sJSza{9Z+N1l@yJ9H1dZ%ZfZ{W1H)|6pT>K;RV(oi&o2TPE@FcUq_7)m~P2LD+BX z6@`tBkH^?)(7=#-*~bU6zlZW*r^2pVf9e8GKd46tQ9M)dN5Pk@TMrJX_-ZTfI(F&M zA0#jMDcg zSY~6>1}eI~BUX@=d-|TREsZwxv9wfgzBvEsv-Q5e;)P%*H)YyHZT*K2ABv0DpE{m9 zuo@e*ES8kUynkO)6CdxjKC+2tZKXBkCTcru-fKr$THfs=9-Gul*6yEE+`rFJQ*2Rf zWNiF>)tmv-JoN$-6z}yCIWp8RSlsZ%SnkuApwNon{n731ho~L_hDcsaR=``q{)F$% z&6bvb?a258N8V4a4we7wN;Nl+NK8!ZGc!4(wEvwq(-?_-5U)Hr?sx3&^LyAYpw`EL z$HdS@J|!mG%O`g4kW2B=ZA{C$h;U5KJsI3&B<|n3sF3zCj;aUcTJ?IiX=ff!*o{$* zxlc*ObU0n>ZzArqb#RM=oqa`%;Okqm*A%v%ImF_Tw9wo7@npX+(@IUayQ@p2sAgqj zV{#N6`WQW+2V)tLGpZ0H=vE6!l^66&vAA?|L=Qb9W5`kO^H!juW+*{|m;(d2^ z&~wpQ{RtHF6&0$VoZ!(}I;#Ly z<71XerMrsP_A;!+1Ld*5-!3m__k9xWp5u_gg7pe%)KS<+l>JM#5^Pt_iCFo1RZz7Qz$xgsZ+h086({=8j`?F;zTv3 zC%N5%ke*TfG*BABMBJKLBa=GG>Rhpqpm3$8CrVF!nJ$IKzPX|i8cLessa%rC)Uo*S z<>LDqKJE;Cb8cO!o-l#yyu8?i87GV+vy+@5%8uSfvR^@gIWp==nfJ@Z$V9&H9U1Jk znT~agGFUwEo4DQ4>{KTiyRKFW>aQJw?4wRX?QU98jP*)5C7vsRR+Rd?I>8BL-=!_p z1A2O9ElHACLN#22f_>7eebEKapL)wNB*Dby?gtvwYRah4K8Un&Dd3A!Q&*3NCzY0_ zTSP?%^RNppWMixyQsJM6GR%DAMaecPb>?d99iA$Mg(K+b$_la_<_(@!WXOxRcgkq* zUfaK}mVu)eD{J4!B&1Efc#+Qjsi8sDf}x^d!48q%&;cj8@`UHJ!Dx=k7+NP1rfY0L z7rRbLgMWr;`X=yNT9AW^gk7Ba3SR|7(p`%`o)q@Ouij2R_@(GlIgD~|GFq89rH%+k zgokt1y=Fs6&bUrqt=3~=cw*Z~ofsgsGy5$+nVy0_yzJ*fK3sLWkZm z{9nCSQ@7(FnN%o#B@)YFXXVU`$oYW6rv!7j3@dBNV+9!VT<(reiw zs`;?|BePtDBPPawI&gMF`N5>`mcE#Uw~~^d?(^x%FKdZPx)E0T3byyk^>$*nrbfC& zQbVyVj}z$#zVQU~@Xe|OcSE!JK>b=5PGx7e*E_VauKjKBL?q_zS4Y1R{%M%58h%+b z#Psl;nel8C7(B>+aDow^ysbs$Bx>n4vS5BoERcm=G0t{%q-mMxn`O1dN^-_T)e@q1 zR;s%AkVK_S0EgDDUb<`~rfg={mbo^-;Zv&iQdxlB#zfU-_azUHs2TDmlj7B}wAr(3 zmFH*=jHDW@^N|n*lw5LVCIZV~_9nOgoxx#Vrl)D`wLsb53`19hjEr6u3I1g@7G;Xn zBh?Tp54UVhFevE`tf+xGJ8E|p=%dOoIm)YsjKQ?DwqZi}NKcs`jQJNF)V$;GE~E;u z5(sQ|oMr^q85f82?93Z|vBqmzp9q{J;b^4NyNgu}T9;P5lW&sVe12zdkAfP%xE((q z%ki$spWj(*#n?09jRXSe$4JGT{2&j)E62#MNj}IZduzrToBvByRtpwQmktXPrO*42 z#!W137{jOKRUsFB@IgYUKk7tOy}XuBmzK*sh_NLsJixlPUXmE>_kC(fK*$DB6OV5y5cpDL?aWVWz|xH6QWXOry)nU-b| zFisn10Yxk8-Z}0mX1?!1tYeA_jgXZ)1?DJD{*ZUXo-lYRy<}Ko$-@meB@_mcM$4MB UiKJ>>KEV)eO#_W$HEYa&02%gsQ2+n{ literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/slightly-smiling-face.png b/src/modules/cs/static/emoji/slightly-smiling-face.png new file mode 100644 index 0000000000000000000000000000000000000000..a0d68d143670f86a470f17d698344f3deb0ffafe GIT binary patch literal 3737 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 IM6N<$g1LwSkN^Mx literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/smiling-face-with-horns.png b/src/modules/cs/static/emoji/smiling-face-with-horns.png new file mode 100644 index 0000000000000000000000000000000000000000..f0b56af317f9129fb29f2392a386f94edc01c6d6 GIT binary patch literal 3957 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 P00000NkvXXu0mjfnYKJH literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/smiling-face-with-open-mouth-and-smiling-eyes.png b/src/modules/cs/static/emoji/smiling-face-with-open-mouth-and-smiling-eyes.png new file mode 100644 index 0000000000000000000000000000000000000000..a702c9355a956dc2de1c3b12335c9493e21e68b2 GIT binary patch literal 3710 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 X4wN$N)#mwszchqV*Hx=fu@3n^HY}O; literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/smiling-face-with-open-mouth.png b/src/modules/cs/static/emoji/smiling-face-with-open-mouth.png new file mode 100644 index 0000000000000000000000000000000000000000..7798b4114bcce2d2c2ce97aa15a84e4bfb07afe1 GIT binary patch literal 3581 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 Vqw&a&Z*%|v002ovPDHLkV1lXsSdRbz literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/smiling-face-with-smiling-eyes-and-three-hearts.png b/src/modules/cs/static/emoji/smiling-face-with-smiling-eyes-and-three-hearts.png new file mode 100644 index 0000000000000000000000000000000000000000..d621f0dd7030683fac524dccf253219de6060677 GIT binary patch literal 3986 zcmdT{hdUMi!@bMidvDdX*9{S_tSftKP-a&~gk&Yk&IpmLj1n@--ZLZX+I)?S?2B}5 z_w)AqGv4z&=bY#K0p~o=i8eOWp`qlYgg_uPx)@EIOhUQWaJpCKNp8>mAd zl?hZAb|n8Y)Z0|&7Nl&5dlLeILyYy!wRAl7OPtyMM34+ThNPKt)YHPN>?p0&RDBis zQw@065*a^}!}4y>eWQeZqJ(wPB03$RHDu8KK-MC9*nyjUy*(Tk!gN+Z@reR9`V`TF zh1S}UoxFwaq%%)kVd`-u8KZ)?(IB?ba{cb`%@9aE33QQ{zRnI&??}=eNV*uqRIUZ7 zA%Pwh&<)ceHZaP|LGZO;=xQ*0F^2)4LeFhF0F1~=&OPii6nCdlb4e0Dk7-*B-e}@3aw&1+n?dLv>TqoSAEih#Hb<|KzaAfTE(LJv0W4*?xQV?Hpjbp3-3>@1%Q)Cn$x!@cAo*;pE| zc^h<#b(2FkVqwI+C*b%ISW^GXO}D~G2D+&){5`;F238UR`UF8a`~qtO2u7d~4r-AE zVb&39_?fL4IQ9bT=xJKS8zU%~V+JF%m)B7ENppejRxm9A_G7`g-lPaE z*bKNp%XGxEgM*-AZ`uYX*vX4$=TRXm6V{h2ZulDUL-vn;gCbN!+!@`HL?X$<6c{ zTz)Srek_YT!_f4LFAjHVveUVc5=UMY(s_@4Y++}Uhe8$N3MS?@p6J-qXy6Z}5| zv8jIY-$qbf*VR-vx9%LD@;5iL;wAG`=RmOCbu-Z!Vt&v78DtjX4lP|}={ zyuAE}f4r_>FtF8h;(jZug2zmyk%mJ~5~bjjh=}{0^Ncq!4O_N>hWBlK3BIxu6aHS@ z=gq}i`@3V>Sb#hn?h%}oN)(Z8b?Y$2>u<9O>E%7OA}U29jq zE9Qi?9UJx{WD-~6R0frqo-8!EyfD)-B12-jcVc);O$d>8HaC2@!#|RXWAeDw@ax9| zV@konDm*+qDld&0o3~R|vl7~i@T$8zJ3H&^I@T|U5Wk+CST%?z&4X9Rl>YXMX3;z=0-n{7_ceJXHDGs4fjHJ!#dTZ}Uj;I#D z`0G7+XM+jK!EtB(!k|XO^%yJg$nMwHga%SL0@Z;ge_R(pNIRHYdsG@XLJf7yI`pfz zC@9*vn=&;&jrC>L(oSBb&qX_|7)dFYWmQi``cJ~w>26mYf9Oan)AACELt0u|>Xb@r zKNSvTzTfvRsLydv^fo6_^E}?dCIU_i;`25g^mBG&zE6k{L(hKIj!~1(At2veO2iE)U5LH{p zKX5o4nXBU?Y?>KGb&-o>flFnARpOBI!D*1?nX*@x(@axvT0GbEW>Z*kx|M^}lY$6s zZTwl*&xJ~hf$AvS_&92@U-IEGL5;nYgd=2M>6T<|)Ji#D)2;){{_GF!!q1;2WteLv zhmX!K@0Ml0Iy^Lp-P|D_v`0mi3v84;7zn?)FDPigPq_D@{jzd&v`;srbeJWzFd)q` zI!abeZSj1G5bMB!{9#NK;R~X^Zt3jUR7ZGNxqf<>ZJL{!_u0pBj%EJ#`;=D^hgPy> zXi+h-1ay46-qn-Nw{L_-0>XP+vsI`ZW*cQ{MD@x!ItQ~kNR#Y*zK4fH%mf5FM`B$8u>X>25XBYp%iGaq#iwB#wCUBK4FDLL(n&ic7RtVlkK~G6cJ% z1tQT0>K6MtcT~$fK7`gA&OQb_CkyzIEJ51iKC5@ ziYoF&5bEXy7CDV01EE&?^h$qL=ut4i8*?OAm7g}s`j0}Mj`+S6j`mQSz*B82WZ6u7 zSkiF*@-891B4|#>W<$3ty;4%Lfm~23&qh3Erj)$$ylu(`3*6JwT~!9$uK z*KlneBM(w@J#*=A)Hpjk*BtwcKg8j>mQjl>wmA>EbGlU@_z2S=S*2KV*wOj*RN)?J z1tXR(|9cIE*3>u~8uB`u1+~lNplSB9fU6YBBEZrD@u+`PEId%N533V zyGZiE&Lzvd=1%C0RB^lZBcM(s{D6W6roQiwktRyeXFkD1N&Q?sdSft~cZi1d((w1G z{1FMeu%RH+MIEy%T_+KajX1=$MZP{^7FFrVa@NZvODa)A_pPoysFolI!ujPGm3AEU zg!momY*HQE^wFq$*C-rG=~wKl9My}<;~ih0H#Xc3c>U2DCTnZ?hD(}RjH`HA^hfeNuH*B$rXM#yW9eYx&!A-Vf^2=CWY{NmXpp~A--FI|J26k>;OPLi+RL7o8 z20lK2t6+Te!ZF>9F&xklqR60}%_)#Upe{`>}e&I{OUQi1t$4ImdeV$)l^2t7%{vtR`gVDj@3Ob zTEXU9LMO)<+z7uAENfiI&ULSU)s(hl!Aa;xTWYUUYL8$vsVli(FX~jtK1Tkr-^A2Z zYxJ!DqlV|OiC-qwW0jviVb{Z_dlA7A&DAL>BgH*Df>YHvH}i3{mDM8y8ykZcG{O(^ z(@724N%a2Y8h2p&=2ajbaym?7s}rl;YQ}Tv56q}Z>iqK947If8g`eT)^Ly=5Dgx~w z;cpX-*`N1}qEHr32WiJyH5lk7YEKQ!<*s$mp`~SSmpo@SNqVi)DuoYKxN4?pw7fB^ zteEMAs(&A(ZmPL6eJ9lQ`TgLGjLs*Qo9v=VJ(Rd^v*BDeOud4C>z2);96LucXa32F zOlP|QJt=E|dP7vohh`IwgpwzD7>U65nrzYU^=~VW^R(=in&s!49gkTz!6wBhnKJh` zXNQJnBY!>8*H6nGb6e-Q?dMfk)w!K7vBC{eb3(tZEDr1ne|H92h-nGu zjU_M5b&1w{KYEtO`xEM(hrI_1U2ixNN?*myyXu7=HK4tcl1u|;N3&y>x0REluciyR zY16q@=*Mht{b_6w z{8+xpeI?z$Wk~G=R~wectCo$M2^bt4v#6x79kH2?CJV#)h8hn3t*x!4r8Bu9$?X%R ze)DVYwj18yX6Xc;Ur!>j|KJ;m`Qlow)O)3DiZ13FMf0tZk&wHUffJ;;NZKWq!MKO2 z%nV`arsOp2XNW85F&nA52P8_>4p%of5jTZ2(8 z8H3r2jpsQS8r+$0vrJ}KVapy#7|il1OV#~S!E)qc6}juURM{N4>noSk^^DDrYhAkU zad}~1s-BTtp@g+g|Gvd&=U!=D%PADSfE=LSz2)@mr^a1oq;;Dy1GOn^SdXfJzN_Vl zU8%Qp$3{Y%4dBT&6&o=yv2zvzl$3 z|KzX7FVebKbc*v>+f_c~pM%!S z+ph|Xf8URuhF$-A-4IPx&08mU+MMrQ<+lCg#Z3yUt58i_r<%AW_dl&kt4E3NB z+=C?HhZe(U0>Wnl>xBW-e;DJ36X1FP%4q0nv5~*mMHWeHiYE2J)R5%xeJ0 zXaL%H2=$5q$7llAa|7Ra0p@-J`<(^#n-1A_1pA~2>V^lyXamq~0KsMg+jRlocmnjK z8~m*f$!P)0Z3NzX2j6-G_KN`Xiv#P33hkI3_@NE@sS*9P5%!b?|IRi4k_P|LGyj?l z?VAbcmIn8x4gRkX|F<6etPlU54*#tdlT-omq6`1{dH*tp0{_Ax|HLZ(?Nx|P0{{7x%btf<*w-b^zOh1OL$~`jZ0x{o4QNdjI4@{^2eE=re>!0hn3=|JFj`js*Ll2mhY} z|Gy{y#xUfN1pmAr|L$G?%_jfqO#Zz zvj(Sm5C8t`enJ8N`H}zSRsZE;|LR{IRD28|GyId@pRaL0sr<_|Jqgm^NIe- zJ)v(6|JoArvnc=RYV?={{-_QA<4ynFU;nxq{NywLvm>y982|8V|Hvr+&_MsvP5&S6w2KysRR!*r2gHUDnq>&anj-(~hyUw1&XN@W z&JOR$Dd(sl*|{^hkskl=LTN7m(WWf&-Anc6IQr{A?2!cj??eC09rnpNy?71WmK*=x z9psxD;>s$qZwq2600?I$PXGV_IdoD^QveEB91;!~0Rja~te*ZYS^n~A;5D)&r~{ndZ|fT;ig3cX20K~z}7yw?X@lV=_W zaFC5-b$`BdK5Ga%i z>M@idqwKwpS*_zx+q?ds=Y3h?+4k-}_xt!UZ=Uzb_w#=ic_Sh5zlnmx|1EOz3J`g@ z|HhD)R#sfEuKpK@y83!WWoh~UXwVB()OEgG_0>HhRbQ?8Qb%27!5^Iq8AbIkR%PES z9O~?R`0&j`Nas-Dz3f$As4GhU0Y+LyM?1TvaOmM1&*Yo;4;2^hxc`0f)i)1^3R|+Z zb=Lo0NI7=3j*8^(aTYAo=7kn^RtMSdJ4)tT z@^znEab#ys5_Gj$}l3I8k|6Y<+WAH-%5pn#ukB^VN{RC!B z6oM3JvbTqu$B$>4yhoB*hMj{451u}KdKyLq8TQD4o6sfGB$UTcd*?hUtQHHw$+Wfo z8iF*kG$K9(a_}JBw6)F59MBIHir2ks?Nk=VW#4Pe%*?!a@uDr($l7p-Ixu2;@nRly zi>S+vTd4AmeYn-TP7A} z*GEJI1qBuFfS6&9D2@;a!psca-2vSqB8ozV47ImUi{ipKo?l}GmbM*2nc${^TJWtP z>_jxOd9wy5y_g{kZD?#PE4z=f4P}zsF)19vH&BnVva-g;$5lecV%fK6n&=z(+PM>V$VX05 zLzPfGqa?+J3}Hw^QBl!%keeu%1Ah7`;PTEHx%?fp-6R72pnF3T&tv*5$uCz!9{bn@lUb<#I**ob=>KNIG<`FKNmWx+_gixBL<>cVSi&TeluTAVs#P zM#tFL*l;@3epP-3QSD&b=e6CSyXSYl@sfxtmWlx$c#_% zPY4XuH!*Q2815F}4;`cBcHAM;kxRo(HiGWq0tXX){lLHkfB*Ops<)NayotqEwemKk zFhk1XwmJ);emm5pu~rSPyn)=LqZysR^E0j z3WdTJ2g8+J?qLXjq*f^iYm?RdKLwZ!+oi}epMExR2+K6 zQh+pxWs}rIP?`}AV5W(Ksr|vh&CSi8zeMLmk;xJL;@R9B93I@y8LTp;(&g<@7 z7F!OG3?7**K)&uC8;Af!W%_4iBn1a=-|kdh?L-EywnITletszwy3rQ0^Ojg|Un|SS z((ouCCn-fj=H}*l=H?*Lsgy)mJAiNyLtO0+Ps;GCGDQy2cf$q`cV92d`RG|rUPkJ3 zfW!=jp5Ffb`}IIFf@y+X5WE|*eKPO{6_jW}r8{mgC;FP3lMu|df{fG>RRkgDNf~7=!=e${0aR)-*^DAS)%+B^Z*C(paHn}5>YBVuNdU#}(T8apgla!se(vlb( z1XjY4gs?O{aN_+_LMSY{BVIM+U}&YX3_2q)Nt&ukc-_qL>N{S!!U_RMS63g0FmTuj z{vkCR@AiO%b_;{y4kJ_5l$At}l=3pt8Y|G)+!-EdKRVKn@n1c{f957ABJh@-kiz_# z%YRLGM3jezdpQ)eLPL@`YYH;+K8AIJJ?VjVJ-y~N|H-kRv)H4GaE$ zVCcoA^ULQw=Zfe1?Eq(%z_ zeeES#Op2+2i`{0|13gF1I4}Qb<}b5FcW+-fvhjfHDMN$979fp3?c>5%WOQyS2PHLE z1mBKaT5VesI2MNu3{Sak-FW1}gS&s8`B)<{=fkiE*Een4N=Q}AEOuLeDUAr1e2u9c z9Zv}%oy}V}Zo2*;Y{m@XJ@COE=PQ#EDk`2m-ab}p6GI|>wf$}535YQzoCBM#Uvb_u zSLW@9NaauOI-emWpiC_&aB$h`yCmrk_K?!n&8N`#3~*f&K?-8PIRMUu2QWdh5^}R4ISJXhdz?Xn zU9i;^Q-&@E)8gV{XlRGZ0q6P|a273ihXYOW5>j*V=x%Jn6dE9o3nG54#Ew-Ygn=*s zui?%_3=2kCFtDxI+&#vwp6zBvMGj77|4s-W|4gG15Kv~u zu8#Nznk88desq&13M{~ue4~O4j)nV4cN~)V+h-R1h^2r+i!{s+D#Se7(TA*x`Hn-9 zh}eh0Qbd9E_*(KH1*XC}CRdRFXYa0!lDNtYc0Rfu21INPJlF}Ct_fLMh$Cl~c*O47 zG%%p+pOZxm4Qr>~!B9=mHG9W%hgW~<#A9{OOlW-9w$Akw=UP+-qiYolaptVlC1x_$ zCJc@D9mN9@jLsz^=GqDJAWxU4PQPx&08mU+MMrQ<$!P%Pe*nX01IBzS$A2QlY6Rwwi_0n2Ox=a3xrn-0oq0nT+0_^%b`f(7V+0?}{);eQAEngi^R75t+I^rRd1jR5GF z3HY4~zi0%}aRknB3FCVK^Nj?pUIXTV0pWZC@`wWZsS^6B5ey9t5D^gw2@4@1A^fcm z?wttoqY4-p7$hYm^rj5ie*yo_G!qmRC@Ci^EGr%#9{bgZ|N5%>st(n50Opbe|NO!Kw;nDp zE&ut9|N5Q&wG{vT)%dFm)_MT{`?deqF!`O#*K<0RQ~Z|NYuQLO=e= zE!~I${@g3?j{s(9X8OM@@0bDMngx%Nj`+hQ|I9M#i~#?;B>T=JN=r!pw*vpD0sr}I zd3$;P{_OwT6aV&!{MIP$rw;YD9{=NR|Lk)(Iye9Gb^pf*sH>;wzBt}{0RQig>7ohk zt{lUR4*$Rs|DXZ?;$?AjaHwqz|LueS^Mo`g3R67_Yi(+mS_E}E0RQNHL^KQk`Czqr z3;*+M|L$k%&{y@iDbJ-S-IW68q!4FL4FB|x14`V6-<-;K7w;Od+3XEY1nwLnlxU>J_HpaqE@y9;x%Txc( zK3z*5qhSWVqZ@x>4y~*@00000ICN4@Qvmb)s|F350s$3O9x^Ce`TfLbrj^)={$Hj0 z^0-aC=S%yUzW9yrbM&bG=GSzwd*{yd+|AeH<$&1?eEWSk9y~T{*pke4NSr-;x36j9RbH~mcJ$mNYxxUvRtBj9cw?yaHs?--R zSr-kOw7wn(|3f>!#q8X9(CcVlAf?jyH6CrfHAwTiXY&_3DZZs%5ACk*L#j2p+Vo$J z^J!gSe8TH9TfYDf0ZPfCL+FhE&Ax>A!nzGQztDA!wQGtBD~rp${t6sCNVzDGJLVWZbU9F!NY3a_1j!!7t z?B(UPd-rbAg>5PcoQ6TK)AA3R7|0`Ub8y%UjLArp5;W%EP#y3AlLpjPC~w=hZ_l1R z<7E`2#KFM|qdR4v)Yr~Oo|`xAw{6=-%9J)$5bS_q`P)c@KZOR}K7AT)2sgI_DjVo9 z+fm1^5MhBckZaB{@6HY;mq+%w*HEUhhQ?Z~v!s8LFCZF}XW4lu8v-Q0f@#iaIG8(=C&R#93;1WDe z#p6d0Zr!@|=E0*Kt|CGxNQt?c(a_Lv_G(ND2g#W2KIxy0?IoS*=~)+$C+5iU_&9y`yqMLRqdv&&dZAEmok}Lh%+*B zM9JN|M;f?Ho{%FXSj>usk&%&aNFY)MWP&1_PC7g1tx#8OqP#4_?q)Vq5EFCtYDUIc zL3Lbmaz^i$&SZXjNfR$xjI3mr%&#vDm}%@7vidf&+d&U*PKM ziZVecLd1ODm2DM~KK&1rNFPMTB{eT#HF+X@RXzzy zHg;tjfoL`PbV1hyQFuws$%~Li2oiKffdDSd6%hpcLUCH)_f1R!b|_YmDgj?8;e<-ee*?b|t%exqFxgh&(l^hs(K6mz@p6CGIYzpz6Qh z+$05>n*Vh#M;_&?WP6nSS^x9)i{#k-j_mH*QWy8I@Cb{`PRb3rY}MsB5QW)z2PqV# zwY5XnPoBI!^b9mn+qY91AWAMT?S>&3>@KM-RRnq4xJ$wfoiDFa-XgP>n#6@dliXc` zY;A2bGjnosQd3jS%*>(?+cEfUlv!$O4k9x%)7Cb~8&wt>XR=gz(=tASCP5@?h-7JL zDU-=^b4^V_WEK!LMnC{MV``e43lm_PEe>R5vCDIw@@s^q85`_lh$H3O!W2tICNd#a zz!;>)G*_ljAWu-f3kqcD9PG3}^`$*?l}Vh16&VRknT?ILI20L62|6~-O&w}EiSj^) ztRWCowrcDzo7Oz1U}qGGG|Bp(whU#;#8^ezsCQzSqBrqrrwn@_kg=ijF3+W7U;gTv zu_lorR>n{qHdQ4(P3))zl~_!E9{&=HWtl^btxp@p;K>J%#Mmm-$0T;fxQC==(>xDD zBs&Rdk_{#$Cnt3bDJ+qTLYAqx-k6x2e4|Q6=8>Qd9rm0(?hhngV-ZBMfZ`0Su}R#S zl-Qb_*wS%*NTGnW&Q#P6^>*xSO>Rw8daMngu#iZT*pJ?Oiuic8Q?L&z&VbA$9=ws5 z2$ye7+}rX~W8+UPN%$Z!sd1bK3Jdm}J@@^`nnkftTqtF7$q$lM_3uqef?Tc17$iwb zYPt6otvls`!XVIuN86dxoKSIKl9*fj*?8w$iU<2!_98%#gx}m4trFYjd)xR~hk?h) z&?3^rY4*nxpN%y)c?P3489N%7L!(@fs_I_<4N|tZmX5C=R~*yA*9E}cF}q^>_3nNXkXP$+Y2@V<^hhX67enl8lgBuiYUDuOtVj{Z`S+bUvwAEo7G4<6ftT-XCqA zN%0T!))p|5QYwXo#B@ae{*n<*+8kO^Px&08mU+MMrQ<&VeJuX$0eb0LgF*$9Wpaf*<;;4eX#D z!+kgXnE}aZ0L^oC8Nz!f$8HGdju+;S8p&$|#%ct}aSrr}0rH6e$7liU zhz9$f1@fB~#AgD;au3LB1j=dw)pZ8uf(G)71JQ5*$Y}z|X#&!51HWbh$884eg#_w{ z3iOu?^rRd7tq$gW0q1`M_M#KX`=r-YoyVEc>qw|HC)`w;KPvDF6DY&20ex z(lhz04*&Rj)_DNhfdSZl0Qsy6|IIV@rVQnh1m~3p|NO%L+%4ja1ONTW|G_i{nh`%CI8qd|M`s3asca<0{`}8|Kc_O+bV%Z z0sr++|M7SKtOEbUF8?Ek+KYA^t% zWCQS<0{`}j|L~Ci-!cEcG5^(2-lZS$w;|%TE`v-2^P>gupauWwcmL{v|J4uw%|EVx z75&pi|J_pm=3CdE6rW-Q?Z`Oji~;}s-2dGc|L|-7?_1WsIRD{n|LSf3$TEAOj)-3OJMzlqS-V zqAo?cbfwz6v+I8My%z{UotfP|`#(7ca_^TP-@UK!A|mp?iIm9yEs`WD2ubq4F-W2^ z%1U2q{1Kw@rINCY=zr9|l)SRW7k}BJd6|gjmcM?Xp(OuXEd^zbKW}+?`CM0B-EAy& zbzSE!zufXS4Q0{az=$esRv$bI%I#x4=i`#EUQLcWKeXpISZ4>-H!J;GYN;~D)ZOk$ zHr;va@K0C{-#X%cwP{x!PC6NicXMEuMQE_nb{y!tAcUp$IZ z${Vr<&vm6IJDxakA~W+AO6QH5*$oeNW&0Nf`fyPA9p|QUYh{YXYwC)5^78jLfqZ?m?Pc@Qeep49VSf=mY5^y z13f9gy7Cfgw{PEW3Yo|7AOk~8O`&3Y$ygq&?!Eg&+9aEr_qyip-Mjnu@5eO^<7du* zX(%eu&r6lv^uA-KydrDxaz{x?iJ#x!AsB}B#t&|wIX^!rs|^~=TA{S~ZXv$`HYmR~ z87_#ZC_g{GhH*GS7=C_HQE<^D*LGZv&)Oif=tNu(3#+fG1u6L}NL183hX*GBfmtZ3 zt*tGG#jO&$S|pdl!Xi3qYHH%*;w&w{gFroMZazr(3e8wrLJ^d91j6FPg^t)~&`^0t zM`vdz6vZ92L?Jjl-{A!0=uxPE!j6u7<5bzt7n}o<;w!S^Bl7bL3ky5{0SOHaJ&MIr zAlLxRgrbU0D9z8WAIwry6ly5zWn+GRep6Fn;d#iZZ?T|{h4MciijSe`^My@?O;BEr zS`s{&makZrno?eVP&72F2y&M)Au8cS&l!T2{Pv zBa#|hE1y)hJ|Aj3uzNSWgmU0OTiej{*2>Dt)<)zs*6U`et(re8PINu)_8X9`=q>snoWM}kMylrf(uQ$k&-8z3RiLa1NHMmnBSkc1C z;S5|K;N)cY_8fwIzaT>}($mi%N*H~W_4V4cR9Ut8(|f(zg;Wy?rJ0vAFx!6bpZD5l z2S#4}h;m3E*uaaCf&NEO@@RG-C!>Nwq3JABTgP9U!sluibWCW?xtw>8dM-kKJan(U zXYbdGvv+ULqkD(GMm0T;-W6q3&}gm~)c(Xjm1MU5a6!jlCL_E5c{-Vuo}L~Pa}gEq z70SiEXz*f843x0Q>CgK)>=_f+3qNdK%|G}{0D3jb=pFO%A+tO^OG{%QUS8jLeIpbv zFEmqH>d7LLeSF6H8I!L7TEag86cO|$cUC_r$S2Xm!_%o0s68lN3jza3DfM*n^zcaZ z2?~1Kzsj$6(Ebr;qLJ8prC}r#K=e|Cqu(++qNCsBa~w>XlUr<6d9SAm=F{c z9zH%YiJ)~P99jnubF?BNVor*Ujok$SOCZ~}p)r`6b0Wffi|r7EA2um~t}q`Kq@>JC zVBwNra0Yi0kdu;9P{9P~_xMdoN@dB-E0=VQ4R9#v_4vbP$isx#1r**+WWxBvist5u zYp)SxV616#MF6@F&<-35zcw|Rl*{077`db4v4*=01>zK&INpqA8I0_VXWbaIK2oqmjn<{|T5Z8z*{mjW~ zv^HAB@Vox@_V#{uOL1{a60gz|1q|#bYkGM03Czx-SvHT|O)D@4=;WFEV#MKB=4;(?) zBpY4h0@`UvTAF!lAGDv;o9^MkBI1F?+y}y{gujtATzMKbbm@w{|3>~LBD2=onOem} zErH~nhNS+!h9^(SWDkLm%^^NMwzgxf(+$&=fv6#yDyp;9TK+|=@H_e$hw4kvl7UH? z%V}?*%%47gnvOn5o{Kr==-b++rRk3~)_c@K0}RaY9cU;-&;AU3?flJON-WyS znObcJiz6h8wss&;2!o(bY|*Q1S$;Akx za10N6r#-=YTt+y$HcTx&`!Kqk;A4csC(+i<4sIc^I7}*YF+vjLP9UB;chEb^RL=0v zWw^708RF*PY!&_Kf~N>^Ij(;I)y>X=kZ2$o8EFfJGQlny{+Vzr?A)jU{&YFTgR=8m%uGc= z_#skeWi;I%Ng+%foWd-~#Kg>-smbCD267r+X}Y@VFu|b)z`|CpUi4HeA}-Cf4#OqX zlxv8C{TqjXXU`^!*%{fKBDgd1lKOhPHK8452<~vD6(uDXK~k!6xa1aMXJMhkEQUKX zgHx0PDJo**jy!JYt?c&I)xjNyfG{hrnd(vz$;B;+ikZQNBT35+nwlQDjxsVb+}AMO zTXC&B#TU8(#xBGSb6|rcL?jnOk|Gi+W^`-+0DG#N7Br=K4NrG>K?*wHfZ&Xl8x?mx zk$;Z^LnIM-6*I22zq37RDg@gqEILFiEbI`YN44iWr1wCWBq0X-$aV|_y zFvcOk+25Kgr}6;@lO#n|rK9OqVbBy%$U+*H5TqamwPPJEttuh;0SA*vOT|~BjCSf{2QhCvxRcoDDQ!Ueo+pJ=pQApJ)|zPykyfJ`~N1cF(E)EF(q z_TH#W^XP|QEW2;}l289Q6XB)RE^OAkuG8K+Z_47dExD9A=cH~mZHx=c;Qbi$hy-Vv z2a8PJIV4F$!Km%DJJa2hmW6=K>HJc0Dx!E>_Z59%wCY*Q@n-$Oe2pc6fKmce3dpep o|NfEz9NHD66qT{yGLoZw0H4`6Fj(b*eEPx&08mU+MMrQ< z_LK$qp$+?~5uc83($2=n!@1C-a?ET3%WMO{WdfyC1JcdIql8t<$H4lm4F9?z@}LL* z&olk75dYsS{j?PC@bLfAGXL&C|N5){{J`_|_y5u;|K%^6l7|2Igy`qw>6ZrgsR{r2 zjQ{(x|IRA^%QXM+N$c$D_xk<+#47%`8vn&K|JpA9$}04w3je|}jZgvq!6e(*(*O8* z|MzMC_i_LEm*3pi?wtt#$T9!wHr|E;|NO_*(aQh7EC2n}|Jf*{oR9zZVgL47*?s_h zLIMA~1poS_)_4H_)-3<71ON3=|JE_u)y@C=x&Qpl|GX#T;@|)IlK=O9=8^>e?{&+> zxr9mqkM z74Mn?|L~Ik@MNl^nf>KTmRkbX&AX>~CI94Ev8$o$;KKjF68z#m&bEiHrkmNoHtwn) zVXiK#W=l%exQ4-l`he!IaC25%sz*zM^jb z_M*v}9JH>b%BCap;5pNp6R?aS`^_ct*jT}%GV8^6>Ap$enHQN<0;PEn`Q3i?)hqn_ z=+clJo_V!ZOLS6BQvg%`Xb2ns0R$5MD*jUUC#-k%N99hYlsWpO zW%s{-xPF9a^}(0N=8mA5w8rmUo2vcyvfSm=p|t7ouJO{^<^J8?_0Yr1p1|_&xj?tu z000WDNkl2v2r5T_ z5E4=)1_^=^mc&E|3IRlf2*?#Kxk`l~U|A3W5k$GW+_JdB|%Qd!gl0yo^<-AkQ8`rP!LPEx6(qa&`u zdJQ>}(|RKY@58cE?PXcn=4mI>1O_ysE`?OPLU!pihhl5xCB;KRSn9@vaF^RG+0$tbqVfYfb7;~ zk3n>o@b(2*b_teOVo8RFY0g6Plz__0^z=)RQ7=tI3EN`A}mb6i13+h|)F^3KwI*7k@e;hn?h+J+NzFYegnBwBf!9nXX%S{#rdvBRy%^6gy;bu#I z*N~*Kr_;+T-|Nt%zy(NWeF0zcw50`>s?~#>kQB>>XMR@POm3z zMHJ=WH6;$fsEXDPcy+=;wVLV431OILt=4~!GW#OSoDt0vc{1&Z+>b~DwUP#ns-Qk-Jg@H#X&NL$UElyCm)m=AeSAW! zZML49HAe?o%{sT0;Ts*`!{vH7OQn8D1r5~lRSlZf9v7WcU#n?Q@rSyCGHm_){G?K6 z4=&dyAlet9q4cVNCFoMf^5MFl{%wyJRrc@ zmjTdX>az-cF+e-*d;vP1l;jLjD8%+Azp$`X^2$XIHNb0rK`03_XCjjQ{;p}mf-=KQc6=dsY>n9(6{ThZ3e+?RV#BY#1d<+gO^MO5s5&55= zXl*e-TN!q~a3)dm@Bl|MbpJsGY@gqgj*hBy`4I)^h@C^!)tqa=fI<(n_RAlKz0c2i_{;;Ia_8O z#UYWBL?Ui)e~H}Q9-bDdeV;dhZauA+<5K~bnevjZ`yvrS*1p);(S0Y^>L1!q!bxvf z5)ULMJdg1FLi(K0VRziB>p=%X?l8gO;e?ZeNa&4}6wTxD;Ggp(>(zZM zfSg!vid%K}2}~SMSinOE3L+!=i|P0q8w}~ZKa>Q`@pv4L$kP+3VQ>Iqv6YHj$p!V4 z0t`SLo~|JOz0lcb`ZqS(oVlbo25ek3B?0xMgE&x7_Qfm7kH(mkwgw3J!{8@(^i_K=UNMp)u3gS=-INdM0DTfp^cdg;K z>=jf=(A?OwK7|__>TrXCoLEX>cD8_mIAd={yT^FE?w3V{c9eq!eLU$?#M-%$XVPBu zK1T(5auW)aY$qotcD6tu%vM~#L7+F9B1z%si(E}nt*;G(;SY`Ni!>NDZQgjA z%B_n+H#Q5uTqqW2Q;s-CArt~s*VfA$Yt=L~!23Vsg&jK3SooxZ=6&^du@DMzqPANp zPRPNREVeKwE&(9%^&4*+dwFjpDn7g+NmQy@GH7gU@`Q}&i~c0Qkx)!rUV?z?FCi}v zp^H~;x8?SB7fSez{dZr%w}gjwSPwey%{QC;(Rk6H#Y)&GH=#H#m2%ihK|*R`UXDO0 zh`W(o)R0>!QPqyYyHL;FyWrrUp|Ooirv6a0=*zsEoScL__*!t-_*}##-b;)VV`sPD z{`|8BS{j9?0uG3-9&qN)rqQPUIP=lH#MH#ZvseS}ezrL8-qrS0=nM+F{kBLW;qzO& z!GY=Qq#S7MQxl`95bcl6?Pt%TBczT$E+Fa~UpilwDxT^-O25eo%jehyHEVGv#Yrq01af|pstT>uC) zHJEuNSb-LBV2>c-+ z+P*;=NmCNB5as|&ae(x5K+Rw$d8|!Ii4$xED!`)w01zhm^l)T;ng9R*07*qoM6N<$ Ef}M+JX#fBK literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/thinking-face.png b/src/modules/cs/static/emoji/thinking-face.png new file mode 100644 index 0000000000000000000000000000000000000000..7431ed01ee4c7e13e09787212877cd20b3c99532 GIT binary patch literal 3763 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 BK~Vqz literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/upside-down-face.png b/src/modules/cs/static/emoji/upside-down-face.png new file mode 100644 index 0000000000000000000000000000000000000000..677f7460ac9f189e797e2ea4c1427d9c507b4165 GIT binary patch literal 3688 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 AE&u=k literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/weary-face.png b/src/modules/cs/static/emoji/weary-face.png new file mode 100644 index 0000000000000000000000000000000000000000..74325b9be3d81ceeb8456ef4e243348185344c45 GIT binary patch literal 3726 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 literal 0 HcmV?d00001 diff --git a/src/modules/cs/static/emoji/worried-face.png b/src/modules/cs/static/emoji/worried-face.png new file mode 100644 index 0000000000000000000000000000000000000000..8638920c4bf8da93de56986ce9f1a647b9342db1 GIT binary patch literal 3655 zcmV-N4!H4&P)Px&08mU+MMrQ<$$})!fgJm;8T_LL%8M<_eH-zf6#1qH zy_HkSeICkc0Lph2#c2cfpby1p0{4vo%W46|Z3o|m7VCll#%>A5Y6Qe=3CD2^-g*Gb zdlk!S0mO3?%WVVrlLG351m}?*-HIUWl^Vfj0oQc_)OQN+mjmX73*CMU@RbnnoF2q! z1jTCx#%KZSf&%oI3;L)J(QyLBX93}V2knRj(QpCuj05F;0rr{+_o5K=p&0n56!xSP z|F|Ikwi^HAE%>Sn{`Go(+ zD)68O+k^t~qYV784*&hi|H(A}{m}pRWBX`_mV*%NJ0sr}X|L;rx z%ryVtEdT!2Z!`e^$|nEMF#p{&bU6V3{^r+u0OgGW|H~}@#W4TCBm1fj|M`}bR|435 z0sqA$|J^J9(kcI=0{{Kn|N5x^!v=>-0sifP|MN`$_;dgJwg3CE{gwd#)hhq~;ekZ~ z|J4!y^-=%!SMik#|G+Wzo(=uoKL6-V|FZ-CsRIAH1pnwg|L|b{!YKdmQl4J{jZXmX zlL7zcMg5)u|Km9S-bw%OSpUondq4pG-)P-|0srDZ|I|PKt_1(+b%am{>!~9C)>Qw) zG46=~|I|nS@sy8K0srSz|G5$W^NRo2H~;jUwwXhteJ$jJ0Ku&k2&iWS zr+XccUJ|!{4D!n+|Jqu+jwIW&J>$}8^yzy0$tL&1G_!*j-@HK9u`Z-^49uoD`R{G= z-f)_19`4mO_p~L!m?O@X73#k$&!Zgf#zXhjE5VWz|K1+$!z0YKO=2bh|HKvO)m!Mt zQ~%K)^}Z?Aj1c+JC`{)iL;wH)J#e}JK;L|V>!>9lN3YzRzI}jH zK_U{QBSqTMk#>%2S$l6=-gjq)K+yHM@4d4>JjcnsGe7>{H*;qu$jSYGP?7s1A(Q`C zgR;ERI?c7(n>TOPUaPiTLGiy6Dk`nr{I|bvo1=FPp*LsSKQ?KvQy~4`YE;*5`s;DM zYp<)$5mdc?ea(H_*3Ih_e}{&oq`lR?AhW9K+;bhrLzhcRN-kLLd=9eAqPlIHH03|m zqpJOvLeQLhem={_|H#>IfwLK&4qbO@o+Fj|Cbf@BrKq_z9cd~$9B|1LXU}?u)yJL7 zEGk^RLFpsnY*<}blzFbbYwwYaBN-VPX9-~Q2;kq5UzJ&qzE$%-T!G|uYs3(zrggvP zUQbU?I5L9(Heq1z-o5@=W+;^Q2Ro Y1jsXlu@00-eN4BP_4{`*h1R~6OeY*+Yz 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 N002ovPDHLkV1m#$SbqQj literal 0 HcmV?d00001 diff --git a/src/modules/cs/types/index.d.ts b/src/modules/cs/types/index.d.ts new file mode 100644 index 0000000..2256fa8 --- /dev/null +++ b/src/modules/cs/types/index.d.ts @@ -0,0 +1,13 @@ +export declare namespace Cs { + interface Content { + type: "text" | "image" | "voice" | "video" | "file" | "link" | "location" | "emoji"; + data: string; + } + + interface Msg extends Eps.CsMsgEntity { + content: Content; + sessionId?: number; + } + + interface Session extends Eps.CsSessionEntity {} +} diff --git a/src/modules/cs/utils/index.ts b/src/modules/cs/utils/index.ts new file mode 100644 index 0000000..9499acb --- /dev/null +++ b/src/modules/cs/utils/index.ts @@ -0,0 +1,52 @@ +import dayjs from "dayjs"; +import type { Cs } from "../types"; + +// 消息格式化 +export function msgFormatter(msg: Cs.Msg) { + if (!msg) { + return ""; + } + + switch (msg.content.type) { + case "text": + return msg.content.data; + + case "emoji": + return "【表情】"; + + case "image": + return "【图片】"; + + case "link": + return "【链接】"; + + case "voice": + return "【音频】"; + + case "file": + return "【文件】"; + + case "video": + return "【视频】"; + + case "location": + return "【定位】"; + } +} + +// 日期格式化 +export function dateFormatter(date?: Date) { + const t = dayjs(date); + + // 在今天之前 + if (t.isBefore(dayjs().hour(0).minute(0).second(0))) { + return t.format("YYYY-MM-DD HH:mm"); + } else { + return t.format("HH:mm"); + } +} + +// 表情链接 +export function getEmoji(name: string) { + return new URL(`../static/emoji/${name}`, import.meta.url).href; +} diff --git a/src/modules/demo/config.ts b/src/modules/demo/config.ts new file mode 100644 index 0000000..5c3e17d --- /dev/null +++ b/src/modules/demo/config.ts @@ -0,0 +1,7 @@ +import type { ModuleConfig } from "/@/cool"; + +export default (): ModuleConfig => { + return { + components: [() => import("./views/crud/components/code.vue")] + }; +}; diff --git a/src/modules/demo/directives/color.ts b/src/modules/demo/directives/color.ts new file mode 100644 index 0000000..8c5b1c6 --- /dev/null +++ b/src/modules/demo/directives/color.ts @@ -0,0 +1,5 @@ +export default { + created(el: HTMLElement, binding: any) { + el.style.color = binding.value; + } +}; diff --git a/src/modules/demo/service/test.ts b/src/modules/demo/service/test.ts new file mode 100644 index 0000000..880a801 --- /dev/null +++ b/src/modules/demo/service/test.ts @@ -0,0 +1,154 @@ +import { Service } from "/@/cool"; +import Mock from "mockjs"; +import { uuid } from "/@/cool/utils"; +import { orderBy } from "lodash-es"; + +interface User { + id: string; + name: string; + wages: number; + status: 0 | 1; + occupation: number; + avatar: string; + phone: string; + createTime: string; +} + +// 模拟数据 +const data = Mock.mock({ + "list|66": [ + { + id: "@id", + name: "@cname", + createTime: "@datetime(yyyy-MM-dd)", + "wages|50000-100000": 50000, + "status|0-1": 0, + account() { + return Mock.Random.string("lower", 8); + }, + occupation() { + return Mock.Random.integer(0, 5); + }, + avatar() { + return Mock.Random.image("40x40", Mock.Random.color(), "#FFF", "png", this.name[0]); + }, + phone() { + return Mock.Random.integer(13000000000, 19999999999); + } + } + ] +}); + +const userList: User[] = data.list; + +@Service("test") +class Test { + // 分页列表 + async page(params: any) { + const { keyWord, page, size, sort, order } = params || {}; + + console.log("[test]", params); + + // 关键字查询 + const keyWordLikeFields = ["phone", "name"]; + + // 等值查询 + const fieldEq = ["createTime", "occupation", "status"]; + + // 模糊查询 + const likeFields = ["phone", "name"]; + + // 过滤后的列表 + const list = orderBy(userList, order, sort).filter((e: any) => { + let f = true; + + if (keyWord !== undefined) { + f = !!keyWordLikeFields.find((k) => String(e[k]).includes(String(params.keyWord))); + } + + fieldEq.forEach((k) => { + if (f) { + if (params[k] !== undefined) { + f = e[k] == params[k]; + } + } + }); + + likeFields.forEach((k) => { + if (f) { + if (params[k] !== undefined) { + f = String(e[k]).includes(String(params[k])); + } + } + }); + + return f; + }); + + return new Promise((resolve) => { + // 模拟延迟 + setTimeout( + () => { + resolve({ + list: list.slice((page - 1) * size, page * size), + pagination: { + total: list.length, + page, + size + }, + subData: { + wages: list.reduce((a, b) => { + return a + b.wages; + }, 0) + } + }); + }, + Mock.Random.integer(300, 600) + ); + }); + } + + // 更新 + async update(params: { id: any; [key: string]: any }) { + const item = userList.find((e) => e.id == params.id); + + if (item) { + Object.assign(item, params); + } + } + + // 新增 + async add(params: any) { + const id = uuid(); + + userList.push({ + id, + ...params + }); + + return id; + } + + // 详情 + async info(params: { id: any }) { + const { id } = params || {}; + return userList.find((e) => e.id == id); + } + + // 删除 + async delete(params: { ids: any[] }) { + const { ids = [] } = params || {}; + + ids.forEach((id) => { + const index = userList.findIndex((e) => e.id == id); + userList.splice(index, 1); + }); + } + + // 全部列表 + async list() { + return userList; + } +} + +export default Test; diff --git a/src/modules/demo/service/user/follow.ts b/src/modules/demo/service/user/follow.ts new file mode 100644 index 0000000..64bf0b6 --- /dev/null +++ b/src/modules/demo/service/user/follow.ts @@ -0,0 +1,6 @@ +import { BaseService, Service } from "/@/cool"; + +@Service("demo/user/follow") +class DemoUserFollow extends BaseService {} + +export default DemoUserFollow; diff --git a/src/modules/demo/service/user/info.ts b/src/modules/demo/service/user/info.ts new file mode 100644 index 0000000..d47ff1a --- /dev/null +++ b/src/modules/demo/service/user/info.ts @@ -0,0 +1,33 @@ +import axios from "axios"; +import { BaseService, Service } from "/@/cool"; +import dayjs from "dayjs"; + +@Service("demo/user/info") +class DemoUserInfo extends BaseService { + // 测试方法,使用 request 请求数据 + t1() { + return this.request({ + url: "/t1" // 测试地址,实际项目中请更换为真实接口地址 + }); + } + + // 自定义请求,通过 axios 返回数据 + t2() { + return axios({ + url: "https://" + }); + } + + // 自定义请求,通过 Promise 返回数据 + t3() { + return new Promise((resolve, reject) => { + setTimeout(() => { + resolve({ + date: dayjs().format("YYYY-MM-DD HH:mm:ss") + }); + }, 1500); + }); + } +} + +export default DemoUserInfo; diff --git a/src/modules/demo/views/crud/components/adv-search/base.vue b/src/modules/demo/views/crud/components/adv-search/base.vue new file mode 100644 index 0000000..a0936e0 --- /dev/null +++ b/src/modules/demo/views/crud/components/adv-search/base.vue @@ -0,0 +1,138 @@ + + + diff --git a/src/modules/demo/views/crud/components/adv-search/custom.vue b/src/modules/demo/views/crud/components/adv-search/custom.vue new file mode 100644 index 0000000..bed54c6 --- /dev/null +++ b/src/modules/demo/views/crud/components/adv-search/custom.vue @@ -0,0 +1,152 @@ + + + diff --git a/src/modules/demo/views/crud/components/code.vue b/src/modules/demo/views/crud/components/code.vue new file mode 100644 index 0000000..7b8ffb1 --- /dev/null +++ b/src/modules/demo/views/crud/components/code.vue @@ -0,0 +1,41 @@ + + + diff --git a/src/modules/demo/views/crud/components/crud/all.vue b/src/modules/demo/views/crud/components/crud/all.vue new file mode 100644 index 0000000..81b8760 --- /dev/null +++ b/src/modules/demo/views/crud/components/crud/all.vue @@ -0,0 +1,574 @@ + + + diff --git a/src/modules/demo/views/crud/components/crud/base.vue b/src/modules/demo/views/crud/components/crud/base.vue new file mode 100644 index 0000000..2189ae6 --- /dev/null +++ b/src/modules/demo/views/crud/components/crud/base.vue @@ -0,0 +1,145 @@ + + + diff --git a/src/modules/demo/views/crud/components/crud/dict.vue b/src/modules/demo/views/crud/components/crud/dict.vue new file mode 100644 index 0000000..f694a6e --- /dev/null +++ b/src/modules/demo/views/crud/components/crud/dict.vue @@ -0,0 +1,164 @@ + + + diff --git a/src/modules/demo/views/crud/components/crud/event.vue b/src/modules/demo/views/crud/components/crud/event.vue new file mode 100644 index 0000000..12dce33 --- /dev/null +++ b/src/modules/demo/views/crud/components/crud/event.vue @@ -0,0 +1,182 @@ + + + diff --git a/src/modules/demo/views/crud/components/crud/service.vue b/src/modules/demo/views/crud/components/crud/service.vue new file mode 100644 index 0000000..fb561c8 --- /dev/null +++ b/src/modules/demo/views/crud/components/crud/service.vue @@ -0,0 +1,185 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/children.vue b/src/modules/demo/views/crud/components/form/children.vue new file mode 100644 index 0000000..e3a6451 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/children.vue @@ -0,0 +1,118 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/component/index.vue b/src/modules/demo/views/crud/components/form/component/index.vue new file mode 100644 index 0000000..35a1300 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/component/index.vue @@ -0,0 +1,124 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/component/select-labels.vue b/src/modules/demo/views/crud/components/form/component/select-labels.vue new file mode 100644 index 0000000..4d01f0d --- /dev/null +++ b/src/modules/demo/views/crud/components/form/component/select-labels.vue @@ -0,0 +1,38 @@ + + + + diff --git a/src/modules/demo/views/crud/components/form/component/select-status.vue b/src/modules/demo/views/crud/components/form/component/select-status.vue new file mode 100644 index 0000000..22b7372 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/component/select-status.vue @@ -0,0 +1,43 @@ + + + + diff --git a/src/modules/demo/views/crud/components/form/component/select-work.vue b/src/modules/demo/views/crud/components/form/component/select-work.vue new file mode 100644 index 0000000..7ff2c0e --- /dev/null +++ b/src/modules/demo/views/crud/components/form/component/select-work.vue @@ -0,0 +1,59 @@ + + + + diff --git a/src/modules/demo/views/crud/components/form/component/select-work2.vue b/src/modules/demo/views/crud/components/form/component/select-work2.vue new file mode 100644 index 0000000..acc0eb6 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/component/select-work2.vue @@ -0,0 +1,38 @@ + + + + diff --git a/src/modules/demo/views/crud/components/form/config.vue b/src/modules/demo/views/crud/components/form/config.vue new file mode 100644 index 0000000..0a4e938 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/config.vue @@ -0,0 +1,122 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/crud.vue b/src/modules/demo/views/crud/components/form/crud.vue new file mode 100644 index 0000000..dc7fac3 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/crud.vue @@ -0,0 +1,154 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/disabled.vue b/src/modules/demo/views/crud/components/form/disabled.vue new file mode 100644 index 0000000..a58db8e --- /dev/null +++ b/src/modules/demo/views/crud/components/form/disabled.vue @@ -0,0 +1,64 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/event.vue b/src/modules/demo/views/crud/components/form/event.vue new file mode 100644 index 0000000..b5fc7c0 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/event.vue @@ -0,0 +1,93 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/group.vue b/src/modules/demo/views/crud/components/form/group.vue new file mode 100644 index 0000000..52cb362 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/group.vue @@ -0,0 +1,105 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/hidden.vue b/src/modules/demo/views/crud/components/form/hidden.vue new file mode 100644 index 0000000..c1edfda --- /dev/null +++ b/src/modules/demo/views/crud/components/form/hidden.vue @@ -0,0 +1,77 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/layout.vue b/src/modules/demo/views/crud/components/form/layout.vue new file mode 100644 index 0000000..4ebf73b --- /dev/null +++ b/src/modules/demo/views/crud/components/form/layout.vue @@ -0,0 +1,98 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/open.vue b/src/modules/demo/views/crud/components/form/open.vue new file mode 100644 index 0000000..f60ece1 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/open.vue @@ -0,0 +1,83 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/options.vue b/src/modules/demo/views/crud/components/form/options.vue new file mode 100644 index 0000000..a300b12 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/options.vue @@ -0,0 +1,172 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/plugin/index.vue b/src/modules/demo/views/crud/components/form/plugin/index.vue new file mode 100644 index 0000000..a2c7ecd --- /dev/null +++ b/src/modules/demo/views/crud/components/form/plugin/index.vue @@ -0,0 +1,109 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/plugin/role.ts b/src/modules/demo/views/crud/components/form/plugin/role.ts new file mode 100644 index 0000000..68e84b8 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/plugin/role.ts @@ -0,0 +1,20 @@ +/** + * 角色权限控制 + * @param role + * @returns + */ +export function setRole(role?: string): ClForm.Plugin { + return ({ exposed }) => { + function deep(arr: ClForm.Item[]) { + arr.forEach((e) => { + if (e.role) { + e.hidden = e.role != role; + } + + deep(e.children || []); + }); + } + + deep(exposed.config.items); + }; +} diff --git a/src/modules/demo/views/crud/components/form/required.vue b/src/modules/demo/views/crud/components/form/required.vue new file mode 100644 index 0000000..41885a1 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/required.vue @@ -0,0 +1,75 @@ + + + diff --git a/src/modules/demo/views/crud/components/form/rules.vue b/src/modules/demo/views/crud/components/form/rules.vue new file mode 100644 index 0000000..03d4ab8 --- /dev/null +++ b/src/modules/demo/views/crud/components/form/rules.vue @@ -0,0 +1,123 @@ + + + + + diff --git a/src/modules/demo/views/crud/components/other/select-user.vue b/src/modules/demo/views/crud/components/other/select-user.vue new file mode 100644 index 0000000..e41579c --- /dev/null +++ b/src/modules/demo/views/crud/components/other/select-user.vue @@ -0,0 +1,287 @@ + + + + + diff --git a/src/modules/demo/views/crud/components/other/tips.vue b/src/modules/demo/views/crud/components/other/tips.vue new file mode 100644 index 0000000..b4b3a7c --- /dev/null +++ b/src/modules/demo/views/crud/components/other/tips.vue @@ -0,0 +1,168 @@ + + + diff --git a/src/modules/demo/views/crud/components/other/tsx/index.scss b/src/modules/demo/views/crud/components/other/tsx/index.scss new file mode 100644 index 0000000..15b5da5 --- /dev/null +++ b/src/modules/demo/views/crud/components/other/tsx/index.scss @@ -0,0 +1,28 @@ +.tsx-list { + .item { + display: flex; + align-items: center; + justify-content: space-between; + border: 1px solid var(--el-border-color); + padding: 10px; + margin-bottom: 10px; + cursor: pointer; + border-radius: 4px; + + .el-icon { + display: none; + } + + &:hover { + background-color: var(--el-bg-color-page); + } + + &.is-active { + color: var(--el-color-primary); + + .el-icon { + display: block; + } + } + } +} diff --git a/src/modules/demo/views/crud/components/other/tsx/index.tsx b/src/modules/demo/views/crud/components/other/tsx/index.tsx new file mode 100644 index 0000000..5903a9f --- /dev/null +++ b/src/modules/demo/views/crud/components/other/tsx/index.tsx @@ -0,0 +1,107 @@ +import { defineComponent, ref } from "vue"; +import { Check } from "@element-plus/icons-vue"; +import "./index.scss"; + +interface Item { + name: string; + value: number; +} + +export default defineComponent({ + emits: ["checked"], + + setup(props, { emit, expose, slots }) { + // 列表数据 + const list = ref([ + { + name: "鸡腿堡", + value: 1 + }, + { + name: "牛肉堡", + value: 2 + } + ]); + + // 选择值 + const active = ref(); + + // 是否可见 + const visible = ref(false); + + // 打开 + function open() { + visible.value = true; + } + + // 选择 + function toCheck(item: Item) { + active.value = item.value; + + // 自定义事件 + emit("checked", item); + } + + // 暴露方法和变量,使上级可以使用 ref 的方式来调用 + expose({ + toCheck + }); + + // 必须返回一个方法 + return () => { + return ( +