diff --git a/backend/.gitignore b/backend/.gitignore new file mode 100644 index 0000000..7b4c556 --- /dev/null +++ b/backend/.gitignore @@ -0,0 +1,95 @@ +# Byte-compiled / optimized / DLL files +*.py[cod] +*$py.class +__pycache__/ +# C extensions +*.so + +# Distribution / packaging +.Python +env/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +*.egg-info/ +.installed.cfg +*.egg + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*,cover +.hypothesis/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +target/ + +# IPython Notebook +.ipynb_checkpoints + +# pyenv +.python-version + +# celery beat schedule file +celerybeat-schedule + +# dotenv +.env + +# virtualenv +venv/ +ENV/ + +# Spyder project settings +.spyderproject + +# Rope project settings +.ropeproject +.idea/ +*.db +.DS_Store +__pycache__ +migrations/ +*.pyc +config.py +db.sqlite3 diff --git a/backend/README.md b/backend/README.md new file mode 100644 index 0000000..84dc5fe --- /dev/null +++ b/backend/README.md @@ -0,0 +1,6 @@ +## 开发 +~~~ +python3 manage.py makemigrations +python3 manage.py migrate +python3 manage.py createsuperuser +~~~ diff --git a/backend/application/__init__.py b/backend/application/__init__.py new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/backend/application/__init__.py @@ -0,0 +1 @@ + diff --git a/backend/application/settings.py b/backend/application/settings.py new file mode 100644 index 0000000..35a6a21 --- /dev/null +++ b/backend/application/settings.py @@ -0,0 +1,295 @@ +""" +Django settings for application project. + +Generated by 'django-admin startproject' using Django 1.11.21. + +For more information on this file, see +https://docs.djangoproject.com/en/1.11/topics/settings/ + +For the full list of settings and their values, see +https://docs.djangoproject.com/en/1.11/ref/settings/ +""" + +# 导入全局环境变量 +import datetime +import os + +from mongoengine import connect + +from environment import * + +# Build paths inside the project like this: os.path.join(BASE_DIR, ...) +BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) + +# Quick-start development settings - unsuitable for production +# See https://docs.djangoproject.com/en/1.11/howto/deployment/checklist/ + +# SECURITY WARNING: keep the secret key used in production secret! +SECRET_KEY = ')ns9h-%fl^&ro=+-vgl*b-!+a%2=tuwc#&xbpmcavj0*ufpyjh' + +# SECURITY WARNING: don't run with debug turned on in production! +DEBUG = True + +ALLOWED_HOSTS = ALLOWED_HOSTS + +# Application definition + +INSTALLED_APPS = [ + 'django.contrib.admin', + 'django.contrib.auth', + 'django.contrib.contenttypes', + 'django.contrib.sessions', + 'django.contrib.messages', + 'django.contrib.staticfiles', + 'rest_framework', + 'corsheaders', + 'captcha', + # 自定义app + 'apps.system', +] + +MIDDLEWARE = [ + 'corsheaders.middleware.CorsMiddleware', + 'django.middleware.security.SecurityMiddleware', + 'django.contrib.sessions.middleware.SessionMiddleware', + 'django.middleware.common.CommonMiddleware', + # 'django.middleware.csrf.CsrfViewMiddleware', + 'django.contrib.auth.middleware.AuthenticationMiddleware', + 'django.contrib.messages.middleware.MessageMiddleware', + 'django.middleware.clickjacking.XFrameOptionsMiddleware', +] +# 允许跨域源 +CORS_ORIGIN_ALLOW_ALL = CORS_ORIGIN_ALLOW_ALL +# 允许ajax请求携带cookie +CORS_ALLOW_CREDENTIALS = CORS_ALLOW_CREDENTIALS + +ROOT_URLCONF = 'application.urls' + +TEMPLATES = [ + { + 'BACKEND': 'django.template.backends.django.DjangoTemplates', + 'DIRS': [], + 'APP_DIRS': True, + 'OPTIONS': { + 'context_processors': [ + 'django.template.context_processors.debug', + 'django.template.context_processors.request', + 'django.contrib.auth.context_processors.auth', + 'django.contrib.messages.context_processors.messages', + ], + }, + }, +] + +WSGI_APPLICATION = 'application.wsgi.application' + +# Database +# https://docs.djangoproject.com/en/1.11/ref/settings/#databases + + +# Password validation +# https://docs.djangoproject.com/en/1.11/ref/settings/#auth-password-validators + +AUTH_PASSWORD_VALIDATORS = [ + { + 'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator', + }, + { + 'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator', + }, + { + 'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator', + }, + { + 'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator', + }, +] + +# Internationalization +# https://docs.djangoproject.com/en/1.11/topics/i18n/ +# 配置语言 +LANGUAGE_CODE = 'zh-hans' +TIME_ZONE = 'Asia/Shanghai' + +USE_I18N = True + +USE_L10N = True + +USE_TZ = True + +# Static files (CSS, JavaScript, Images) +# https://docs.djangoproject.com/en/1.11/howto/static-files/ +""" +静态目录、多媒体配置 +""" +# 访问静态文件的url地址前缀 +STATIC_URL = '/static/' +# 设置django的静态文件目录 +STATICFILES_DIRS = [ + os.path.join(BASE_DIR, "static") +] +# 访问上传文件的url地址前缀 +MEDIA_URL = "/media/" +# 项目中存储上传文件的根目录 +MEDIA_ROOT = os.path.join(BASE_DIR, "media") + +""" +日志配置 +""" +# log 配置部分BEGIN # +SERVER_LOGS_FILE = os.path.join(BASE_DIR, 'logs', 'server.log') +ERROR_LOGS_FILE = os.path.join(BASE_DIR, 'logs', 'error.log') + +# 格式:[2020-04-22 23:33:01][micoservice.apps.ready():16] [INFO] 这是一条日志: +# 格式:[日期][模块.函数名称():行号] [级别] 信息 +STANDARD_LOG_FORMAT = '[%(asctime)s][%(name)s.%(funcName)s():%(lineno)d] [%(levelname)s] %(message)s' +CONSOLE_LOG_FORMAT = '[%(asctime)s][%(name)s.%(funcName)s():%(lineno)d] [%(levelname)s] %(message)s' + +LOGGING = { + 'version': 1, + 'disable_existing_loggers': False, + 'formatters': { + 'standard': { + 'format': STANDARD_LOG_FORMAT + }, + 'console': { + 'format': CONSOLE_LOG_FORMAT, + 'datefmt': '%Y-%m-%d %H:%M:%S', + }, + }, + 'handlers': { + 'file': { + 'level': 'INFO', + 'class': 'logging.handlers.RotatingFileHandler', + 'filename': SERVER_LOGS_FILE, + 'maxBytes': 1024 * 1024 * 100, # 100 MB + 'backupCount': 5, # 最多备份5个 + 'formatter': 'standard', + 'encoding': 'utf-8', + }, + 'error': { + 'level': 'ERROR', + 'class': 'logging.handlers.RotatingFileHandler', + 'filename': ERROR_LOGS_FILE, + 'maxBytes': 1024 * 1024 * 100, # 100 MB + 'backupCount': 3, # 最多备份3个 + 'formatter': 'standard', + 'encoding': 'utf-8', + }, + 'console': { + 'level': 'INFO', + 'class': 'logging.StreamHandler', + 'formatter': 'console', + } + }, + 'loggers': { + # default日志 + '': { + 'handlers': ['console'], + 'level': 'INFO', + }, + # 数据库相关日志 + 'django.db.backends': { + 'handlers': [], + 'propagate': True, + 'level': 'INFO', + }, + } +} + +# ================================================= # +# ************** 数据库 配置 ************** # +# ================================================= # + +if DATABASE_TYPE == "MYSQL": + # Mysql数据库 + DATABASES = { + "default": { + "ENGINE": "django.db.backends.mysql", + "HOST": DATABASE_HOST, + "PORT": DATABASE_PORT, + "USER": DATABASE_USER, + "PASSWORD": DATABASE_PASSWORD, + "NAME": DATABASE_NAME, + } + } +else: + # sqlite3 数据库 + DATABASES = { + 'default': { + 'ENGINE': 'django.db.backends.sqlite3', + 'NAME': os.path.join(BASE_DIR, 'db.sqlite3'), + } + } + +connect(MONGO_DATABASE_NAME, host=MONGO_HOST, port=MONGO_PORT, serverSelectionTimeoutMS=1000, connect=False) +# redis 缓存 +REDIS_URL = f'redis://:{REDIS_PASSWORD if REDIS_PASSWORD else ""}@{REDIS_HOST}:{REDIS_PORT}/{REDIS_DB}' +CACHES = { + "default": { + "BACKEND": "django_redis.cache.RedisCache", + "LOCATION": REDIS_URL, + "OPTIONS": { + "CLIENT_CLASS": "django_redis.client.DefaultClient", + } + }, +} +# ================================================= # +# ******************** JWT配置 ******************** # +# ================================================= # +JWT_AUTH = { + 'JWT_ALLOW_REFRESH': True, + 'JWT_EXPIRATION_DELTA': datetime.timedelta(seconds=60 * 60 * 24), # JWT有效时间24小时 + 'JWT_AUTH_HEADER_PREFIX': 'Bearer', # JWT的Header认证头以'JWT '开始 + 'JWT_AUTH_COOKIE': 'AUTH_JWT', + 'JWT_VERIFY_EXPIRATION': True, + 'JWT_PAYLOAD_HANDLER': 'utils.jwt_util.jwt_payload_handler', + 'JWT_GET_USER_SECRET_KEY': 'utils.jwt_util.jwt_get_user_secret_key', + 'JWT_RESPONSE_PAYLOAD_HANDLER': 'utils.jwt_util.jwt_response_payload_handler', +} + +# ================================================= # +# ************** REST_FRAMEWORK 配置 ************** # +# ================================================= # +REST_FRAMEWORK = { + 'DEFAULT_PERMISSION_CLASSES': ( + 'rest_framework.permissions.IsAuthenticated', + ), + + 'DEFAULT_AUTHENTICATION_CLASSES': ( + 'rest_framework_jwt.authentication.JSONWebTokenAuthentication', + 'rest_framework.authentication.BasicAuthentication', + 'rest_framework.authentication.SessionAuthentication', + 'utils.authentication.RedisOpAuthJwtAuthentication' + ), + + 'DEFAULT_SCHEMA_CLASS': 'rest_framework.schemas.AutoSchema', + 'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination', + 'DEFAULT_FILTER_BACKENDS': ('django_filters.rest_framework.DjangoFilterBackend',), + 'EXCEPTION_HANDLER': 'utils.exceptions.op_exception_handler', +} +# ================================================= # +# ************** 登录方式配置 ************** # +# ================================================= # +AUTHENTICATION_BACKENDS = ( + 'utils.backends.CustomBackend', + 'utils.backends.SessionAuthentication', +) +AUTH_USER_MODEL = 'system.UserProfile' + +# ================================================= # +# ************** 登录验证码配置 ************** # +# ================================================= # +CAPTCHA_STATE = CAPTCHA_STATE +#字母验证码 +CAPTCHA_IMAGE_SIZE = (160, 60) # 设置 captcha 图片大小 +CAPTCHA_LENGTH = 4 # 字符个数 +CAPTCHA_TIMEOUT = 1 # 超时(minutes) +#加减乘除验证码 +CAPTCHA_OUTPUT_FORMAT = '%(image)s %(text_field)s %(hidden_field)s ' +CAPTCHA_NOISE_FUNCTIONS = ('captcha.helpers.noise_null', + 'captcha.helpers.noise_arcs', # 线 + 'captcha.helpers.noise_dots', # 点 + ) +# CAPTCHA_CHALLENGE_FUNCT = 'captcha.helpers.random_char_challenge' +CAPTCHA_CHALLENGE_FUNCT = 'captcha.helpers.math_challenge' diff --git a/backend/application/urls.py b/backend/application/urls.py new file mode 100644 index 0000000..0cd0d90 --- /dev/null +++ b/backend/application/urls.py @@ -0,0 +1,59 @@ +"""application URL Configuration + +The `urlpatterns` list routes URLs to views. For more information please see: + https://docs.djangoproject.com/en/1.11/topics/http/urls/ +Examples: +Function views + 1. Add an import: from my_app import views + 2. Add a URL to urlpatterns: url(r'^$', views.home, name='home') +Class-based views + 1. Add an import: from other_app.views import Home + 2. Add a URL to urlpatterns: url(r'^$', Home.as_view(), name='home') +Including another URLconf + 1. Import the include() function: from django.conf.urls import url, include + 2. Add a URL to urlpatterns: url(r'^blog/', include('blog.urls')) +""" +import json + +from captcha.conf import settings as ca_settings +from captcha.helpers import captcha_image_url, captcha_audio_url +from captcha.models import CaptchaStore +from django.conf import settings +from django.conf.urls import url +from django.contrib import admin +from django.urls import re_path, include +from django.views.static import serve +from rest_framework.views import APIView + +from apps.system.views import GetUserView, GetRouters +from utils.login import LoginView, LogoutView +from utils.response import SuccessResponse + + +class CaptchaRefresh(APIView): + authentication_classes = [] + permission_classes = [] + + def get(self, request): + new_key = CaptchaStore.pick() + to_json_response = { + "key": new_key, + "image_url": captcha_image_url(new_key), + "audio_url": captcha_audio_url(new_key) if ca_settings.CAPTCHA_FLITE_PATH else None, + } + return SuccessResponse(to_json_response) + + +urlpatterns = [ + re_path('api-token-auth/', LoginView.as_view(), name='api_token_auth'), + re_path(r'^admin/', admin.site.urls), + re_path(r'media/(?P.*)', serve, {"document_root": settings.MEDIA_ROOT}), + re_path(r'^login/$', LoginView.as_view()), + re_path(r'^logout/$', LogoutView.as_view()), + re_path(r'^getInfo/$', GetUserView.as_view()), + re_path(r'^getRouters/$', GetRouters.as_view()), + url(r'^api-auth/', include('rest_framework.urls', namespace='rest_framework')), + url(r"captcha/refresh/$", CaptchaRefresh.as_view(), name="captcha-refresh"), # 刷新验证码 + re_path('captcha/', include('captcha.urls')), # 图片验证码 路由 + +] diff --git a/backend/application/wsgi.py b/backend/application/wsgi.py new file mode 100644 index 0000000..09ff1dd --- /dev/null +++ b/backend/application/wsgi.py @@ -0,0 +1,16 @@ +""" +WSGI config for application project. + +It exposes the WSGI callable as a module-level variable named ``application``. + +For more information on this file, see +https://docs.djangoproject.com/en/1.11/howto/deployment/wsgi/ +""" + +import os + +from django.core.wsgi import get_wsgi_application + +os.environ.setdefault("DJANGO_SETTINGS_MODULE", "application.settings") + +application = get_wsgi_application() diff --git a/backend/apps/system/__init__.py b/backend/apps/system/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/apps/system/admin.py b/backend/apps/system/admin.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/apps/system/apps.py b/backend/apps/system/apps.py new file mode 100644 index 0000000..5dc4d64 --- /dev/null +++ b/backend/apps/system/apps.py @@ -0,0 +1,5 @@ +from django.apps import AppConfig + + +class SystemConfig(AppConfig): + name = 'system' diff --git a/backend/apps/system/filters.py b/backend/apps/system/filters.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/apps/system/libs/__init__.py b/backend/apps/system/libs/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/apps/system/models/__init__.py b/backend/apps/system/models/__init__.py new file mode 100644 index 0000000..e1e8fa6 --- /dev/null +++ b/backend/apps/system/models/__init__.py @@ -0,0 +1 @@ +from ..models.users import UserProfile diff --git a/backend/apps/system/models/users.py b/backend/apps/system/models/users.py new file mode 100644 index 0000000..83d5cf9 --- /dev/null +++ b/backend/apps/system/models/users.py @@ -0,0 +1,38 @@ +from uuid import uuid4 + +from django.contrib.auth.models import UserManager, AbstractUser +from django.db import models + + +class UserProfile(AbstractUser): + GENDER_CHOICES = ( + ("女", "女"), + ("男", "男"), + ("未知", "未知"), + ) + objects = UserManager() + username = models.CharField(max_length=150, unique=True, db_index=True, + verbose_name='username', help_text='用户昵称') + secret = models.CharField(max_length=255, default=uuid4, verbose_name='加密秘钥') + email = models.CharField(max_length=255, null=True, blank=True, verbose_name="邮箱") + mobile = models.CharField(max_length=255, null=True, blank=True, verbose_name="电话") + avatar = models.TextField(null=True, blank=True, verbose_name="头像", help_text="头像") + + name = models.CharField(max_length=40, unique=False, null=True, + blank=True, verbose_name="姓名") + gender = models.CharField(max_length=2, default=GENDER_CHOICES[2][0], choices=GENDER_CHOICES, + verbose_name="性别", help_text="性别") + remark = models.TextField(null=True, blank=True, verbose_name="备注", help_text="备注") + create_datetime = models.DateTimeField(auto_now_add=True, null=True, blank=True, + verbose_name=u'创建时间', help_text=u"创建时间") + update_datetime = models.DateTimeField(auto_now=True, null=True, blank=True, + verbose_name=u'更新时间', help_text=u"更新时间") + + class Meta: + verbose_name = '用户' + verbose_name_plural = verbose_name + + def __str__(self): + if self.name: + return f"{self.username}({self.name})" + return f"{self.username}" diff --git a/backend/apps/system/serializers.py b/backend/apps/system/serializers.py new file mode 100644 index 0000000..61a4371 --- /dev/null +++ b/backend/apps/system/serializers.py @@ -0,0 +1,10 @@ +from rest_framework import serializers + +from apps.system.models import UserProfile + + +class GetUserInfoSerializer(serializers.ModelSerializer): + class Meta: + model = UserProfile + fields = ( + 'username',) diff --git a/backend/apps/system/tasks.py b/backend/apps/system/tasks.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/apps/system/tests.py b/backend/apps/system/tests.py new file mode 100644 index 0000000..7ce503c --- /dev/null +++ b/backend/apps/system/tests.py @@ -0,0 +1,3 @@ +from django.test import TestCase + +# Create your tests here. diff --git a/backend/apps/system/urls.py b/backend/apps/system/urls.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/apps/system/views.py b/backend/apps/system/views.py new file mode 100644 index 0000000..1e9e6ac --- /dev/null +++ b/backend/apps/system/views.py @@ -0,0 +1,30 @@ +import json + +from rest_framework.response import Response +from rest_framework.views import APIView + +from system.serializers import GetUserInfoSerializer +from utils.response import SuccessResponse + + +class GetUserView(APIView): + """ + 获取用户详细信息 + """ + + def get(self, request, format=None): + # data = GetUserInfoSerializer(request.user).data + data = '{"msg":"操作成功","code":200,"permissions":["*:*:*"],"roles":["admin"],"user":{"searchValue":null,"createBy":"admin","createTime":"2020-11-20 19:29:42","updateBy":null,"updateTime":null,"remark":"管理员","params":{},"userId":1,"deptId":103,"userName":"admin","nickName":"若依","email":"ry@163.com","phonenumber":"15888888888","sex":"1","avatar":"","salt":null,"status":"0","delFlag":"0","loginIp":"127.0.0.1","loginDate":"2020-11-20T19:29:42.000+0800","dept":{"searchValue":null,"createBy":null,"createTime":null,"updateBy":null,"updateTime":null,"remark":null,"params":{},"deptId":103,"parentId":101,"ancestors":null,"deptName":"研发部门","orderNum":"1","leader":"若依","phone":null,"email":null,"status":"0","delFlag":null,"parentName":null,"children":[]},"roles":[{"searchValue":null,"createBy":null,"createTime":null,"updateBy":null,"updateTime":null,"remark":null,"params":{},"roleId":1,"roleName":"超级管理员","roleKey":"admin","roleSort":"1","dataScope":"1","menuCheckStrictly":false,"deptCheckStrictly":false,"status":"0","delFlag":null,"flag":false,"menuIds":null,"deptIds":null,"admin":true}],"roleIds":null,"postIds":null,"admin":true}}' + data = json.loads(data) + return Response(data) + +class GetRouters(APIView): + """ + 获取用户详细信息 + """ + + def get(self, request, format=None): + # data = GetUserInfoSerializer(request.user).data + data = '{"msg":"操作成功","code":200,"data":[{"name":"System","path":"/system","hidden":false,"redirect":"noRedirect","component":"Layout","alwaysShow":true,"meta":{"title":"系统管理","icon":"system","noCache":false},"children":[{"name":"User","path":"user","hidden":false,"component":"system/user/index","meta":{"title":"用户管理","icon":"user","noCache":false}},{"name":"Role","path":"role","hidden":false,"component":"system/role/index","meta":{"title":"角色管理","icon":"peoples","noCache":false}},{"name":"Menu","path":"menu","hidden":false,"component":"system/menu/index","meta":{"title":"菜单管理","icon":"tree-table","noCache":false}},{"name":"Dept","path":"dept","hidden":false,"component":"system/dept/index","meta":{"title":"部门管理","icon":"tree","noCache":false}},{"name":"Post","path":"post","hidden":false,"component":"system/post/index","meta":{"title":"岗位管理","icon":"post","noCache":false}},{"name":"Dict","path":"dict","hidden":false,"component":"system/dict/index","meta":{"title":"字典管理","icon":"dict","noCache":false}},{"name":"Config","path":"config","hidden":false,"component":"system/config/index","meta":{"title":"参数设置","icon":"edit","noCache":false}},{"name":"Notice","path":"notice","hidden":false,"component":"system/notice/index","meta":{"title":"通知公告","icon":"message","noCache":false}},{"name":"Log","path":"log","hidden":false,"redirect":"noRedirect","component":"ParentView","alwaysShow":true,"meta":{"title":"日志管理","icon":"log","noCache":false},"children":[{"name":"Operlog","path":"operlog","hidden":false,"component":"monitor/operlog/index","meta":{"title":"操作日志","icon":"form","noCache":false}},{"name":"Logininfor","path":"logininfor","hidden":false,"component":"monitor/logininfor/index","meta":{"title":"登录日志","icon":"logininfor","noCache":false}}]}]},{"name":"Monitor","path":"/monitor","hidden":false,"redirect":"noRedirect","component":"Layout","alwaysShow":true,"meta":{"title":"系统监控","icon":"monitor","noCache":false},"children":[{"name":"Online","path":"online","hidden":false,"component":"monitor/online/index","meta":{"title":"在线用户","icon":"online","noCache":false}},{"name":"Job","path":"job","hidden":false,"component":"monitor/job/index","meta":{"title":"定时任务","icon":"job","noCache":false}},{"name":"Druid","path":"druid","hidden":false,"component":"monitor/druid/index","meta":{"title":"数据监控","icon":"druid","noCache":false}},{"name":"Server","path":"server","hidden":false,"component":"monitor/server/index","meta":{"title":"服务监控","icon":"server","noCache":false}},{"name":"Cache","path":"cache","hidden":false,"component":"monitor/cache/index","meta":{"title":"缓存监控","icon":"redis","noCache":false}}]},{"name":"Tool","path":"/tool","hidden":false,"redirect":"noRedirect","component":"Layout","alwaysShow":true,"meta":{"title":"系统工具","icon":"tool","noCache":false},"children":[{"name":"Build","path":"build","hidden":false,"component":"tool/build/index","meta":{"title":"表单构建","icon":"build","noCache":false}},{"name":"Gen","path":"gen","hidden":false,"component":"tool/gen/index","meta":{"title":"代码生成","icon":"code","noCache":false}},{"name":"Swagger","path":"swagger","hidden":false,"component":"tool/swagger/index","meta":{"title":"系统接口","icon":"swagger","noCache":false}}]},{"name":"Http://ruoyi.vip","path":"http://ruoyi.vip","hidden":false,"component":"Layout","meta":{"title":"若依官网","icon":"guide","noCache":false}}]}' + data = json.loads(data) + return Response(data) diff --git a/backend/conf/env.py b/backend/conf/env.py new file mode 100644 index 0000000..2e7376f --- /dev/null +++ b/backend/conf/env.py @@ -0,0 +1,51 @@ +""" +本地开发配置 +# 配置使用说明 +- 首先在backend下面新建config.py + eg: + environment_config = 'env' +""" +# ================================================= # +# ************** mysql数据库 配置 ************** # +# ================================================= # +# 数据库类型 MYSQL/SQLITE3 +DATABASE_TYPE = "SQLITE3" +# 数据库地址 +DATABASE_HOST = "127.0.0.1" +# 数据库端口 +DATABASE_PORT = 3306 +# 数据库用户名 +DATABASE_USER = "root" +# 数据库密码 +DATABASE_PASSWORD = "Aa428912." +# 数据库名 +DATABASE_NAME = "django-vue-admin" + +# ================================================= # +# ************** mongodb 数据库配置 ************** # +# ================================================= # +MONGO_DATABASE_NAME = 'django-vue-admin' +MONGO_HOST = 'localhost' +MONGO_PORT = 27017 + +# ================================================= # +# ************** redis 数据库配置 ************** # +# ================================================= # +REDIS_DB = 1 +REDIS_HOST = '127.0.0.1' +REDIS_PORT = 6379 +REDIS_PASSWORD = 'q1w2e3r4T%Y^U&I*2020' + +# ================================================= # +# ************** 默认配置 ************** # +# ================================================= # +# 只允许访问的ip地址列表 +ALLOWED_HOSTS = ['*'] +# 允许跨域源 +CORS_ORIGIN_ALLOW_ALL = True +# 允许ajax请求携带cookie +CORS_ALLOW_CREDENTIALS = False +# username_field +USERNAME_FIELD = 'username' +# 验证码状态 +CAPTCHA_STATE = True diff --git a/backend/conf/prod.py b/backend/conf/prod.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/conf/sit.py b/backend/conf/sit.py new file mode 100644 index 0000000..e69de29 diff --git a/backend/environment.py b/backend/environment.py new file mode 100644 index 0000000..f845224 --- /dev/null +++ b/backend/environment.py @@ -0,0 +1,10 @@ +from config import environment_config + + +if environment_config == 'sit': + from conf.sit import * +elif environment_config == 'prod': + from conf.prod import * +else : + from conf.env import * +print(f"导入[{environment_config}]配置成功!") diff --git a/backend/manage.py b/backend/manage.py new file mode 100755 index 0000000..b2f58a7 --- /dev/null +++ b/backend/manage.py @@ -0,0 +1,22 @@ +#!/usr/bin/env python +import os +import sys + +if __name__ == "__main__": + os.environ.setdefault("DJANGO_SETTINGS_MODULE", "application.settings") + try: + from django.core.management import execute_from_command_line + except ImportError: + # The above import may fail for some other reason. Ensure that the + # issue is really that Django is missing to avoid masking other + # exceptions on Python 2. + try: + import django + except ImportError: + raise ImportError( + "Couldn't import Django. Are you sure it's installed and " + "available on your PYTHONPATH environment variable? Did you " + "forget to activate a virtual environment?" + ) + raise + execute_from_command_line(sys.argv) diff --git a/backend/utils/authentication.py b/backend/utils/authentication.py new file mode 100644 index 0000000..92a0058 --- /dev/null +++ b/backend/utils/authentication.py @@ -0,0 +1,89 @@ +""" +常用的认证以及DRF的认证 +""" +import logging + +from django.conf import settings +from django.contrib.auth import get_user_model +from django.core.cache import cache +from django.utils.six import text_type +from rest_framework_jwt.utils import jwt_decode_handler + +from .decorators import exceptionHandler + +logger = logging.getLogger(__name__) +User = get_user_model() + + +class OpAuthJwtAuthentication(object): + """ + 统一JWT认证(环境允许情况下, 推荐使用RedisOpAuthJwtAuthentication) + """ + + @exceptionHandler() + def authenticate(self, request): + token = self.get_header_authorization(request) or self.get_cookie_authorization(request) + if not token: + return None + payload = jwt_decode_handler(token) + username = payload.get('username', None) + if not username: + return None + username_field = settings.USERNAME_FIELD or 'username' + user = User.objects.filter(**{username_field: username}).first() + if not user or not user.is_active: + return None + return user, token + + def authenticate_header(self, request): + pass + + @classmethod + def get_header_authorization(cls, request): + """ + 获取header里的认证信息, 通常用于跨域携带请求 + :param request: + :return: + """ + auth = request.META.get('HTTP_AUTHORIZATION', b'') + if isinstance(auth, text_type): + auth = auth.encode(settings.JWT_AUTH.get('HTTP_HEADER_ENCODING', 'iso-8859-1')) + if not auth: + return '' + auth = str(auth, encoding='utf-8').split() + if len(auth) != 2 or auth[0].upper() != settings.JWT_AUTH.get('JWT_AUTH_HEADER_PREFIX', 'JWT'): + return '' + return auth[1] + + @classmethod + def get_cookie_authorization(cls, request): + """ + 获取cookie里JWT认证信息 + :param request: + :return: + """ + auth = request.COOKIES.get(settings.JWT_AUTH.get('JWT_AUTH_COOKIE', 'AUTH_JWT'), '') + auth = auth.split() + if len(auth) != 2 or auth[0].upper() != settings.JWT_AUTH.get('JWT_AUTH_HEADER_PREFIX', 'JWT'): + return '' + return auth[1] + + +class RedisOpAuthJwtAuthentication(OpAuthJwtAuthentication): + """ + 基于Redis的统一JWT认证(推荐使用) + """ + prefix = settings.JWT_AUTH.get('JWT_AUTH_HEADER_PREFIX', 'JWT') + + @exceptionHandler() + def authenticate(self, request): + res = super().authenticate(request) + if res: + user, token = super().authenticate(request) + key = f"{self.prefix}_{user.username}" + redis_token = cache.get(key) + if redis_token == token: + return user, token + else: + return None + return None diff --git a/backend/utils/backends.py b/backend/utils/backends.py new file mode 100644 index 0000000..13b5dc2 --- /dev/null +++ b/backend/utils/backends.py @@ -0,0 +1,54 @@ +import logging + +from captcha.models import CaptchaStore +from django.conf import settings +from django.contrib.auth import get_user_model +from django.contrib.auth.backends import ModelBackend +from django.utils import timezone +from rest_framework.authentication import SessionAuthentication as DjangoSessionAuthentication + +from utils.exceptions import GenException + +logger = logging.getLogger(__name__) +UserModel = get_user_model() + + +class CustomBackend(ModelBackend): + """ + Django原生认证方式 + """ + + def authenticate(self, request, username=None, password=None, **kwargs): + msg = '%s 正在使用本地登录...' % username + logger.info(msg) + if username is None: + username = kwargs.get(UserModel.USERNAME_FIELD) + try: + user = UserModel._default_manager.get_by_natural_key(username) + except UserModel.DoesNotExist: + UserModel().set_password(password) + else: + if user.check_password(password) and self.user_can_authenticate(user): + user.last_login = timezone.now() + user.save() + return user + + +class SessionAuthentication(DjangoSessionAuthentication): + """ + Session认证 + """ + + def authenticate(self, request): + """ + Returns a `User` if the request session currently has a logged in user. + Otherwise returns `None`. + """ + # Get the session-based user from the underlying HttpRequest object + user = getattr(request._request, 'user', None) + # Unauthenticated, CSRF validation not required + if not user or not user.is_active: + return None + # self.enforce_csrf(request) + # CSRF passed with authenticated user + return user, None diff --git a/backend/utils/decorators.py b/backend/utils/decorators.py new file mode 100644 index 0000000..9b352cd --- /dev/null +++ b/backend/utils/decorators.py @@ -0,0 +1,277 @@ +""" +常用的装饰器以及DRF的装饰器 +""" +import time +import traceback +from datetime import datetime +import functools +from collections import Iterable +from django.conf import settings +from rest_framework_extensions.settings import extensions_api_settings +from django.utils import six +from django.utils.decorators import available_attrs +from rest_framework.response import Response +from .string_util import bas64_encode_text, bas64_decode_text + + +def get_cache(alias=None): + from django.core.cache import caches + return caches[alias or extensions_api_settings.DEFAULT_USE_CACHE] + + +def print_fun_time(logger=None): + """ + 打印函数执行时间 + :param logger: + :return: + """ + def wrapper(func): + @functools.wraps(func) + def inner(*args, **kwargs): + start_time = datetime.now() + res = func(*args, **kwargs) + end_time = datetime.now() + seconds = (end_time - start_time).seconds + if logger: + logger.info(f"{func.__name__}耗时:{seconds}秒") + else: + print(f"{func.__name__}耗时:{seconds}秒") + return res + return inner + return wrapper + +def print_time(logger=None): + """ + 打印函数执行时间 + :param logger: + :return: + """ + def wrapper(func): + @functools.wraps(func) + def inner(*args, **kwargs): + start_time = time.time() + res = func(*args, **kwargs) + end_time = time.time() + seconds = "%.3f"% (end_time - start_time) + if logger: + logger.info(f"{func.__name__}耗时:{seconds}秒") + else: + print(f"{func.__name__}耗时:{seconds}秒") + return res + return inner + return wrapper + + +def bas64_encode(func): + """ + 装饰器:Base64加密文本(func返回的文本) + :param func: + :return: + """ + def inner(*args, **kwargs): + text = func(*args, **kwargs) + if isinstance(text, str): + text = bas64_encode_text(text) + return text + return inner + + +def bas64_decode(func): + """ + 装饰器:Base64解密文本(func返回的文本) + :param func: + :return: + """ + def inner(*args, **kwargs): + text = func(*args, **kwargs) + if isinstance(text, str): + text = bas64_decode_text(text) + return text + return inner + + +def decode(crypto=""): + """ + 解密装饰器:BASE64 + :param crypto: 解密算法名称(忽略大小写) + """ + def wrapper(func): + def inner(*args, **kwargs): + text = func(*args, **kwargs) + if isinstance(text, str) or crypto: + if crypto.lower() == 'base64': + text = bas64_decode_text(text) + else: + text = text + return text + return inner + return wrapper + + +def encode(crypto=""): + """ + 解密装饰器:BASE64 + :param crypto: 解密算法名称(忽略大小写) + """ + def wrapper(func): + def inner(*args, **kwargs): + text = func(*args, **kwargs) + return text + return inner + return wrapper + + +def envFunction(envs='', execute=True, result=None): + """ + 环境函数装饰器:根据指导环境与当前环境是否匹配,判断是否执行某个函数 + :param envs:为True时,当前环境与指定的envs匹配时,执行该函数 + : 为False时,当前环境与指定的envs匹配时,不执行该函数 + :param result:指定当该函数被越过时的返回值,默认None + 实例:当环境为production时,才会执行robot_broadcast(),否则相当于在robot_broadcast里直接return + @envFunction(envs=['production', ], execute=True) + def robot_broadcast(content=''): + pass + + """ + def wrapper(func): + @functools.wraps(func) + def inner(*args, **kwargs): + environments = [] + if not envs: + environments = [] + elif isinstance(envs, str): + environments = [envs, ] + elif isinstance(envs, (Iterable, )): + environments = envs + if settings.PROJECT_ENV in environments and execute: + return func(*args, **kwargs) + elif settings.PROJECT_ENV not in environments and not execute: + return func(*args, **kwargs) + return result + return inner + return wrapper + + +def exceptionHandler(logger=None, throw=False, result=None, message=None): + """ + 异常装饰器:用于统一处理捕获的异常 + :param logger: 指定Logger + :param throw: 继续抛出这个异常 + :param result: 发生异常时的返回值(throw=True时,无效) + :param message: 错误信息 + :return: + """ + def wrapper(func): + @functools.wraps(func) + def inner(*args, **kwargs): + try: + return func(*args, **kwargs) + except Exception as e: + if logger: + logger.error(traceback.format_exc()) + if message: + logger.error(message) + if throw: + raise e + return result + return inner + return wrapper + + +class CacheResponse(object): + def __init__(self, + timeout=None, + key_func=None, + cache=None, + cache_errors=None): + if timeout is None: + self.timeout = extensions_api_settings.DEFAULT_CACHE_RESPONSE_TIMEOUT + else: + self.timeout = timeout + + if key_func is None: + self.key_func = 'get_cache_key' + else: + self.key_func = key_func + + if cache_errors is None: + self.cache_errors = extensions_api_settings.DEFAULT_CACHE_ERRORS + else: + self.cache_errors = cache_errors + + self.cache = get_cache(cache or extensions_api_settings.DEFAULT_USE_CACHE) + + def __call__(self, func): + this = self + @functools.wraps(func, assigned=available_attrs(func)) + def inner(self, request, *args, **kwargs): + return this.process_cache_response( + view_instance=self, + view_method=func, + request=request, + args=args, + kwargs=kwargs, + ) + return inner + + def process_cache_response(self, + view_instance, + view_method, + request, + args, + kwargs): + key = self.calculate_key( + view_instance=view_instance, + view_method=view_method, + request=request, + args=args, + kwargs=kwargs + ) + is_no_cache = False + is_no_cache_fun = getattr(view_instance, 'is_no_cache', None) + if is_no_cache_fun and is_no_cache_fun(request): + is_no_cache = True + response = None + if not is_no_cache: + response = self.cache.get(key) + if not response: + response = view_method(view_instance, request, *args, **kwargs) + response = view_instance.finalize_response(request, response, *args, **kwargs) + response.render() # should be rendered, before picklining while storing to cache + + if not response.status_code >= 400 or self.cache_errors: + if not is_no_cache: + if isinstance(response, Response): + self.cache.set(key, response.data, self.timeout) + else: + self.cache.set(key, response, self.timeout) + handle_refresh_cache_fun = getattr(view_instance, 'handle_refresh_cache', None) + if handle_refresh_cache_fun: + handle_refresh_cache_fun(request=request, key=key, cache=self.cache) + if not isinstance(response, Response): + response = Response(data=response) + if not hasattr(response, '_closable_objects'): + response._closable_objects = [] + return response + + def calculate_key(self, + view_instance, + view_method, + request, + args, + kwargs): + if isinstance(self.key_func, six.string_types): + key_func = getattr(view_instance, self.key_func) + else: + key_func = self.key_func + return key_func( + view_instance=view_instance, + view_method=view_method, + request=request, + args=args, + kwargs=kwargs, + ) + + +cache_response = CacheResponse + diff --git a/backend/utils/exceptions.py b/backend/utils/exceptions.py new file mode 100644 index 0000000..e6bd249 --- /dev/null +++ b/backend/utils/exceptions.py @@ -0,0 +1,71 @@ +import logging +import traceback + +from utils.response import ErrorResponse + +logger = logging.getLogger(__name__) + +from rest_framework.exceptions import APIException as DRFAPIException + + +class APIException(Exception): + """ + 通用异常:(1)用于接口请求是抛出移除, 此时code会被当做标准返回的code, message会被当做标准返回的msg + """ + + def __init__(self, code=201, message='API异常', args=('API异常',)): + self.args = args + self.code = code + self.message = message + + def __str__(self): + return self.message + + +class GenException(APIException): + pass + + +class FrameworkException(Exception): + """ + 框架异常、配置异常等 + """ + + def __init__(self, message='框架异常', *args: object, **kwargs: object) -> None: + super().__init__(*args, **kwargs) + self.message = message + + def __str__(self) -> str: + return f"{self.message}" + + +class JWTAuthenticationFailedException(APIException): + """ + JWT认证异常 + """ + + def __init__(self, code=201, message=None, args=('异常',)): + if not message: + message = 'JWT authentication failed!' + super().__init__(code, message, args) + + +def op_exception_handler(ex, context): + """ + 统一异常拦截处理 + 目的:(1)取消所有的500异常响应,统一响应为标准错误返回 + (2)准确显示错误信息 + :param ex: + :param context: + :return: + """ + msg = '' + if isinstance(ex, DRFAPIException): + # set_rollback() + msg = ex.detail + elif isinstance(ex, APIException): + msg = ex.message + elif isinstance(ex, Exception): + logger.error(traceback.format_exc()) + msg = str(ex) + return ErrorResponse(msg=msg) diff --git a/backend/utils/jwt_util.py b/backend/utils/jwt_util.py new file mode 100644 index 0000000..1c7a041 --- /dev/null +++ b/backend/utils/jwt_util.py @@ -0,0 +1,51 @@ +from calendar import timegm +from datetime import datetime + +from django.conf import settings +from django.contrib.auth import login + + +def jwt_response_payload_handler(token, user, request): + """ + 重写JWT的返回值 + :param token: + :param user: + :param request: + :return: + """ + login(request, user) + return { + 'token': f"{token}", + } + + +def jwt_get_secret_key(payload=None): + if isinstance(payload, dict): + return payload.get("secret", "") + return getattr(payload, "secret", "") + + +def jwt_get_user_secret_key(user): + """ + 重写JWT的secret的生成 + :param user: + :return: + """ + return str(user.secret) + +def jwt_payload_handler(user): + payload = { + 'user_id': user.pk, + 'username': user.username, + 'secret': jwt_get_user_secret_key(user), + 'exp': datetime.utcnow() + settings.JWT_AUTH.get('JWT_EXPIRATION_DELTA') + } + if settings.JWT_AUTH.get('JWT_ALLOW_REFRESH'): + payload['orig_iat'] = timegm( + datetime.utcnow().utctimetuple() + ) + if settings.JWT_AUTH.get('JWT_AUDIENCE',None) is not None: + payload['aud'] = settings.JWT_AUTH.get('JWT_AUDIENCE',None) + if settings.JWT_AUTH.get('JWT_ISSUER',None) is not None: + payload['iss'] = settings.JWT_AUTH.get('JWT_ISSUER',None) + return payload diff --git a/backend/utils/login.py b/backend/utils/login.py new file mode 100644 index 0000000..a2d8735 --- /dev/null +++ b/backend/utils/login.py @@ -0,0 +1,88 @@ +import datetime +import logging +from uuid import uuid4 + +from captcha.models import CaptchaStore +from django.conf import settings +from django.contrib.auth import get_user_model +from django.core.cache import cache +from rest_framework.permissions import IsAuthenticated +from rest_framework.views import APIView +from rest_framework_jwt.views import ObtainJSONWebToken, jwt_response_payload_handler + +from .exceptions import GenException +from .response import SuccessResponse, ErrorResponse + +# from .jwt_util import jwt_response_payload_handler + +logger = logging.getLogger(__name__) + +User = get_user_model() + + +class LogoutView(APIView): + queryset = User.objects.all() + permission_classes = (IsAuthenticated,) + prefix = settings.JWT_AUTH.get('JWT_AUTH_HEADER_PREFIX', 'JWT') + + def post(self, request): + user = request.user + user.user_secret = uuid4() + user.save() + key = f"{self.prefix}_{user.username}" + cache.delete(key) + return SuccessResponse() + + +class LoginView(ObtainJSONWebToken): + JWT_AUTH_COOKIE = '' + prefix = settings.JWT_AUTH.get('JWT_AUTH_HEADER_PREFIX') + ex = settings.JWT_AUTH.get('JWT_EXPIRATION_DELTA') + + def jarge_captcha(self, request): + """ + 校验验证码 + :param request: + :return: + """ + if not settings.CAPTCHA_STATE: # 未开启验证码则返回 True + return True + idKeyC = request.data.get('idKeyC', None) + idValueC = request.data.get('idValueC', None) + if not idValueC: + raise GenException(message='请输入验证码') + try: + get_captcha = CaptchaStore.objects.get(hashkey=idKeyC) + if str(get_captcha.response).lower() == idValueC.lower(): # 如果验证码匹配 + return True + except: + pass + else: + raise GenException(message='验证码错误') + + def post(self, request, *args, **kwargs): + # 校验验证码 + self.jarge_captcha(request) + serializer = self.get_serializer(data=request.data) + if serializer.is_valid(): + user = serializer.object.get('user') or request.user + token = serializer.object.get('token') + response_data = jwt_response_payload_handler(token, user, request) + response = SuccessResponse(response_data) + if token: + username = user.username + key = f"{self.prefix}_{username}" + cache.set(key, token, self.ex.total_seconds()) + if self.JWT_AUTH_COOKIE and token: + expiration = (datetime.datetime.utcnow() + self.ex) + response.set_cookie(self.JWT_AUTH_COOKIE, + token, + expires=expiration, + domain=settings.SESSION_COOKIE_DOMAIN, + httponly=False) + return response + return ErrorResponse(data=serializer.errors, msg='账户/密码不正确') + + def handle_exception(self, exc): + print(exc) + return ErrorResponse(data=None, msg=exc.message) diff --git a/backend/utils/response.py b/backend/utils/response.py new file mode 100644 index 0000000..fc280af --- /dev/null +++ b/backend/utils/response.py @@ -0,0 +1,52 @@ +""" +常用的Response以及Django的Response、DRF的Response +""" +from django.http.response import DjangoJSONEncoder +from rest_framework.response import Response + + +class OpDRFJSONEncoder(DjangoJSONEncoder): + """ + 重写DjangoJSONEncoder + (1)默认返回支持中文格式的json字符串 + """ + + def __init__(self, *, skipkeys=False, ensure_ascii=True, check_circular=True, allow_nan=True, sort_keys=False, + indent=None, separators=None, default=None): + super().__init__(skipkeys=skipkeys, ensure_ascii=False, check_circular=check_circular, + allow_nan=allow_nan, sort_keys=sort_keys, indent=indent, separators=separators, + default=default) + + +class SuccessResponse(Response): + """ + 标准响应成功的返回, SuccessResponse(data)或者SuccessResponse(data=data) + (1)默认错误码返回2000, 不支持指定其他返回码 + """ + + def __init__(self, data=None, msg='success', status=None, template_name=None, headers=None, exception=False, + content_type=None): + std_data = { + "code": 200, + "data": data, + "msg": msg, + "status": 'success' + } + super().__init__(std_data, status, template_name, headers, exception, content_type) + + +class ErrorResponse(Response): + """ + 标准响应错误的返回,ErrorResponse(msg='xxx') + (1)默认错误码返回2001, 也可以指定其他返回码:ErrorResponse(code=xxx) + """ + + def __init__(self, data=None, msg='error', code=201, status=None, template_name=None, headers=None, + exception=False, content_type=None): + std_data = { + "code": code, + "data": data, + "msg": msg, + "status": 'error' + } + super().__init__(std_data, status, template_name, headers, exception, content_type) diff --git a/backend/utils/string_util.py b/backend/utils/string_util.py new file mode 100644 index 0000000..ff4d9da --- /dev/null +++ b/backend/utils/string_util.py @@ -0,0 +1,111 @@ +""" +封装字符串相关函数:UUID字符串,字符串加密解密 +@author: wanglei +""" +import uuid as UUID +import base64 + +CHAR_SET = ("a", "b", "c", "d", "e", "f", + "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", + "t", "u", "v", "w", "x", "y", "z", "0", "1", "2", "3", "4", "5", + "6", "7", "8", "9", "A", "B", "C", "D", "E", "F", "G", "H", "I", + "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", + "W", "X", "Y", "Z") + + +def uuid(): + """ + 返回一个UUID对象 + :return: + """ + return UUID.uuid4() + + +def uuid_36(): + """ + 返回36字符的UUID字符串(十六进制,含有-) bc5debab-95c3-4430-933f-2e3b6407ac30 + :return: + """ + return str(UUID.uuid4()) + + +def uuid_32(): + """ + 返回32字符的UUID字符串(十六进制) bc5debab95c34430933f2e3b6407ac30 + :return: + """ + return uuid_36().replace('-', '') + + +def uuid_8(): + """ + 返回8字符的UUID字符串(非进制) 3FNWjtlD + :return: + """ + s = uuid_32() + result = '' + for i in range(0, 8): + sub = s[i * 4: i * 4 + 4] + x = int(sub, 16) + result += CHAR_SET[x % 0x3E] + return result + + +def uuid_16(): + """ + 返回16字符的UUID字符串(非进制) 3FNWjtlD3FNWjtlD + :return: + """ + return uuid_8() + uuid_8() + + +def bas64_encode_text(text): + """ + base64加密字符串 + :param text: + :return: + """ + if isinstance(text, str): + return str(base64.b64encode(text.encode('utf-8')), 'utf-8') + return text + + +def bas64_decode_text(text): + """ + base64解密字符串 + :param text: + :return: + """ + if isinstance(text, str): + return str(base64.decodebytes(bytes(text, encoding="utf8")), 'utf-8') + return text + + +def decode_text(text, crypto=""): + """ + 解密字符串 + :param text: 字符串 + :param crypto: 解密算法 + :return: + """ + if crypto: + if crypto.lower() == 'base64': + text = bas64_decode_text(text) + else: + text = text + return text + + +def encode_text(text, crypto=""): + """ + 加密字符串 + :param text: 字符串 + :param crypto: 加密算法 + :return: + """ + if crypto: + if crypto.lower() == 'base64': + text = bas64_encode_text(text) + else: + text = text + return text \ No newline at end of file diff --git a/dvadmin-ui/.editorconfig b/dvadmin-ui/.editorconfig new file mode 100755 index 0000000..7034f9b --- /dev/null +++ b/dvadmin-ui/.editorconfig @@ -0,0 +1,22 @@ +# 告诉EditorConfig插件,这是根文件,不用继续往上查找 +root = true + +# 匹配全部文件 +[*] +# 设置字符集 +charset = utf-8 +# 缩进风格,可选space、tab +indent_style = space +# 缩进的空格数 +indent_size = 2 +# 结尾换行符,可选lf、cr、crlf +end_of_line = lf +# 在文件结尾插入新行 +insert_final_newline = true +# 删除一行中的前后空格 +trim_trailing_whitespace = true + +# 匹配md结尾的文件 +[*.md] +insert_final_newline = false +trim_trailing_whitespace = false diff --git a/dvadmin-ui/.env.development b/dvadmin-ui/.env.development new file mode 100755 index 0000000..d4ac021 --- /dev/null +++ b/dvadmin-ui/.env.development @@ -0,0 +1,8 @@ +# 开发环境配置 +ENV = 'development' + +# 若依管理系统/开发环境 +VUE_APP_BASE_API = 'http://127.0.0.1:8000' + +# 路由懒加载 +VUE_CLI_BABEL_TRANSPILE_MODULES = true diff --git a/dvadmin-ui/.env.production b/dvadmin-ui/.env.production new file mode 100755 index 0000000..27c717e --- /dev/null +++ b/dvadmin-ui/.env.production @@ -0,0 +1,5 @@ +# 生产环境配置 +ENV = 'production' + +# 若依管理系统/生产环境 +VUE_APP_BASE_API = '/prod-api' diff --git a/dvadmin-ui/.env.staging b/dvadmin-ui/.env.staging new file mode 100755 index 0000000..6195736 --- /dev/null +++ b/dvadmin-ui/.env.staging @@ -0,0 +1,7 @@ +NODE_ENV = production + +# 测试环境配置 +ENV = 'staging' + +# 若依管理系统/测试环境 +VUE_APP_BASE_API = '/stage-api' diff --git a/dvadmin-ui/.eslintignore b/dvadmin-ui/.eslintignore new file mode 100755 index 0000000..89be6f6 --- /dev/null +++ b/dvadmin-ui/.eslintignore @@ -0,0 +1,10 @@ +# 忽略build目录下类型为js的文件的语法检查 +build/*.js +# 忽略src/assets目录下文件的语法检查 +src/assets +# 忽略public目录下文件的语法检查 +public +# 忽略当前目录下为js的文件的语法检查 +*.js +# 忽略当前目录下为vue的文件的语法检查 +*.vue \ No newline at end of file diff --git a/dvadmin-ui/.eslintrc.js b/dvadmin-ui/.eslintrc.js new file mode 100755 index 0000000..82bbdee --- /dev/null +++ b/dvadmin-ui/.eslintrc.js @@ -0,0 +1,199 @@ +// ESlint 检查配置 +module.exports = { + root: true, + parserOptions: { + parser: 'babel-eslint', + sourceType: 'module' + }, + env: { + browser: true, + node: true, + es6: true, + }, + extends: ['plugin:vue/recommended', 'eslint:recommended'], + + // add your custom rules here + //it is base on https://github.com/vuejs/eslint-config-vue + rules: { + "vue/max-attributes-per-line": [2, { + "singleline": 10, + "multiline": { + "max": 1, + "allowFirstLine": false + } + }], + "vue/singleline-html-element-content-newline": "off", + "vue/multiline-html-element-content-newline":"off", + "vue/name-property-casing": ["error", "PascalCase"], + "vue/no-v-html": "off", + 'accessor-pairs': 2, + 'arrow-spacing': [2, { + 'before': true, + 'after': true + }], + 'block-spacing': [2, 'always'], + 'brace-style': [2, '1tbs', { + 'allowSingleLine': true + }], + 'camelcase': [0, { + 'properties': 'always' + }], + 'comma-dangle': [2, 'never'], + 'comma-spacing': [2, { + 'before': false, + 'after': true + }], + 'comma-style': [2, 'last'], + 'constructor-super': 2, + 'curly': [2, 'multi-line'], + 'dot-location': [2, 'property'], + 'eol-last': 2, + 'eqeqeq': ["error", "always", {"null": "ignore"}], + 'generator-star-spacing': [2, { + 'before': true, + 'after': true + }], + 'handle-callback-err': [2, '^(err|error)$'], + 'indent': [2, 2, { + 'SwitchCase': 1 + }], + 'jsx-quotes': [2, 'prefer-single'], + 'key-spacing': [2, { + 'beforeColon': false, + 'afterColon': true + }], + 'keyword-spacing': [2, { + 'before': true, + 'after': true + }], + 'new-cap': [2, { + 'newIsCap': true, + 'capIsNew': false + }], + 'new-parens': 2, + 'no-array-constructor': 2, + 'no-caller': 2, + 'no-console': 'off', + 'no-class-assign': 2, + 'no-cond-assign': 2, + 'no-const-assign': 2, + 'no-control-regex': 0, + 'no-delete-var': 2, + 'no-dupe-args': 2, + 'no-dupe-class-members': 2, + 'no-dupe-keys': 2, + 'no-duplicate-case': 2, + 'no-empty-character-class': 2, + 'no-empty-pattern': 2, + 'no-eval': 2, + 'no-ex-assign': 2, + 'no-extend-native': 2, + 'no-extra-bind': 2, + 'no-extra-boolean-cast': 2, + 'no-extra-parens': [2, 'functions'], + 'no-fallthrough': 2, + 'no-floating-decimal': 2, + 'no-func-assign': 2, + 'no-implied-eval': 2, + 'no-inner-declarations': [2, 'functions'], + 'no-invalid-regexp': 2, + 'no-irregular-whitespace': 2, + 'no-iterator': 2, + 'no-label-var': 2, + 'no-labels': [2, { + 'allowLoop': false, + 'allowSwitch': false + }], + 'no-lone-blocks': 2, + 'no-mixed-spaces-and-tabs': 2, + 'no-multi-spaces': 2, + 'no-multi-str': 2, + 'no-multiple-empty-lines': [2, { + 'max': 1 + }], + 'no-native-reassign': 2, + 'no-negated-in-lhs': 2, + 'no-new-object': 2, + 'no-new-require': 2, + 'no-new-symbol': 2, + 'no-new-wrappers': 2, + 'no-obj-calls': 2, + 'no-octal': 2, + 'no-octal-escape': 2, + 'no-path-concat': 2, + 'no-proto': 2, + 'no-redeclare': 2, + 'no-regex-spaces': 2, + 'no-return-assign': [2, 'except-parens'], + 'no-self-assign': 2, + 'no-self-compare': 2, + 'no-sequences': 2, + 'no-shadow-restricted-names': 2, + 'no-spaced-func': 2, + 'no-sparse-arrays': 2, + 'no-this-before-super': 2, + 'no-throw-literal': 2, + 'no-trailing-spaces': 2, + 'no-undef': 2, + 'no-undef-init': 2, + 'no-unexpected-multiline': 2, + 'no-unmodified-loop-condition': 2, + 'no-unneeded-ternary': [2, { + 'defaultAssignment': false + }], + 'no-unreachable': 2, + 'no-unsafe-finally': 2, + 'no-unused-vars': [2, { + 'vars': 'all', + 'args': 'none' + }], + 'no-useless-call': 2, + 'no-useless-computed-key': 2, + 'no-useless-constructor': 2, + 'no-useless-escape': 0, + 'no-whitespace-before-property': 2, + 'no-with': 2, + 'one-var': [2, { + 'initialized': 'never' + }], + 'operator-linebreak': [2, 'after', { + 'overrides': { + '?': 'before', + ':': 'before' + } + }], + 'padded-blocks': [2, 'never'], + 'quotes': [2, 'single', { + 'avoidEscape': true, + 'allowTemplateLiterals': true + }], + 'semi': [2, 'never'], + 'semi-spacing': [2, { + 'before': false, + 'after': true + }], + 'space-before-blocks': [2, 'always'], + 'space-before-function-paren': [2, 'never'], + 'space-in-parens': [2, 'never'], + 'space-infix-ops': 2, + 'space-unary-ops': [2, { + 'words': true, + 'nonwords': false + }], + 'spaced-comment': [2, 'always', { + 'markers': ['global', 'globals', 'eslint', 'eslint-disable', '*package', '!', ','] + }], + 'template-curly-spacing': [2, 'never'], + 'use-isnan': 2, + 'valid-typeof': 2, + 'wrap-iife': [2, 'any'], + 'yield-star-spacing': [2, 'both'], + 'yoda': [2, 'never'], + 'prefer-const': 2, + 'no-debugger': process.env.NODE_ENV === 'production' ? 2 : 0, + 'object-curly-spacing': [2, 'always', { + objectsInObjects: false + }], + 'array-bracket-spacing': [2, 'never'] + } +} diff --git a/dvadmin-ui/.gitignore b/dvadmin-ui/.gitignore new file mode 100755 index 0000000..78a752d --- /dev/null +++ b/dvadmin-ui/.gitignore @@ -0,0 +1,23 @@ +.DS_Store +node_modules/ +dist/ +npm-debug.log* +yarn-debug.log* +yarn-error.log* +**/*.log + +tests/**/coverage/ +tests/e2e/reports +selenium-debug.log + +# Editor directories and files +.idea +.vscode +*.suo +*.ntvs* +*.njsproj +*.sln +*.local + +package-lock.json +yarn.lock diff --git a/dvadmin-ui/README.md b/dvadmin-ui/README.md new file mode 100755 index 0000000..a857617 --- /dev/null +++ b/dvadmin-ui/README.md @@ -0,0 +1,30 @@ +## 开发 + +```bash +# 克隆项目 +git clone https://gitee.com/y_project/RuoYi-Vue + +# 进入项目目录 +cd ruoyi-ui + +# 安装依赖 +npm install + +# 建议不要直接使用 cnpm 安装依赖,会有各种诡异的 bug。可以通过如下操作解决 npm 下载速度慢的问题 +npm install --registry=https://registry.npm.taobao.org + +# 启动服务 +npm run dev +``` + +浏览器访问 http://localhost:80 + +## 发布 + +```bash +# 构建测试环境 +npm run build:stage + +# 构建生产环境 +npm run build:prod +``` \ No newline at end of file diff --git a/dvadmin-ui/babel.config.js b/dvadmin-ui/babel.config.js new file mode 100755 index 0000000..b99f001 --- /dev/null +++ b/dvadmin-ui/babel.config.js @@ -0,0 +1,13 @@ +module.exports = { + presets: [ + // https://github.com/vuejs/vue-cli/tree/master/packages/@vue/babel-preset-app + '@vue/cli-plugin-babel/preset' + ], + 'env': { + 'development': { + // babel-plugin-dynamic-import-node plugin only does one thing by converting all import() to require(). + // This plugin can significantly increase the speed of hot updates, when you have a large number of pages. + 'plugins': ['dynamic-import-node'] + } + } +} diff --git a/dvadmin-ui/build/index.js b/dvadmin-ui/build/index.js new file mode 100755 index 0000000..0c57de2 --- /dev/null +++ b/dvadmin-ui/build/index.js @@ -0,0 +1,35 @@ +const { run } = require('runjs') +const chalk = require('chalk') +const config = require('../vue.config.js') +const rawArgv = process.argv.slice(2) +const args = rawArgv.join(' ') + +if (process.env.npm_config_preview || rawArgv.includes('--preview')) { + const report = rawArgv.includes('--report') + + run(`vue-cli-service build ${args}`) + + const port = 9526 + const publicPath = config.publicPath + + var connect = require('connect') + var serveStatic = require('serve-static') + const app = connect() + + app.use( + publicPath, + serveStatic('./dist', { + index: ['index.html', '/'] + }) + ) + + app.listen(port, function () { + console.log(chalk.green(`> Preview at http://localhost:${port}${publicPath}`)) + if (report) { + console.log(chalk.green(`> Report at http://localhost:${port}${publicPath}report.html`)) + } + + }) +} else { + run(`vue-cli-service build ${args}`) +} diff --git a/dvadmin-ui/package.json b/dvadmin-ui/package.json new file mode 100755 index 0000000..567842d --- /dev/null +++ b/dvadmin-ui/package.json @@ -0,0 +1,87 @@ +{ + "name": "ruoyi", + "version": "3.3.0", + "description": "若依管理系统", + "author": "若依", + "license": "MIT", + "scripts": { + "dev": "vue-cli-service serve", + "build:prod": "vue-cli-service build", + "build:stage": "vue-cli-service build --mode staging", + "preview": "node build/index.js --preview", + "lint": "eslint --ext .js,.vue src" + }, + "husky": { + "hooks": { + "pre-commit": "lint-staged" + } + }, + "lint-staged": { + "src/**/*.{js,vue}": [ + "eslint --fix", + "git add" + ] + }, + "keywords": [ + "vue", + "admin", + "dashboard", + "element-ui", + "boilerplate", + "admin-template", + "management-system" + ], + "repository": { + "type": "git", + "url": "https://gitee.com/y_project/RuoYi-Vue.git" + }, + "dependencies": { + "@riophae/vue-treeselect": "0.4.0", + "axios": "0.21.0", + "clipboard": "2.0.6", + "core-js": "3.8.1", + "echarts": "4.9.0", + "element-ui": "2.15.0", + "file-saver": "2.0.4", + "fuse.js": "6.4.3", + "highlight.js": "9.18.5", + "js-beautify": "1.13.0", + "js-cookie": "2.2.1", + "jsencrypt": "3.0.0-rc.1", + "nprogress": "0.2.0", + "quill": "1.3.7", + "screenfull": "5.0.2", + "sortablejs": "1.10.2", + "vue": "2.6.12", + "vue-count-to": "1.0.13", + "vue-cropper": "0.5.5", + "vue-router": "3.4.9", + "vuedraggable": "2.24.3", + "vuex": "3.6.0" + }, + "devDependencies": { + "@vue/cli-plugin-babel": "4.4.6", + "@vue/cli-plugin-eslint": "4.4.6", + "@vue/cli-service": "4.4.6", + "babel-eslint": "10.1.0", + "chalk": "4.1.0", + "connect": "3.6.6", + "eslint": "7.15.0", + "eslint-plugin-vue": "7.2.0", + "lint-staged": "10.5.3", + "runjs": "4.4.2", + "sass": "1.32.0", + "sass-loader": "10.1.0", + "script-ext-html-webpack-plugin": "2.1.5", + "svg-sprite-loader": "5.1.1", + "vue-template-compiler": "2.6.12" + }, + "engines": { + "node": ">=8.9", + "npm": ">= 3.0.0" + }, + "browserslist": [ + "> 1%", + "last 2 versions" + ] +} diff --git a/dvadmin-ui/public/favicon.ico b/dvadmin-ui/public/favicon.ico new file mode 100755 index 0000000..e263760 Binary files /dev/null and b/dvadmin-ui/public/favicon.ico differ diff --git a/dvadmin-ui/public/index.html b/dvadmin-ui/public/index.html new file mode 100755 index 0000000..6d64bf9 --- /dev/null +++ b/dvadmin-ui/public/index.html @@ -0,0 +1,207 @@ + + + + + + + + + <%= webpackConfig.name %> + + + +
+
+
+
+
+
正在加载系统资源,请耐心等待
+
+
+ + diff --git a/dvadmin-ui/public/robots.txt b/dvadmin-ui/public/robots.txt new file mode 100755 index 0000000..77470cb --- /dev/null +++ b/dvadmin-ui/public/robots.txt @@ -0,0 +1,2 @@ +User-agent: * +Disallow: / \ No newline at end of file diff --git a/dvadmin-ui/src/App.vue b/dvadmin-ui/src/App.vue new file mode 100755 index 0000000..e448b11 --- /dev/null +++ b/dvadmin-ui/src/App.vue @@ -0,0 +1,11 @@ + + + diff --git a/dvadmin-ui/src/api/login.js b/dvadmin-ui/src/api/login.js new file mode 100755 index 0000000..d2f6dad --- /dev/null +++ b/dvadmin-ui/src/api/login.js @@ -0,0 +1,40 @@ +import request from '@/utils/request' + +// 登录方法 +export function login(username, password, code, uuid) { + const data = { + username, + password, + idValueC: code, + idKeyC: uuid + } + return request({ + url: '/login/', + method: 'post', + data: data + }) +} + +// 获取用户详细信息 +export function getInfo() { + return request({ + url: '/getInfo/', + method: 'get' + }) +} + +// 退出方法 +export function logout() { + return request({ + url: '/logout/', + method: 'post' + }) +} + +// 获取验证码 +export function getCodeImg() { + return request({ + url: '/captcha/refresh/', + method: 'get' + }) +} diff --git a/dvadmin-ui/src/api/menu.js b/dvadmin-ui/src/api/menu.js new file mode 100755 index 0000000..faef101 --- /dev/null +++ b/dvadmin-ui/src/api/menu.js @@ -0,0 +1,9 @@ +import request from '@/utils/request' + +// 获取路由 +export const getRouters = () => { + return request({ + url: '/getRouters', + method: 'get' + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/monitor/cache.js b/dvadmin-ui/src/api/monitor/cache.js new file mode 100755 index 0000000..59d3505 --- /dev/null +++ b/dvadmin-ui/src/api/monitor/cache.js @@ -0,0 +1,9 @@ +import request from '@/utils/request' + +// 查询缓存详细 +export function getCache() { + return request({ + url: '/monitor/cache', + method: 'get' + }) +} diff --git a/dvadmin-ui/src/api/monitor/job.js b/dvadmin-ui/src/api/monitor/job.js new file mode 100755 index 0000000..58c4343 --- /dev/null +++ b/dvadmin-ui/src/api/monitor/job.js @@ -0,0 +1,80 @@ +import request from '@/utils/request' + +// 查询定时任务调度列表 +export function listJob(query) { + return request({ + url: '/monitor/job/list', + method: 'get', + params: query + }) +} + +// 查询定时任务调度详细 +export function getJob(jobId) { + return request({ + url: '/monitor/job/' + jobId, + method: 'get' + }) +} + +// 新增定时任务调度 +export function addJob(data) { + return request({ + url: '/monitor/job', + method: 'post', + data: data + }) +} + +// 修改定时任务调度 +export function updateJob(data) { + return request({ + url: '/monitor/job', + method: 'put', + data: data + }) +} + +// 删除定时任务调度 +export function delJob(jobId) { + return request({ + url: '/monitor/job/' + jobId, + method: 'delete' + }) +} + +// 导出定时任务调度 +export function exportJob(query) { + return request({ + url: '/monitor/job/export', + method: 'get', + params: query + }) +} + +// 任务状态修改 +export function changeJobStatus(jobId, status) { + const data = { + jobId, + status + } + return request({ + url: '/monitor/job/changeStatus', + method: 'put', + data: data + }) +} + + +// 定时任务立即执行一次 +export function runJob(jobId, jobGroup) { + const data = { + jobId, + jobGroup + } + return request({ + url: '/monitor/job/run', + method: 'put', + data: data + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/monitor/jobLog.js b/dvadmin-ui/src/api/monitor/jobLog.js new file mode 100755 index 0000000..be1fffd --- /dev/null +++ b/dvadmin-ui/src/api/monitor/jobLog.js @@ -0,0 +1,35 @@ +import request from '@/utils/request' + +// 查询调度日志列表 +export function listJobLog(query) { + return request({ + url: '/monitor/jobLog/list', + method: 'get', + params: query + }) +} + +// 删除调度日志 +export function delJobLog(jobLogId) { + return request({ + url: '/monitor/jobLog/' + jobLogId, + method: 'delete' + }) +} + +// 清空调度日志 +export function cleanJobLog() { + return request({ + url: '/monitor/jobLog/clean', + method: 'delete' + }) +} + +// 导出调度日志 +export function exportJobLog(query) { + return request({ + url: '/monitor/jobLog/export', + method: 'get', + params: query + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/monitor/logininfor.js b/dvadmin-ui/src/api/monitor/logininfor.js new file mode 100755 index 0000000..383d61f --- /dev/null +++ b/dvadmin-ui/src/api/monitor/logininfor.js @@ -0,0 +1,35 @@ +import request from '@/utils/request' + +// 查询登录日志列表 +export function list(query) { + return request({ + url: '/monitor/logininfor/list', + method: 'get', + params: query + }) +} + +// 删除登录日志 +export function delLogininfor(infoId) { + return request({ + url: '/monitor/logininfor/' + infoId, + method: 'delete' + }) +} + +// 清空登录日志 +export function cleanLogininfor() { + return request({ + url: '/monitor/logininfor/clean', + method: 'delete' + }) +} + +// 导出登录日志 +export function exportLogininfor(query) { + return request({ + url: '/monitor/logininfor/export', + method: 'get', + params: query + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/monitor/online.js b/dvadmin-ui/src/api/monitor/online.js new file mode 100755 index 0000000..bd22137 --- /dev/null +++ b/dvadmin-ui/src/api/monitor/online.js @@ -0,0 +1,18 @@ +import request from '@/utils/request' + +// 查询在线用户列表 +export function list(query) { + return request({ + url: '/monitor/online/list', + method: 'get', + params: query + }) +} + +// 强退用户 +export function forceLogout(tokenId) { + return request({ + url: '/monitor/online/' + tokenId, + method: 'delete' + }) +} diff --git a/dvadmin-ui/src/api/monitor/operlog.js b/dvadmin-ui/src/api/monitor/operlog.js new file mode 100755 index 0000000..f09b8ed --- /dev/null +++ b/dvadmin-ui/src/api/monitor/operlog.js @@ -0,0 +1,35 @@ +import request from '@/utils/request' + +// 查询操作日志列表 +export function list(query) { + return request({ + url: '/monitor/operlog/list', + method: 'get', + params: query + }) +} + +// 删除操作日志 +export function delOperlog(operId) { + return request({ + url: '/monitor/operlog/' + operId, + method: 'delete' + }) +} + +// 清空操作日志 +export function cleanOperlog() { + return request({ + url: '/monitor/operlog/clean', + method: 'delete' + }) +} + +// 导出操作日志 +export function exportOperlog(query) { + return request({ + url: '/monitor/operlog/export', + method: 'get', + params: query + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/monitor/server.js b/dvadmin-ui/src/api/monitor/server.js new file mode 100755 index 0000000..feed783 --- /dev/null +++ b/dvadmin-ui/src/api/monitor/server.js @@ -0,0 +1,9 @@ +import request from '@/utils/request' + +// 查询服务器详细 +export function getServer() { + return request({ + url: '/monitor/server', + method: 'get' + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/config.js b/dvadmin-ui/src/api/system/config.js new file mode 100755 index 0000000..93d11fc --- /dev/null +++ b/dvadmin-ui/src/api/system/config.js @@ -0,0 +1,69 @@ +import request from '@/utils/request' + +// 查询参数列表 +export function listConfig(query) { + return request({ + url: '/system/config/list', + method: 'get', + params: query + }) +} + +// 查询参数详细 +export function getConfig(configId) { + return request({ + url: '/system/config/' + configId, + method: 'get' + }) +} + +// 根据参数键名查询参数值 +export function getConfigKey(configKey) { + return request({ + url: '/system/config/configKey/' + configKey, + method: 'get' + }) +} + +// 新增参数配置 +export function addConfig(data) { + return request({ + url: '/system/config', + method: 'post', + data: data + }) +} + +// 修改参数配置 +export function updateConfig(data) { + return request({ + url: '/system/config', + method: 'put', + data: data + }) +} + +// 删除参数配置 +export function delConfig(configId) { + return request({ + url: '/system/config/' + configId, + method: 'delete' + }) +} + +// 清理参数缓存 +export function clearCache() { + return request({ + url: '/system/config/clearCache', + method: 'delete' + }) +} + +// 导出参数 +export function exportConfig(query) { + return request({ + url: '/system/config/export', + method: 'get', + params: query + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/dept.js b/dvadmin-ui/src/api/system/dept.js new file mode 100755 index 0000000..2804676 --- /dev/null +++ b/dvadmin-ui/src/api/system/dept.js @@ -0,0 +1,68 @@ +import request from '@/utils/request' + +// 查询部门列表 +export function listDept(query) { + return request({ + url: '/system/dept/list', + method: 'get', + params: query + }) +} + +// 查询部门列表(排除节点) +export function listDeptExcludeChild(deptId) { + return request({ + url: '/system/dept/list/exclude/' + deptId, + method: 'get' + }) +} + +// 查询部门详细 +export function getDept(deptId) { + return request({ + url: '/system/dept/' + deptId, + method: 'get' + }) +} + +// 查询部门下拉树结构 +export function treeselect() { + return request({ + url: '/system/dept/treeselect', + method: 'get' + }) +} + +// 根据角色ID查询部门树结构 +export function roleDeptTreeselect(roleId) { + return request({ + url: '/system/dept/roleDeptTreeselect/' + roleId, + method: 'get' + }) +} + +// 新增部门 +export function addDept(data) { + return request({ + url: '/system/dept', + method: 'post', + data: data + }) +} + +// 修改部门 +export function updateDept(data) { + return request({ + url: '/system/dept', + method: 'put', + data: data + }) +} + +// 删除部门 +export function delDept(deptId) { + return request({ + url: '/system/dept/' + deptId, + method: 'delete' + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/dict/data.js b/dvadmin-ui/src/api/system/dict/data.js new file mode 100755 index 0000000..d7aca89 --- /dev/null +++ b/dvadmin-ui/src/api/system/dict/data.js @@ -0,0 +1,61 @@ +import request from '@/utils/request' + +// 查询字典数据列表 +export function listData(query) { + return request({ + url: '/system/dict/data/list', + method: 'get', + params: query + }) +} + +// 查询字典数据详细 +export function getData(dictCode) { + return request({ + url: '/system/dict/data/' + dictCode, + method: 'get' + }) +} + +// 根据字典类型查询字典数据信息 +export function getDicts(dictType) { + return request({ + url: '/system/dict/data/type/' + dictType, + method: 'get' + }) +} + +// 新增字典数据 +export function addData(data) { + return request({ + url: '/system/dict/data', + method: 'post', + data: data + }) +} + +// 修改字典数据 +export function updateData(data) { + return request({ + url: '/system/dict/data', + method: 'put', + data: data + }) +} + +// 删除字典数据 +export function delData(dictCode) { + return request({ + url: '/system/dict/data/' + dictCode, + method: 'delete' + }) +} + +// 导出字典数据 +export function exportData(query) { + return request({ + url: '/system/dict/data/export', + method: 'get', + params: query + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/dict/type.js b/dvadmin-ui/src/api/system/dict/type.js new file mode 100755 index 0000000..bc77692 --- /dev/null +++ b/dvadmin-ui/src/api/system/dict/type.js @@ -0,0 +1,69 @@ +import request from '@/utils/request' + +// 查询字典类型列表 +export function listType(query) { + return request({ + url: '/system/dict/type/list', + method: 'get', + params: query + }) +} + +// 查询字典类型详细 +export function getType(dictId) { + return request({ + url: '/system/dict/type/' + dictId, + method: 'get' + }) +} + +// 新增字典类型 +export function addType(data) { + return request({ + url: '/system/dict/type', + method: 'post', + data: data + }) +} + +// 修改字典类型 +export function updateType(data) { + return request({ + url: '/system/dict/type', + method: 'put', + data: data + }) +} + +// 删除字典类型 +export function delType(dictId) { + return request({ + url: '/system/dict/type/' + dictId, + method: 'delete' + }) +} + +// 清理参数缓存 +export function clearCache() { + return request({ + url: '/system/dict/type/clearCache', + method: 'delete' + }) +} + +// 导出字典类型 +export function exportType(query) { + return request({ + url: '/system/dict/type/export', + method: 'get', + params: query + }) +} + +// 获取字典选择框列表 +export function optionselect() { + return request({ + url: '/system/dict/type/optionselect', + method: 'get' + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/menu.js b/dvadmin-ui/src/api/system/menu.js new file mode 100755 index 0000000..f6415c6 --- /dev/null +++ b/dvadmin-ui/src/api/system/menu.js @@ -0,0 +1,60 @@ +import request from '@/utils/request' + +// 查询菜单列表 +export function listMenu(query) { + return request({ + url: '/system/menu/list', + method: 'get', + params: query + }) +} + +// 查询菜单详细 +export function getMenu(menuId) { + return request({ + url: '/system/menu/' + menuId, + method: 'get' + }) +} + +// 查询菜单下拉树结构 +export function treeselect() { + return request({ + url: '/system/menu/treeselect', + method: 'get' + }) +} + +// 根据角色ID查询菜单下拉树结构 +export function roleMenuTreeselect(roleId) { + return request({ + url: '/system/menu/roleMenuTreeselect/' + roleId, + method: 'get' + }) +} + +// 新增菜单 +export function addMenu(data) { + return request({ + url: '/system/menu', + method: 'post', + data: data + }) +} + +// 修改菜单 +export function updateMenu(data) { + return request({ + url: '/system/menu', + method: 'put', + data: data + }) +} + +// 删除菜单 +export function delMenu(menuId) { + return request({ + url: '/system/menu/' + menuId, + method: 'delete' + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/notice.js b/dvadmin-ui/src/api/system/notice.js new file mode 100755 index 0000000..c274ea5 --- /dev/null +++ b/dvadmin-ui/src/api/system/notice.js @@ -0,0 +1,44 @@ +import request from '@/utils/request' + +// 查询公告列表 +export function listNotice(query) { + return request({ + url: '/system/notice/list', + method: 'get', + params: query + }) +} + +// 查询公告详细 +export function getNotice(noticeId) { + return request({ + url: '/system/notice/' + noticeId, + method: 'get' + }) +} + +// 新增公告 +export function addNotice(data) { + return request({ + url: '/system/notice', + method: 'post', + data: data + }) +} + +// 修改公告 +export function updateNotice(data) { + return request({ + url: '/system/notice', + method: 'put', + data: data + }) +} + +// 删除公告 +export function delNotice(noticeId) { + return request({ + url: '/system/notice/' + noticeId, + method: 'delete' + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/post.js b/dvadmin-ui/src/api/system/post.js new file mode 100755 index 0000000..434cd35 --- /dev/null +++ b/dvadmin-ui/src/api/system/post.js @@ -0,0 +1,53 @@ +import request from '@/utils/request' + +// 查询岗位列表 +export function listPost(query) { + return request({ + url: '/system/post/list', + method: 'get', + params: query + }) +} + +// 查询岗位详细 +export function getPost(postId) { + return request({ + url: '/system/post/' + postId, + method: 'get' + }) +} + +// 新增岗位 +export function addPost(data) { + return request({ + url: '/system/post', + method: 'post', + data: data + }) +} + +// 修改岗位 +export function updatePost(data) { + return request({ + url: '/system/post', + method: 'put', + data: data + }) +} + +// 删除岗位 +export function delPost(postId) { + return request({ + url: '/system/post/' + postId, + method: 'delete' + }) +} + +// 导出岗位 +export function exportPost(query) { + return request({ + url: '/system/post/export', + method: 'get', + params: query + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/role.js b/dvadmin-ui/src/api/system/role.js new file mode 100755 index 0000000..463501c --- /dev/null +++ b/dvadmin-ui/src/api/system/role.js @@ -0,0 +1,75 @@ +import request from '@/utils/request' + +// 查询角色列表 +export function listRole(query) { + return request({ + url: '/system/role/list', + method: 'get', + params: query + }) +} + +// 查询角色详细 +export function getRole(roleId) { + return request({ + url: '/system/role/' + roleId, + method: 'get' + }) +} + +// 新增角色 +export function addRole(data) { + return request({ + url: '/system/role', + method: 'post', + data: data + }) +} + +// 修改角色 +export function updateRole(data) { + return request({ + url: '/system/role', + method: 'put', + data: data + }) +} + +// 角色数据权限 +export function dataScope(data) { + return request({ + url: '/system/role/dataScope', + method: 'put', + data: data + }) +} + +// 角色状态修改 +export function changeRoleStatus(roleId, status) { + const data = { + roleId, + status + } + return request({ + url: '/system/role/changeStatus', + method: 'put', + data: data + }) +} + +// 删除角色 +export function delRole(roleId) { + return request({ + url: '/system/role/' + roleId, + method: 'delete' + }) +} + +// 导出角色 +export function exportRole(query) { + return request({ + url: '/system/role/export', + method: 'get', + params: query + }) +} \ No newline at end of file diff --git a/dvadmin-ui/src/api/system/user.js b/dvadmin-ui/src/api/system/user.js new file mode 100755 index 0000000..3b9a776 --- /dev/null +++ b/dvadmin-ui/src/api/system/user.js @@ -0,0 +1,127 @@ +import request from '@/utils/request' +import { praseStrEmpty } from "@/utils/ruoyi"; + +// 查询用户列表 +export function listUser(query) { + return request({ + url: '/system/user/list', + method: 'get', + params: query + }) +} + +// 查询用户详细 +export function getUser(userId) { + return request({ + url: '/system/user/' + praseStrEmpty(userId), + method: 'get' + }) +} + +// 新增用户 +export function addUser(data) { + return request({ + url: '/system/user', + method: 'post', + data: data + }) +} + +// 修改用户 +export function updateUser(data) { + return request({ + url: '/system/user', + method: 'put', + data: data + }) +} + +// 删除用户 +export function delUser(userId) { + return request({ + url: '/system/user/' + userId, + method: 'delete' + }) +} + +// 导出用户 +export function exportUser(query) { + return request({ + url: '/system/user/export', + method: 'get', + params: query + }) +} + +// 用户密码重置 +export function resetUserPwd(userId, password) { + const data = { + userId, + password + } + return request({ + url: '/system/user/resetPwd', + method: 'put', + data: data + }) +} + +// 用户状态修改 +export function changeUserStatus(userId, status) { + const data = { + userId, + status + } + return request({ + url: '/system/user/changeStatus', + method: 'put', + data: data + }) +} + +// 查询用户个人信息 +export function getUserProfile() { + return request({ + url: '/system/user/profile', + method: 'get' + }) +} + +// 修改用户个人信息 +export function updateUserProfile(data) { + return request({ + url: '/system/user/profile', + method: 'put', + data: data + }) +} + +// 用户密码重置 +export function updateUserPwd(oldPassword, newPassword) { + const data = { + oldPassword, + newPassword + } + return request({ + url: '/system/user/profile/updatePwd', + method: 'put', + params: data + }) +} + +// 用户头像上传 +export function uploadAvatar(data) { + return request({ + url: '/system/user/profile/avatar', + method: 'post', + data: data + }) +} + +// 下载用户导入模板 +export function importTemplate() { + return request({ + url: '/system/user/importTemplate', + method: 'get' + }) +} diff --git a/dvadmin-ui/src/api/tool/gen.js b/dvadmin-ui/src/api/tool/gen.js new file mode 100755 index 0000000..4506927 --- /dev/null +++ b/dvadmin-ui/src/api/tool/gen.js @@ -0,0 +1,76 @@ +import request from '@/utils/request' + +// 查询生成表数据 +export function listTable(query) { + return request({ + url: '/tool/gen/list', + method: 'get', + params: query + }) +} +// 查询db数据库列表 +export function listDbTable(query) { + return request({ + url: '/tool/gen/db/list', + method: 'get', + params: query + }) +} + +// 查询表详细信息 +export function getGenTable(tableId) { + return request({ + url: '/tool/gen/' + tableId, + method: 'get' + }) +} + +// 修改代码生成信息 +export function updateGenTable(data) { + return request({ + url: '/tool/gen', + method: 'put', + data: data + }) +} + +// 导入表 +export function importTable(data) { + return request({ + url: '/tool/gen/importTable', + method: 'post', + params: data + }) +} + +// 预览生成代码 +export function previewTable(tableId) { + return request({ + url: '/tool/gen/preview/' + tableId, + method: 'get' + }) +} + +// 删除表数据 +export function delTable(tableId) { + return request({ + url: '/tool/gen/' + tableId, + method: 'delete' + }) +} + +// 生成代码(自定义路径) +export function genCode(tableName) { + return request({ + url: '/tool/gen/genCode/' + tableName, + method: 'get' + }) +} + +// 同步数据库 +export function synchDb(tableName) { + return request({ + url: '/tool/gen/synchDb/' + tableName, + method: 'get' + }) +} diff --git a/dvadmin-ui/src/assets/401_images/401.gif b/dvadmin-ui/src/assets/401_images/401.gif new file mode 100755 index 0000000..cd6e0d9 Binary files /dev/null and b/dvadmin-ui/src/assets/401_images/401.gif differ diff --git a/dvadmin-ui/src/assets/404_images/404.png b/dvadmin-ui/src/assets/404_images/404.png new file mode 100755 index 0000000..3d8e230 Binary files /dev/null and b/dvadmin-ui/src/assets/404_images/404.png differ diff --git a/dvadmin-ui/src/assets/404_images/404_cloud.png b/dvadmin-ui/src/assets/404_images/404_cloud.png new file mode 100755 index 0000000..c6281d0 Binary files /dev/null and b/dvadmin-ui/src/assets/404_images/404_cloud.png differ diff --git a/dvadmin-ui/src/assets/icons/index.js b/dvadmin-ui/src/assets/icons/index.js new file mode 100755 index 0000000..2c6b309 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/index.js @@ -0,0 +1,9 @@ +import Vue from 'vue' +import SvgIcon from '@/components/SvgIcon'// svg component + +// register globally +Vue.component('svg-icon', SvgIcon) + +const req = require.context('./svg', false, /\.svg$/) +const requireAll = requireContext => requireContext.keys().map(requireContext) +requireAll(req) diff --git a/dvadmin-ui/src/assets/icons/svg/404.svg b/dvadmin-ui/src/assets/icons/svg/404.svg new file mode 100755 index 0000000..6df5019 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/404.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/bug.svg b/dvadmin-ui/src/assets/icons/svg/bug.svg new file mode 100755 index 0000000..05a150d --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/bug.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/build.svg b/dvadmin-ui/src/assets/icons/svg/build.svg new file mode 100755 index 0000000..97c4688 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/build.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/button.svg b/dvadmin-ui/src/assets/icons/svg/button.svg new file mode 100755 index 0000000..904fddc --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/button.svg @@ -0,0 +1 @@ +
\ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/cascader.svg b/dvadmin-ui/src/assets/icons/svg/cascader.svg new file mode 100755 index 0000000..e256024 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/cascader.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/chart.svg b/dvadmin-ui/src/assets/icons/svg/chart.svg new file mode 100755 index 0000000..27728fb --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/chart.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/checkbox.svg b/dvadmin-ui/src/assets/icons/svg/checkbox.svg new file mode 100755 index 0000000..013fd3a --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/checkbox.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/clipboard.svg b/dvadmin-ui/src/assets/icons/svg/clipboard.svg new file mode 100755 index 0000000..90923ff --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/clipboard.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/code.svg b/dvadmin-ui/src/assets/icons/svg/code.svg new file mode 100755 index 0000000..ed4d23c --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/code.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/color.svg b/dvadmin-ui/src/assets/icons/svg/color.svg new file mode 100755 index 0000000..44a81aa --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/color.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/component.svg b/dvadmin-ui/src/assets/icons/svg/component.svg new file mode 100755 index 0000000..29c3458 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/component.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/dashboard.svg b/dvadmin-ui/src/assets/icons/svg/dashboard.svg new file mode 100755 index 0000000..5317d37 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/dashboard.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/date-range.svg b/dvadmin-ui/src/assets/icons/svg/date-range.svg new file mode 100755 index 0000000..fda571e --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/date-range.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/date.svg b/dvadmin-ui/src/assets/icons/svg/date.svg new file mode 100755 index 0000000..52dc73e --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/date.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/dict.svg b/dvadmin-ui/src/assets/icons/svg/dict.svg new file mode 100755 index 0000000..4849377 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/dict.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/documentation.svg b/dvadmin-ui/src/assets/icons/svg/documentation.svg new file mode 100755 index 0000000..7043122 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/documentation.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/download.svg b/dvadmin-ui/src/assets/icons/svg/download.svg new file mode 100755 index 0000000..c896951 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/download.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/drag.svg b/dvadmin-ui/src/assets/icons/svg/drag.svg new file mode 100755 index 0000000..4185d3c --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/drag.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/druid.svg b/dvadmin-ui/src/assets/icons/svg/druid.svg new file mode 100755 index 0000000..a2b4b4e --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/druid.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/edit.svg b/dvadmin-ui/src/assets/icons/svg/edit.svg new file mode 100755 index 0000000..d26101f --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/edit.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/education.svg b/dvadmin-ui/src/assets/icons/svg/education.svg new file mode 100755 index 0000000..7bfb01d --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/education.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/email.svg b/dvadmin-ui/src/assets/icons/svg/email.svg new file mode 100755 index 0000000..74d25e2 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/email.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/example.svg b/dvadmin-ui/src/assets/icons/svg/example.svg new file mode 100755 index 0000000..46f42b5 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/example.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/excel.svg b/dvadmin-ui/src/assets/icons/svg/excel.svg new file mode 100755 index 0000000..74d97b8 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/excel.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/exit-fullscreen.svg b/dvadmin-ui/src/assets/icons/svg/exit-fullscreen.svg new file mode 100755 index 0000000..485c128 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/exit-fullscreen.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/eye-open.svg b/dvadmin-ui/src/assets/icons/svg/eye-open.svg new file mode 100755 index 0000000..88dcc98 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/eye-open.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/eye.svg b/dvadmin-ui/src/assets/icons/svg/eye.svg new file mode 100755 index 0000000..16ed2d8 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/eye.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/form.svg b/dvadmin-ui/src/assets/icons/svg/form.svg new file mode 100755 index 0000000..dcbaa18 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/form.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/fullscreen.svg b/dvadmin-ui/src/assets/icons/svg/fullscreen.svg new file mode 100755 index 0000000..0e86b6f --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/fullscreen.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/github.svg b/dvadmin-ui/src/assets/icons/svg/github.svg new file mode 100755 index 0000000..db0a0d4 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/github.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/guide.svg b/dvadmin-ui/src/assets/icons/svg/guide.svg new file mode 100755 index 0000000..b271001 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/guide.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/icon.svg b/dvadmin-ui/src/assets/icons/svg/icon.svg new file mode 100755 index 0000000..82be8ee --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/icon.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/input.svg b/dvadmin-ui/src/assets/icons/svg/input.svg new file mode 100755 index 0000000..ab91381 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/input.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/international.svg b/dvadmin-ui/src/assets/icons/svg/international.svg new file mode 100755 index 0000000..e9b56ee --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/international.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/job.svg b/dvadmin-ui/src/assets/icons/svg/job.svg new file mode 100755 index 0000000..2a93a25 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/job.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/language.svg b/dvadmin-ui/src/assets/icons/svg/language.svg new file mode 100755 index 0000000..0082b57 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/language.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/link.svg b/dvadmin-ui/src/assets/icons/svg/link.svg new file mode 100755 index 0000000..48197ba --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/link.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/list.svg b/dvadmin-ui/src/assets/icons/svg/list.svg new file mode 100755 index 0000000..20259ed --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/list.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/lock.svg b/dvadmin-ui/src/assets/icons/svg/lock.svg new file mode 100755 index 0000000..74fee54 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/lock.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/log.svg b/dvadmin-ui/src/assets/icons/svg/log.svg new file mode 100755 index 0000000..d879d33 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/log.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/logininfor.svg b/dvadmin-ui/src/assets/icons/svg/logininfor.svg new file mode 100755 index 0000000..267f844 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/logininfor.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/message.svg b/dvadmin-ui/src/assets/icons/svg/message.svg new file mode 100755 index 0000000..14ca817 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/message.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/money.svg b/dvadmin-ui/src/assets/icons/svg/money.svg new file mode 100755 index 0000000..c1580de --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/money.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/monitor.svg b/dvadmin-ui/src/assets/icons/svg/monitor.svg new file mode 100755 index 0000000..bc308cb --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/monitor.svg @@ -0,0 +1,2 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/nested.svg b/dvadmin-ui/src/assets/icons/svg/nested.svg new file mode 100755 index 0000000..06713a8 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/nested.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/number.svg b/dvadmin-ui/src/assets/icons/svg/number.svg new file mode 100755 index 0000000..ad5ce9a --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/number.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/online.svg b/dvadmin-ui/src/assets/icons/svg/online.svg new file mode 100755 index 0000000..330a202 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/online.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/password.svg b/dvadmin-ui/src/assets/icons/svg/password.svg new file mode 100755 index 0000000..6c64def --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/password.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/pdf.svg b/dvadmin-ui/src/assets/icons/svg/pdf.svg new file mode 100755 index 0000000..957aa0c --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/pdf.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/people.svg b/dvadmin-ui/src/assets/icons/svg/people.svg new file mode 100755 index 0000000..2bd54ae --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/people.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/peoples.svg b/dvadmin-ui/src/assets/icons/svg/peoples.svg new file mode 100755 index 0000000..aab852e --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/peoples.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/phone.svg b/dvadmin-ui/src/assets/icons/svg/phone.svg new file mode 100755 index 0000000..ab8e8c4 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/phone.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/post.svg b/dvadmin-ui/src/assets/icons/svg/post.svg new file mode 100755 index 0000000..2922c61 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/post.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/qq.svg b/dvadmin-ui/src/assets/icons/svg/qq.svg new file mode 100755 index 0000000..ee13d4e --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/qq.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/question.svg b/dvadmin-ui/src/assets/icons/svg/question.svg new file mode 100755 index 0000000..cf75bd4 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/question.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/radio.svg b/dvadmin-ui/src/assets/icons/svg/radio.svg new file mode 100755 index 0000000..0cde345 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/radio.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/rate.svg b/dvadmin-ui/src/assets/icons/svg/rate.svg new file mode 100755 index 0000000..aa3b14d --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/rate.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/redis.svg b/dvadmin-ui/src/assets/icons/svg/redis.svg new file mode 100755 index 0000000..2f1d62d --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/redis.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/row.svg b/dvadmin-ui/src/assets/icons/svg/row.svg new file mode 100755 index 0000000..0780992 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/row.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/search.svg b/dvadmin-ui/src/assets/icons/svg/search.svg new file mode 100755 index 0000000..84233dd --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/search.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/select.svg b/dvadmin-ui/src/assets/icons/svg/select.svg new file mode 100755 index 0000000..d628382 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/select.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/server.svg b/dvadmin-ui/src/assets/icons/svg/server.svg new file mode 100755 index 0000000..ca37b00 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/server.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/shopping.svg b/dvadmin-ui/src/assets/icons/svg/shopping.svg new file mode 100755 index 0000000..87513e7 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/shopping.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/size.svg b/dvadmin-ui/src/assets/icons/svg/size.svg new file mode 100755 index 0000000..ddb25b8 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/size.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/skill.svg b/dvadmin-ui/src/assets/icons/svg/skill.svg new file mode 100755 index 0000000..a3b7312 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/skill.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/slider.svg b/dvadmin-ui/src/assets/icons/svg/slider.svg new file mode 100755 index 0000000..fbe4f39 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/slider.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/star.svg b/dvadmin-ui/src/assets/icons/svg/star.svg new file mode 100755 index 0000000..6cf86e6 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/star.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/swagger.svg b/dvadmin-ui/src/assets/icons/svg/swagger.svg new file mode 100755 index 0000000..05d4e7b --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/swagger.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/switch.svg b/dvadmin-ui/src/assets/icons/svg/switch.svg new file mode 100755 index 0000000..0ba61e3 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/switch.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/system.svg b/dvadmin-ui/src/assets/icons/svg/system.svg new file mode 100755 index 0000000..dba28cf --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/system.svg @@ -0,0 +1,2 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/tab.svg b/dvadmin-ui/src/assets/icons/svg/tab.svg new file mode 100755 index 0000000..b4b48e4 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/tab.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/table.svg b/dvadmin-ui/src/assets/icons/svg/table.svg new file mode 100755 index 0000000..0e3dc9d --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/table.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/textarea.svg b/dvadmin-ui/src/assets/icons/svg/textarea.svg new file mode 100755 index 0000000..2709f29 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/textarea.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/theme.svg b/dvadmin-ui/src/assets/icons/svg/theme.svg new file mode 100755 index 0000000..5982a2f --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/theme.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/time-range.svg b/dvadmin-ui/src/assets/icons/svg/time-range.svg new file mode 100755 index 0000000..13c1202 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/time-range.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/time.svg b/dvadmin-ui/src/assets/icons/svg/time.svg new file mode 100755 index 0000000..b376e32 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/time.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/tool.svg b/dvadmin-ui/src/assets/icons/svg/tool.svg new file mode 100755 index 0000000..c813067 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/tool.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/tree-table.svg b/dvadmin-ui/src/assets/icons/svg/tree-table.svg new file mode 100755 index 0000000..8aafdb8 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/tree-table.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/tree.svg b/dvadmin-ui/src/assets/icons/svg/tree.svg new file mode 100755 index 0000000..dd4b7dd --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/tree.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/upload.svg b/dvadmin-ui/src/assets/icons/svg/upload.svg new file mode 100755 index 0000000..bae49c0 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/upload.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/user.svg b/dvadmin-ui/src/assets/icons/svg/user.svg new file mode 100755 index 0000000..0ba0716 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/user.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/validCode.svg b/dvadmin-ui/src/assets/icons/svg/validCode.svg new file mode 100755 index 0000000..cfb1021 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/validCode.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/wechat.svg b/dvadmin-ui/src/assets/icons/svg/wechat.svg new file mode 100755 index 0000000..c586e55 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/wechat.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svg/zip.svg b/dvadmin-ui/src/assets/icons/svg/zip.svg new file mode 100755 index 0000000..f806fc4 --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svg/zip.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/icons/svgo.yml b/dvadmin-ui/src/assets/icons/svgo.yml new file mode 100755 index 0000000..d11906a --- /dev/null +++ b/dvadmin-ui/src/assets/icons/svgo.yml @@ -0,0 +1,22 @@ +# replace default config + +# multipass: true +# full: true + +plugins: + + # - name + # + # or: + # - name: false + # - name: true + # + # or: + # - name: + # param1: 1 + # param2: 2 + +- removeAttrs: + attrs: + - 'fill' + - 'fill-rule' diff --git a/dvadmin-ui/src/assets/images/dark.svg b/dvadmin-ui/src/assets/images/dark.svg new file mode 100755 index 0000000..f646bd7 --- /dev/null +++ b/dvadmin-ui/src/assets/images/dark.svg @@ -0,0 +1,39 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/images/light.svg b/dvadmin-ui/src/assets/images/light.svg new file mode 100755 index 0000000..ab7cc08 --- /dev/null +++ b/dvadmin-ui/src/assets/images/light.svg @@ -0,0 +1,39 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/dvadmin-ui/src/assets/images/login-background.jpg b/dvadmin-ui/src/assets/images/login-background.jpg new file mode 100755 index 0000000..8a89eb8 Binary files /dev/null and b/dvadmin-ui/src/assets/images/login-background.jpg differ diff --git a/dvadmin-ui/src/assets/images/profile.jpg b/dvadmin-ui/src/assets/images/profile.jpg new file mode 100755 index 0000000..b3a940b Binary files /dev/null and b/dvadmin-ui/src/assets/images/profile.jpg differ diff --git a/dvadmin-ui/src/assets/logo/logo.png b/dvadmin-ui/src/assets/logo/logo.png new file mode 100755 index 0000000..e263760 Binary files /dev/null and b/dvadmin-ui/src/assets/logo/logo.png differ diff --git a/dvadmin-ui/src/assets/styles/btn.scss b/dvadmin-ui/src/assets/styles/btn.scss new file mode 100755 index 0000000..e6ba1a8 --- /dev/null +++ b/dvadmin-ui/src/assets/styles/btn.scss @@ -0,0 +1,99 @@ +@import './variables.scss'; + +@mixin colorBtn($color) { + background: $color; + + &:hover { + color: $color; + + &:before, + &:after { + background: $color; + } + } +} + +.blue-btn { + @include colorBtn($blue) +} + +.light-blue-btn { + @include colorBtn($light-blue) +} + +.red-btn { + @include colorBtn($red) +} + +.pink-btn { + @include colorBtn($pink) +} + +.green-btn { + @include colorBtn($green) +} + +.tiffany-btn { + @include colorBtn($tiffany) +} + +.yellow-btn { + @include colorBtn($yellow) +} + +.pan-btn { + font-size: 14px; + color: #fff; + padding: 14px 36px; + border-radius: 8px; + border: none; + outline: none; + transition: 600ms ease all; + position: relative; + display: inline-block; + + &:hover { + background: #fff; + + &:before, + &:after { + width: 100%; + transition: 600ms ease all; + } + } + + &:before, + &:after { + content: ''; + position: absolute; + top: 0; + right: 0; + height: 2px; + width: 0; + transition: 400ms ease all; + } + + &::after { + right: inherit; + top: inherit; + left: 0; + bottom: 0; + } +} + +.custom-button { + display: inline-block; + line-height: 1; + white-space: nowrap; + cursor: pointer; + background: #fff; + color: #fff; + -webkit-appearance: none; + text-align: center; + box-sizing: border-box; + outline: 0; + margin: 0; + padding: 10px 15px; + font-size: 14px; + border-radius: 4px; +} diff --git a/dvadmin-ui/src/assets/styles/element-ui.scss b/dvadmin-ui/src/assets/styles/element-ui.scss new file mode 100755 index 0000000..955d3ca --- /dev/null +++ b/dvadmin-ui/src/assets/styles/element-ui.scss @@ -0,0 +1,84 @@ +// cover some element-ui styles + +.el-breadcrumb__inner, +.el-breadcrumb__inner a { + font-weight: 400 !important; +} + +.el-upload { + input[type="file"] { + display: none !important; + } +} + +.el-upload__input { + display: none; +} + +.cell { + .el-tag { + margin-right: 0px; + } +} + +.small-padding { + .cell { + padding-left: 5px; + padding-right: 5px; + } +} + +.fixed-width { + .el-button--mini { + padding: 7px 10px; + width: 60px; + } +} + +.status-col { + .cell { + padding: 0 10px; + text-align: center; + + .el-tag { + margin-right: 0px; + } + } +} + +// to fixed https://github.com/ElemeFE/element/issues/2461 +.el-dialog { + transform: none; + left: 0; + position: relative; + margin: 0 auto; +} + +// refine element ui upload +.upload-container { + .el-upload { + width: 100%; + + .el-upload-dragger { + width: 100%; + height: 200px; + } + } +} + +// dropdown +.el-dropdown-menu { + a { + display: block + } +} + +// fix date-picker ui bug in filter-item +.el-range-editor.el-input__inner { + display: inline-flex !important; +} + +// to fix el-date-picker css style +.el-range-separator { + box-sizing: content-box; +} diff --git a/dvadmin-ui/src/assets/styles/element-variables.scss b/dvadmin-ui/src/assets/styles/element-variables.scss new file mode 100755 index 0000000..8b7a48e --- /dev/null +++ b/dvadmin-ui/src/assets/styles/element-variables.scss @@ -0,0 +1,31 @@ +/** +* I think element-ui's default theme color is too light for long-term use. +* So I modified the default color and you can modify it to your liking. +**/ + +/* theme color */ +$--color-primary: #1890ff; +$--color-success: #13ce66; +$--color-warning: #ffba00; +$--color-danger: #ff4949; +// $--color-info: #1E1E1E; + +$--button-font-weight: 400; + +// $--color-text-regular: #1f2d3d; + +$--border-color-light: #dfe4ed; +$--border-color-lighter: #e6ebf5; + +$--table-border:1px solid#dfe6ec; + +/* icon font path, required */ +$--font-path: '~element-ui/lib/theme-chalk/fonts'; + +@import "~element-ui/packages/theme-chalk/src/index"; + +// the :export directive is the magic sauce for webpack +// https://www.bluematador.com/blog/how-to-share-variables-between-js-and-sass +:export { + theme: $--color-primary; +} diff --git a/dvadmin-ui/src/assets/styles/index.scss b/dvadmin-ui/src/assets/styles/index.scss new file mode 100755 index 0000000..96095ef --- /dev/null +++ b/dvadmin-ui/src/assets/styles/index.scss @@ -0,0 +1,191 @@ +@import './variables.scss'; +@import './mixin.scss'; +@import './transition.scss'; +@import './element-ui.scss'; +@import './sidebar.scss'; +@import './btn.scss'; + +body { + height: 100%; + -moz-osx-font-smoothing: grayscale; + -webkit-font-smoothing: antialiased; + text-rendering: optimizeLegibility; + font-family: Helvetica Neue, Helvetica, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; +} + +label { + font-weight: 700; +} + +html { + height: 100%; + box-sizing: border-box; +} + +#app { + height: 100%; +} + +*, +*:before, +*:after { + box-sizing: inherit; +} + +.no-padding { + padding: 0px !important; +} + +.padding-content { + padding: 4px 0; +} + +a:focus, +a:active { + outline: none; +} + +a, +a:focus, +a:hover { + cursor: pointer; + color: inherit; + text-decoration: none; +} + +div:focus { + outline: none; +} + +.fr { + float: right; +} + +.fl { + float: left; +} + +.pr-5 { + padding-right: 5px; +} + +.pl-5 { + padding-left: 5px; +} + +.block { + display: block; +} + +.pointer { + cursor: pointer; +} + +.inlineBlock { + display: block; +} + +.clearfix { + &:after { + visibility: hidden; + display: block; + font-size: 0; + content: " "; + clear: both; + height: 0; + } +} + +aside { + background: #eef1f6; + padding: 8px 24px; + margin-bottom: 20px; + border-radius: 2px; + display: block; + line-height: 32px; + font-size: 16px; + font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Fira Sans", "Droid Sans", "Helvetica Neue", sans-serif; + color: #2c3e50; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + + a { + color: #337ab7; + cursor: pointer; + + &:hover { + color: rgb(32, 160, 255); + } + } +} + +//main-container全局样式 +.app-container { + padding: 20px; +} + +.components-container { + margin: 30px 50px; + position: relative; +} + +.pagination-container { + margin-top: 30px; +} + +.text-center { + text-align: center +} + +.sub-navbar { + height: 50px; + line-height: 50px; + position: relative; + width: 100%; + text-align: right; + padding-right: 20px; + transition: 600ms ease position; + background: linear-gradient(90deg, rgba(32, 182, 249, 1) 0%, rgba(32, 182, 249, 1) 0%, rgba(33, 120, 241, 1) 100%, rgba(33, 120, 241, 1) 100%); + + .subtitle { + font-size: 20px; + color: #fff; + } + + &.draft { + background: #d0d0d0; + } + + &.deleted { + background: #d0d0d0; + } +} + +.link-type, +.link-type:focus { + color: #337ab7; + cursor: pointer; + + &:hover { + color: rgb(32, 160, 255); + } +} + +.filter-container { + padding-bottom: 10px; + + .filter-item { + display: inline-block; + vertical-align: middle; + margin-bottom: 10px; + } +} + +//refine vue-multiselect plugin +.multiselect { + line-height: 16px; +} + +.multiselect--active { + z-index: 1000 !important; +} diff --git a/dvadmin-ui/src/assets/styles/mixin.scss b/dvadmin-ui/src/assets/styles/mixin.scss new file mode 100755 index 0000000..06fa061 --- /dev/null +++ b/dvadmin-ui/src/assets/styles/mixin.scss @@ -0,0 +1,66 @@ +@mixin clearfix { + &:after { + content: ""; + display: table; + clear: both; + } +} + +@mixin scrollBar { + &::-webkit-scrollbar-track-piece { + background: #d3dce6; + } + + &::-webkit-scrollbar { + width: 6px; + } + + &::-webkit-scrollbar-thumb { + background: #99a9bf; + border-radius: 20px; + } +} + +@mixin relative { + position: relative; + width: 100%; + height: 100%; +} + +@mixin pct($pct) { + width: #{$pct}; + position: relative; + margin: 0 auto; +} + +@mixin triangle($width, $height, $color, $direction) { + $width: $width/2; + $color-border-style: $height solid $color; + $transparent-border-style: $width solid transparent; + height: 0; + width: 0; + + @if $direction==up { + border-bottom: $color-border-style; + border-left: $transparent-border-style; + border-right: $transparent-border-style; + } + + @else if $direction==right { + border-left: $color-border-style; + border-top: $transparent-border-style; + border-bottom: $transparent-border-style; + } + + @else if $direction==down { + border-top: $color-border-style; + border-left: $transparent-border-style; + border-right: $transparent-border-style; + } + + @else if $direction==left { + border-right: $color-border-style; + border-top: $transparent-border-style; + border-bottom: $transparent-border-style; + } +} diff --git a/dvadmin-ui/src/assets/styles/ruoyi.scss b/dvadmin-ui/src/assets/styles/ruoyi.scss new file mode 100755 index 0000000..b1a1661 --- /dev/null +++ b/dvadmin-ui/src/assets/styles/ruoyi.scss @@ -0,0 +1,240 @@ + /** + * 通用css样式布局处理 + * Copyright (c) 2019 ruoyi + */ + + /** 基础通用 **/ +.pt5 { + padding-top: 5px; +} +.pr5 { + padding-right: 5px; +} +.pb5 { + padding-bottom: 5px; +} +.mt5 { + margin-top: 5px; +} +.mr5 { + margin-right: 5px; +} +.mb5 { + margin-bottom: 5px; +} +.mb8 { + margin-bottom: 8px; +} +.ml5 { + margin-left: 5px; +} +.mt10 { + margin-top: 10px; +} +.mr10 { + margin-right: 10px; +} +.mb10 { + margin-bottom: 10px; +} +.ml0 { + margin-left: 10px; +} +.mt20 { + margin-top: 20px; +} +.mr20 { + margin-right: 20px; +} +.mb20 { + margin-bottom: 20px; +} +.m20 { + margin-left: 20px; +} + +.el-dialog:not(.is-fullscreen){ + margin-top: 6vh !important; +} + +.el-table { + .el-table__header-wrapper, .el-table__fixed-header-wrapper { + th { + word-break: break-word; + background-color: #f8f8f9; + color: #515a6e; + height: 40px; + font-size: 13px; + } + } + .el-table__body-wrapper { + .el-button [class*="el-icon-"] + span { + margin-left: 1px; + } + } +} + +/** 表单布局 **/ +.form-header { + font-size:15px; + color:#6379bb; + border-bottom:1px solid #ddd; + margin:8px 10px 25px 10px; + padding-bottom:5px +} + +/** 表格布局 **/ +.pagination-container { + position: relative; + height: 25px; + margin-bottom: 10px; + margin-top: 15px; + padding: 10px 20px !important; +} + +/* tree border */ +.tree-border { + margin-top: 5px; + border: 1px solid #e5e6e7; + background: #FFFFFF none; + border-radius:4px; +} + +.pagination-container .el-pagination { + right: 0; + position: absolute; +} + +.el-table .fixed-width .el-button--mini { + color: #409EFF; + padding-left: 0; + padding-right: 0; + width: inherit; +} + +.el-tree-node__content > .el-checkbox { + margin-right: 8px; +} + +.list-group-striped > .list-group-item { + border-left: 0; + border-right: 0; + border-radius: 0; + padding-left: 0; + padding-right: 0; +} + +.list-group { + padding-left: 0px; + list-style: none; +} + +.list-group-item { + border-bottom: 1px solid #e7eaec; + border-top: 1px solid #e7eaec; + margin-bottom: -1px; + padding: 11px 0px; + font-size: 13px; +} + +.pull-right { + float: right !important; +} + +.el-card__header { + padding: 14px 15px 7px; + min-height: 40px; +} + +.el-card__body { + padding: 15px 20px 20px 20px; +} + +.card-box { + padding-right: 15px; + padding-left: 15px; + margin-bottom: 10px; +} + +/* button color */ +.el-button--cyan.is-active, +.el-button--cyan:active { + background: #20B2AA; + border-color: #20B2AA; + color: #FFFFFF; +} + +.el-button--cyan:focus, +.el-button--cyan:hover { + background: #48D1CC; + border-color: #48D1CC; + color: #FFFFFF; +} + +.el-button--cyan { + background-color: #20B2AA; + border-color: #20B2AA; + color: #FFFFFF; +} + +/* text color */ +.text-navy { + color: #1ab394; +} + +.text-primary { + color: inherit; +} + +.text-success { + color: #1c84c6; +} + +.text-info { + color: #23c6c8; +} + +.text-warning { + color: #f8ac59; +} + +.text-danger { + color: #ed5565; +} + +.text-muted { + color: #888888; +} + +/* image */ +.img-circle { + border-radius: 50%; +} + +.img-lg { + width: 120px; + height: 120px; +} + +.avatar-upload-preview { + position: absolute; + top: 50%; + transform: translate(50%, -50%); + width: 200px; + height: 200px; + border-radius: 50%; + box-shadow: 0 0 4px #ccc; + overflow: hidden; +} + +/* 拖拽列样式 */ +.sortable-ghost{ + opacity: .8; + color: #fff!important; + background: #42b983!important; +} + +.top-right-btn { + position: relative; + float: right; +} diff --git a/dvadmin-ui/src/assets/styles/sidebar.scss b/dvadmin-ui/src/assets/styles/sidebar.scss new file mode 100755 index 0000000..0eb8e78 --- /dev/null +++ b/dvadmin-ui/src/assets/styles/sidebar.scss @@ -0,0 +1,226 @@ +#app { + + .main-container { + min-height: 100%; + transition: margin-left .28s; + margin-left: $sideBarWidth; + position: relative; + } + + .sidebar-container { + -webkit-transition: width .28s; + transition: width 0.28s; + width: $sideBarWidth !important; + background-color: $menuBg; + height: 100%; + position: fixed; + font-size: 0px; + top: 0; + bottom: 0; + left: 0; + z-index: 1001; + overflow: hidden; + -webkit-box-shadow: 2px 0 6px rgba(0,21,41,.35); + box-shadow: 2px 0 6px rgba(0,21,41,.35); + + // reset element-ui css + .horizontal-collapse-transition { + transition: 0s width ease-in-out, 0s padding-left ease-in-out, 0s padding-right ease-in-out; + } + + .scrollbar-wrapper { + overflow-x: hidden !important; + } + + .el-scrollbar__bar.is-vertical { + right: 0px; + } + + .el-scrollbar { + height: 100%; + } + + &.has-logo { + .el-scrollbar { + height: calc(100% - 50px); + } + } + + .is-horizontal { + display: none; + } + + a { + display: inline-block; + width: 100%; + overflow: hidden; + } + + .svg-icon { + margin-right: 16px; + } + + .el-menu { + border: none; + height: 100%; + width: 100% !important; + } + + .el-menu-item, .el-submenu__title { + overflow: hidden !important; + text-overflow: ellipsis !important; + white-space: nowrap !important; + } + + // menu hover + .submenu-title-noDropdown, + .el-submenu__title { + &:hover { + background-color: rgba(0, 0, 0, 0.06) !important; + } + } + + & .theme-dark .is-active > .el-submenu__title { + color: $subMenuActiveText !important; + } + + & .nest-menu .el-submenu>.el-submenu__title, + & .el-submenu .el-menu-item { + min-width: $sideBarWidth !important; + + &:hover { + background-color: rgba(0, 0, 0, 0.06) !important; + } + } + + & .theme-dark .nest-menu .el-submenu>.el-submenu__title, + & .theme-dark .el-submenu .el-menu-item { + background-color: $subMenuBg !important; + + &:hover { + background-color: $subMenuHover !important; + } + } + } + + .hideSidebar { + .sidebar-container { + width: 54px !important; + } + + .main-container { + margin-left: 54px; + } + + .submenu-title-noDropdown { + padding: 0 !important; + position: relative; + + .el-tooltip { + padding: 0 !important; + + .svg-icon { + margin-left: 20px; + } + } + } + + .el-submenu { + overflow: hidden; + + &>.el-submenu__title { + padding: 0 !important; + + .svg-icon { + margin-left: 20px; + } + + .el-submenu__icon-arrow { + display: none; + } + } + } + + .el-menu--collapse { + .el-submenu { + &>.el-submenu__title { + &>span { + height: 0; + width: 0; + overflow: hidden; + visibility: hidden; + display: inline-block; + } + } + } + } + } + + .el-menu--collapse .el-menu .el-submenu { + min-width: $sideBarWidth !important; + } + + // mobile responsive + .mobile { + .main-container { + margin-left: 0px; + } + + .sidebar-container { + transition: transform .28s; + width: $sideBarWidth !important; + } + + &.hideSidebar { + .sidebar-container { + pointer-events: none; + transition-duration: 0.3s; + transform: translate3d(-$sideBarWidth, 0, 0); + } + } + } + + .withoutAnimation { + + .main-container, + .sidebar-container { + transition: none; + } + } +} + +// when menu collapsed +.el-menu--vertical { + &>.el-menu { + .svg-icon { + margin-right: 16px; + } + } + + .nest-menu .el-submenu>.el-submenu__title, + .el-menu-item { + &:hover { + // you can use $subMenuHover + background-color: rgba(0, 0, 0, 0.06) !important; + } + } + + // the scroll bar appears when the subMenu is too long + >.el-menu--popup { + max-height: 100vh; + overflow-y: auto; + + &::-webkit-scrollbar-track-piece { + background: #d3dce6; + } + + &::-webkit-scrollbar { + width: 6px; + } + + &::-webkit-scrollbar-thumb { + background: #99a9bf; + border-radius: 20px; + } + } +} diff --git a/dvadmin-ui/src/assets/styles/transition.scss b/dvadmin-ui/src/assets/styles/transition.scss new file mode 100755 index 0000000..4cb27cc --- /dev/null +++ b/dvadmin-ui/src/assets/styles/transition.scss @@ -0,0 +1,48 @@ +// global transition css + +/* fade */ +.fade-enter-active, +.fade-leave-active { + transition: opacity 0.28s; +} + +.fade-enter, +.fade-leave-active { + opacity: 0; +} + +/* fade-transform */ +.fade-transform-leave-active, +.fade-transform-enter-active { + transition: all .5s; +} + +.fade-transform-enter { + opacity: 0; + transform: translateX(-30px); +} + +.fade-transform-leave-to { + opacity: 0; + transform: translateX(30px); +} + +/* breadcrumb transition */ +.breadcrumb-enter-active, +.breadcrumb-leave-active { + transition: all .5s; +} + +.breadcrumb-enter, +.breadcrumb-leave-active { + opacity: 0; + transform: translateX(20px); +} + +.breadcrumb-move { + transition: all .5s; +} + +.breadcrumb-leave-active { + position: absolute; +} diff --git a/dvadmin-ui/src/assets/styles/variables.scss b/dvadmin-ui/src/assets/styles/variables.scss new file mode 100755 index 0000000..452a1ec --- /dev/null +++ b/dvadmin-ui/src/assets/styles/variables.scss @@ -0,0 +1,44 @@ +// base color +$blue:#324157; +$light-blue:#3A71A8; +$red:#C03639; +$pink: #E65D6E; +$green: #30B08F; +$tiffany: #4AB7BD; +$yellow:#FEC171; +$panGreen: #30B08F; + +// sidebar +$menuText:#bfcbd9; +$menuActiveText:#409EFF; +$subMenuActiveText:#f4f4f5; // https://github.com/ElemeFE/element/issues/12951 + +$menuBg:#304156; +$menuHover:#263445; +$sidebarTitle: #ffffff; + +$menuLightBg:#ffffff; +$menuLightHover:#f0f1f5; +$sidebarLightTitle: #001529; + +$subMenuBg:#1f2d3d; +$subMenuHover:#001528; + +$sideBarWidth: 200px; + +// the :export directive is the magic sauce for webpack +// https://www.bluematador.com/blog/how-to-share-variables-between-js-and-sass +:export { + menuText: $menuText; + menuActiveText: $menuActiveText; + subMenuActiveText: $subMenuActiveText; + menuBg: $menuBg; + menuHover: $menuHover; + menuLightBg: $menuLightBg; + menuLightHover: $menuLightHover; + subMenuBg: $subMenuBg; + subMenuHover: $subMenuHover; + sideBarWidth: $sideBarWidth; + sidebarTitle: $sidebarTitle; + sidebarLightTitle: $sidebarLightTitle +} diff --git a/dvadmin-ui/src/components/Breadcrumb/index.vue b/dvadmin-ui/src/components/Breadcrumb/index.vue new file mode 100755 index 0000000..b313fdd --- /dev/null +++ b/dvadmin-ui/src/components/Breadcrumb/index.vue @@ -0,0 +1,74 @@ + + + + + diff --git a/dvadmin-ui/src/components/Editor/index.vue b/dvadmin-ui/src/components/Editor/index.vue new file mode 100755 index 0000000..ad35bbf --- /dev/null +++ b/dvadmin-ui/src/components/Editor/index.vue @@ -0,0 +1,195 @@ + + + + + \ No newline at end of file diff --git a/dvadmin-ui/src/components/FileUpload/index.vue b/dvadmin-ui/src/components/FileUpload/index.vue new file mode 100755 index 0000000..69942aa --- /dev/null +++ b/dvadmin-ui/src/components/FileUpload/index.vue @@ -0,0 +1,179 @@ + + + + + \ No newline at end of file diff --git a/dvadmin-ui/src/components/Hamburger/index.vue b/dvadmin-ui/src/components/Hamburger/index.vue new file mode 100755 index 0000000..368b002 --- /dev/null +++ b/dvadmin-ui/src/components/Hamburger/index.vue @@ -0,0 +1,44 @@ + + + + + diff --git a/dvadmin-ui/src/components/HeaderSearch/index.vue b/dvadmin-ui/src/components/HeaderSearch/index.vue new file mode 100755 index 0000000..ce9f305 --- /dev/null +++ b/dvadmin-ui/src/components/HeaderSearch/index.vue @@ -0,0 +1,188 @@ + + + + + diff --git a/dvadmin-ui/src/components/IconSelect/index.vue b/dvadmin-ui/src/components/IconSelect/index.vue new file mode 100755 index 0000000..b0ec9fa --- /dev/null +++ b/dvadmin-ui/src/components/IconSelect/index.vue @@ -0,0 +1,68 @@ + + + + + + diff --git a/dvadmin-ui/src/components/IconSelect/requireIcons.js b/dvadmin-ui/src/components/IconSelect/requireIcons.js new file mode 100755 index 0000000..99e5c54 --- /dev/null +++ b/dvadmin-ui/src/components/IconSelect/requireIcons.js @@ -0,0 +1,11 @@ + +const req = require.context('../../assets/icons/svg', false, /\.svg$/) +const requireAll = requireContext => requireContext.keys() + +const re = /\.\/(.*)\.svg/ + +const icons = requireAll(req).map(i => { + return i.match(re)[1] +}) + +export default icons diff --git a/dvadmin-ui/src/components/ImageUpload/index.vue b/dvadmin-ui/src/components/ImageUpload/index.vue new file mode 100755 index 0000000..8996329 --- /dev/null +++ b/dvadmin-ui/src/components/ImageUpload/index.vue @@ -0,0 +1,100 @@ + + + + + \ No newline at end of file diff --git a/dvadmin-ui/src/components/Pagination/index.vue b/dvadmin-ui/src/components/Pagination/index.vue new file mode 100755 index 0000000..c815e13 --- /dev/null +++ b/dvadmin-ui/src/components/Pagination/index.vue @@ -0,0 +1,101 @@ + + + + + diff --git a/dvadmin-ui/src/components/PanThumb/index.vue b/dvadmin-ui/src/components/PanThumb/index.vue new file mode 100755 index 0000000..1bcf417 --- /dev/null +++ b/dvadmin-ui/src/components/PanThumb/index.vue @@ -0,0 +1,142 @@ + + + + + diff --git a/dvadmin-ui/src/components/ParentView/index.vue b/dvadmin-ui/src/components/ParentView/index.vue new file mode 100755 index 0000000..7bf6148 --- /dev/null +++ b/dvadmin-ui/src/components/ParentView/index.vue @@ -0,0 +1,3 @@ + diff --git a/dvadmin-ui/src/components/RightPanel/index.vue b/dvadmin-ui/src/components/RightPanel/index.vue new file mode 100755 index 0000000..fbf27eb --- /dev/null +++ b/dvadmin-ui/src/components/RightPanel/index.vue @@ -0,0 +1,149 @@ + + + + + + + diff --git a/dvadmin-ui/src/components/RightToolbar/index.vue b/dvadmin-ui/src/components/RightToolbar/index.vue new file mode 100755 index 0000000..92c65a5 --- /dev/null +++ b/dvadmin-ui/src/components/RightToolbar/index.vue @@ -0,0 +1,80 @@ + + + diff --git a/dvadmin-ui/src/components/RuoYi/Doc/index.vue b/dvadmin-ui/src/components/RuoYi/Doc/index.vue new file mode 100755 index 0000000..a6187f3 --- /dev/null +++ b/dvadmin-ui/src/components/RuoYi/Doc/index.vue @@ -0,0 +1,21 @@ + + + \ No newline at end of file diff --git a/dvadmin-ui/src/components/RuoYi/Git/index.vue b/dvadmin-ui/src/components/RuoYi/Git/index.vue new file mode 100755 index 0000000..1d09a77 --- /dev/null +++ b/dvadmin-ui/src/components/RuoYi/Git/index.vue @@ -0,0 +1,21 @@ + + + \ No newline at end of file diff --git a/dvadmin-ui/src/components/Screenfull/index.vue b/dvadmin-ui/src/components/Screenfull/index.vue new file mode 100755 index 0000000..d4e539c --- /dev/null +++ b/dvadmin-ui/src/components/Screenfull/index.vue @@ -0,0 +1,57 @@ + + + + + diff --git a/dvadmin-ui/src/components/SizeSelect/index.vue b/dvadmin-ui/src/components/SizeSelect/index.vue new file mode 100755 index 0000000..e88065b --- /dev/null +++ b/dvadmin-ui/src/components/SizeSelect/index.vue @@ -0,0 +1,57 @@ + + + diff --git a/dvadmin-ui/src/components/SvgIcon/index.vue b/dvadmin-ui/src/components/SvgIcon/index.vue new file mode 100755 index 0000000..e4bf5ad --- /dev/null +++ b/dvadmin-ui/src/components/SvgIcon/index.vue @@ -0,0 +1,61 @@ + + + + + diff --git a/dvadmin-ui/src/components/ThemePicker/index.vue b/dvadmin-ui/src/components/ThemePicker/index.vue new file mode 100755 index 0000000..3879c5a --- /dev/null +++ b/dvadmin-ui/src/components/ThemePicker/index.vue @@ -0,0 +1,175 @@ + + + + + diff --git a/dvadmin-ui/src/directive/permission/hasPermi.js b/dvadmin-ui/src/directive/permission/hasPermi.js new file mode 100755 index 0000000..d7107ce --- /dev/null +++ b/dvadmin-ui/src/directive/permission/hasPermi.js @@ -0,0 +1,28 @@ + /** + * 操作权限处理 + * Copyright (c) 2019 ruoyi + */ + +import store from '@/store' + +export default { + inserted(el, binding, vnode) { + const { value } = binding + const all_permission = "*:*:*"; + const permissions = store.getters && store.getters.permissions + + if (value && value instanceof Array && value.length > 0) { + const permissionFlag = value + + const hasPermissions = permissions.some(permission => { + return all_permission === permission || permissionFlag.includes(permission) + }) + + if (!hasPermissions) { + el.parentNode && el.parentNode.removeChild(el) + } + } else { + throw new Error(`请设置操作权限标签值`) + } + } +} diff --git a/dvadmin-ui/src/directive/permission/hasRole.js b/dvadmin-ui/src/directive/permission/hasRole.js new file mode 100755 index 0000000..1303809 --- /dev/null +++ b/dvadmin-ui/src/directive/permission/hasRole.js @@ -0,0 +1,28 @@ + /** + * 角色权限处理 + * Copyright (c) 2019 ruoyi + */ + +import store from '@/store' + +export default { + inserted(el, binding, vnode) { + const { value } = binding + const super_admin = "admin"; + const roles = store.getters && store.getters.roles + + if (value && value instanceof Array && value.length > 0) { + const roleFlag = value + + const hasRole = roles.some(role => { + return super_admin === role || roleFlag.includes(role) + }) + + if (!hasRole) { + el.parentNode && el.parentNode.removeChild(el) + } + } else { + throw new Error(`请设置角色权限标签值"`) + } + } +} diff --git a/dvadmin-ui/src/directive/permission/index.js b/dvadmin-ui/src/directive/permission/index.js new file mode 100755 index 0000000..e3d76d3 --- /dev/null +++ b/dvadmin-ui/src/directive/permission/index.js @@ -0,0 +1,15 @@ +import hasRole from './hasRole' +import hasPermi from './hasPermi' + +const install = function(Vue) { + Vue.directive('hasRole', hasRole) + Vue.directive('hasPermi', hasPermi) +} + +if (window.Vue) { + window['hasRole'] = hasRole + window['hasPermi'] = hasPermi + Vue.use(install); // eslint-disable-line +} + +export default install diff --git a/dvadmin-ui/src/layout/components/AppMain.vue b/dvadmin-ui/src/layout/components/AppMain.vue new file mode 100755 index 0000000..a897638 --- /dev/null +++ b/dvadmin-ui/src/layout/components/AppMain.vue @@ -0,0 +1,57 @@ + + + + + + + diff --git a/dvadmin-ui/src/layout/components/Navbar.vue b/dvadmin-ui/src/layout/components/Navbar.vue new file mode 100755 index 0000000..b11460a --- /dev/null +++ b/dvadmin-ui/src/layout/components/Navbar.vue @@ -0,0 +1,187 @@ + + + + + diff --git a/dvadmin-ui/src/layout/components/Settings/index.vue b/dvadmin-ui/src/layout/components/Settings/index.vue new file mode 100755 index 0000000..9d42790 --- /dev/null +++ b/dvadmin-ui/src/layout/components/Settings/index.vue @@ -0,0 +1,197 @@ + + + + + diff --git a/dvadmin-ui/src/layout/components/Sidebar/FixiOSBug.js b/dvadmin-ui/src/layout/components/Sidebar/FixiOSBug.js new file mode 100755 index 0000000..6823726 --- /dev/null +++ b/dvadmin-ui/src/layout/components/Sidebar/FixiOSBug.js @@ -0,0 +1,25 @@ +export default { + computed: { + device() { + return this.$store.state.app.device + } + }, + mounted() { + // In order to fix the click on menu on the ios device will trigger the mouseleave bug + this.fixBugIniOS() + }, + methods: { + fixBugIniOS() { + const $subMenu = this.$refs.subMenu + if ($subMenu) { + const handleMouseleave = $subMenu.handleMouseleave + $subMenu.handleMouseleave = (e) => { + if (this.device === 'mobile') { + return + } + handleMouseleave(e) + } + } + } + } +} diff --git a/dvadmin-ui/src/layout/components/Sidebar/Item.vue b/dvadmin-ui/src/layout/components/Sidebar/Item.vue new file mode 100755 index 0000000..b515f61 --- /dev/null +++ b/dvadmin-ui/src/layout/components/Sidebar/Item.vue @@ -0,0 +1,29 @@ + diff --git a/dvadmin-ui/src/layout/components/Sidebar/Link.vue b/dvadmin-ui/src/layout/components/Sidebar/Link.vue new file mode 100755 index 0000000..530b3d5 --- /dev/null +++ b/dvadmin-ui/src/layout/components/Sidebar/Link.vue @@ -0,0 +1,43 @@ + + + diff --git a/dvadmin-ui/src/layout/components/Sidebar/Logo.vue b/dvadmin-ui/src/layout/components/Sidebar/Logo.vue new file mode 100755 index 0000000..57dbd3a --- /dev/null +++ b/dvadmin-ui/src/layout/components/Sidebar/Logo.vue @@ -0,0 +1,93 @@ + + + + + diff --git a/dvadmin-ui/src/layout/components/Sidebar/SidebarItem.vue b/dvadmin-ui/src/layout/components/Sidebar/SidebarItem.vue new file mode 100755 index 0000000..c4febee --- /dev/null +++ b/dvadmin-ui/src/layout/components/Sidebar/SidebarItem.vue @@ -0,0 +1,96 @@ + + + diff --git a/dvadmin-ui/src/layout/components/Sidebar/index.vue b/dvadmin-ui/src/layout/components/Sidebar/index.vue new file mode 100755 index 0000000..9e5d2bb --- /dev/null +++ b/dvadmin-ui/src/layout/components/Sidebar/index.vue @@ -0,0 +1,57 @@ + + + diff --git a/dvadmin-ui/src/layout/components/TagsView/ScrollPane.vue b/dvadmin-ui/src/layout/components/TagsView/ScrollPane.vue new file mode 100755 index 0000000..bb753a1 --- /dev/null +++ b/dvadmin-ui/src/layout/components/TagsView/ScrollPane.vue @@ -0,0 +1,94 @@ + + + + + diff --git a/dvadmin-ui/src/layout/components/TagsView/index.vue b/dvadmin-ui/src/layout/components/TagsView/index.vue new file mode 100755 index 0000000..d1e059e --- /dev/null +++ b/dvadmin-ui/src/layout/components/TagsView/index.vue @@ -0,0 +1,303 @@ + + + + + + + diff --git a/dvadmin-ui/src/layout/components/index.js b/dvadmin-ui/src/layout/components/index.js new file mode 100755 index 0000000..104bd3a --- /dev/null +++ b/dvadmin-ui/src/layout/components/index.js @@ -0,0 +1,5 @@ +export { default as AppMain } from './AppMain' +export { default as Navbar } from './Navbar' +export { default as Settings } from './Settings' +export { default as Sidebar } from './Sidebar/index.vue' +export { default as TagsView } from './TagsView/index.vue' diff --git a/dvadmin-ui/src/layout/index.vue b/dvadmin-ui/src/layout/index.vue new file mode 100755 index 0000000..d490771 --- /dev/null +++ b/dvadmin-ui/src/layout/index.vue @@ -0,0 +1,108 @@ + + + + + diff --git a/dvadmin-ui/src/layout/mixin/ResizeHandler.js b/dvadmin-ui/src/layout/mixin/ResizeHandler.js new file mode 100755 index 0000000..e8d0df8 --- /dev/null +++ b/dvadmin-ui/src/layout/mixin/ResizeHandler.js @@ -0,0 +1,45 @@ +import store from '@/store' + +const { body } = document +const WIDTH = 992 // refer to Bootstrap's responsive design + +export default { + watch: { + $route(route) { + if (this.device === 'mobile' && this.sidebar.opened) { + store.dispatch('app/closeSideBar', { withoutAnimation: false }) + } + } + }, + beforeMount() { + window.addEventListener('resize', this.$_resizeHandler) + }, + beforeDestroy() { + window.removeEventListener('resize', this.$_resizeHandler) + }, + mounted() { + const isMobile = this.$_isMobile() + if (isMobile) { + store.dispatch('app/toggleDevice', 'mobile') + store.dispatch('app/closeSideBar', { withoutAnimation: true }) + } + }, + methods: { + // use $_ for mixins properties + // https://vuejs.org/v2/style-guide/index.html#Private-property-names-essential + $_isMobile() { + const rect = body.getBoundingClientRect() + return rect.width - 1 < WIDTH + }, + $_resizeHandler() { + if (!document.hidden) { + const isMobile = this.$_isMobile() + store.dispatch('app/toggleDevice', isMobile ? 'mobile' : 'desktop') + + if (isMobile) { + store.dispatch('app/closeSideBar', { withoutAnimation: true }) + } + } + } + } +} diff --git a/dvadmin-ui/src/main.js b/dvadmin-ui/src/main.js new file mode 100755 index 0000000..5668d91 --- /dev/null +++ b/dvadmin-ui/src/main.js @@ -0,0 +1,73 @@ +import Vue from 'vue' + +import Cookies from 'js-cookie' + +import Element from 'element-ui' +import './assets/styles/element-variables.scss' + +import '@/assets/styles/index.scss' // global css +import '@/assets/styles/ruoyi.scss' // ruoyi css +import App from './App' +import store from './store' +import router from './router' +import permission from './directive/permission' + +import './assets/icons' // icon +import './permission' // permission control +import { getDicts } from "@/api/system/dict/data"; +import { getConfigKey } from "@/api/system/config"; +import { parseTime, resetForm, addDateRange, selectDictLabel, selectDictLabels, download, handleTree } from "@/utils/ruoyi"; +import Pagination from "@/components/Pagination"; +// 自定义表格工具扩展 +import RightToolbar from "@/components/RightToolbar" + +// 全局方法挂载 +Vue.prototype.getDicts = getDicts +Vue.prototype.getConfigKey = getConfigKey +Vue.prototype.parseTime = parseTime +Vue.prototype.resetForm = resetForm +Vue.prototype.addDateRange = addDateRange +Vue.prototype.selectDictLabel = selectDictLabel +Vue.prototype.selectDictLabels = selectDictLabels +Vue.prototype.download = download +Vue.prototype.handleTree = handleTree + +Vue.prototype.msgSuccess = function (msg) { + this.$message({ showClose: true, message: msg, type: "success" }); +} + +Vue.prototype.msgError = function (msg) { + this.$message({ showClose: true, message: msg, type: "error" }); +} + +Vue.prototype.msgInfo = function (msg) { + this.$message.info(msg); +} + +// 全局组件挂载 +Vue.component('Pagination', Pagination) +Vue.component('RightToolbar', RightToolbar) + +Vue.use(permission) + +/** + * If you don't want to use mock-server + * you want to use MockJs for mock api + * you can execute: mockXHR() + * + * Currently MockJs will be used in the production environment, + * please remove it before going online! ! ! + */ + +Vue.use(Element, { + size: Cookies.get('size') || 'medium' // set element-ui default size +}) + +Vue.config.productionTip = false + +new Vue({ + el: '#app', + router, + store, + render: h => h(App) +}) diff --git a/dvadmin-ui/src/permission.js b/dvadmin-ui/src/permission.js new file mode 100755 index 0000000..6a2ec21 --- /dev/null +++ b/dvadmin-ui/src/permission.js @@ -0,0 +1,54 @@ +import router from './router' +import store from './store' +import { Message } from 'element-ui' +import NProgress from 'nprogress' +import 'nprogress/nprogress.css' +import { getToken } from '@/utils/auth' + +NProgress.configure({ showSpinner: false }) + +const whiteList = ['/login', '/auth-redirect', '/bind', '/register'] + +router.beforeEach((to, from, next) => { + NProgress.start() + if (getToken()) { + /* has token*/ + if (to.path === '/login') { + next({ path: '/' }) + NProgress.done() + } else { + if (store.getters.roles.length === 0) { + // 判断当前用户是否已拉取完user_info信息 + store.dispatch('GetInfo').then(res => { + // 拉取user_info + const roles = res.roles + store.dispatch('GenerateRoutes', { roles }).then(accessRoutes => { + // 根据roles权限生成可访问的路由表 + router.addRoutes(accessRoutes) // 动态添加可访问路由表 + next({ ...to, replace: true }) // hack方法 确保addRoutes已完成 + }) + }).catch(err => { + store.dispatch('LogOut').then(() => { + Message.error(err) + next({ path: '/' }) + }) + }) + } else { + next() + } + } + } else { + // 没有token + if (whiteList.indexOf(to.path) !== -1) { + // 在免登录白名单,直接进入 + next() + } else { + next(`/login?redirect=${to.fullPath}`) // 否则全部重定向到登录页 + NProgress.done() + } + } +}) + +router.afterEach(() => { + NProgress.done() +}) diff --git a/dvadmin-ui/src/router/index.js b/dvadmin-ui/src/router/index.js new file mode 100755 index 0000000..1cef875 --- /dev/null +++ b/dvadmin-ui/src/router/index.js @@ -0,0 +1,140 @@ +import Vue from 'vue' +import Router from 'vue-router' + +Vue.use(Router) + +/* Layout */ +import Layout from '@/layout' +import ParentView from '@/components/ParentView'; + +/** + * Note: 路由配置项 + * + * hidden: true // 当设置 true 的时候该路由不会再侧边栏出现 如401,login等页面,或者如一些编辑页面/edit/1 + * alwaysShow: true // 当你一个路由下面的 children 声明的路由大于1个时,自动会变成嵌套的模式--如组件页面 + * // 只有一个时,会将那个子路由当做根路由显示在侧边栏--如引导页面 + * // 若你想不管路由下面的 children 声明的个数都显示你的根路由 + * // 你可以设置 alwaysShow: true,这样它就会忽略之前定义的规则,一直显示根路由 + * redirect: noRedirect // 当设置 noRedirect 的时候该路由在面包屑导航中不可被点击 + * name:'router-name' // 设定路由的名字,一定要填写不然使用时会出现各种问题 + * meta : { + noCache: true // 如果设置为true,则不会被 缓存(默认 false) + title: 'title' // 设置该路由在侧边栏和面包屑中展示的名字 + icon: 'svg-name' // 设置该路由的图标,对应路径src/assets/icons/svg + breadcrumb: false // 如果设置为false,则不会在breadcrumb面包屑中显示 + } + */ + +// 公共路由 +export const constantRoutes = [ + { + path: '/redirect', + component: Layout, + hidden: true, + children: [ + { + path: '/redirect/:path(.*)', + component: (resolve) => require(['@/views/redirect'], resolve) + } + ] + }, + { + path: '/login', + component: (resolve) => require(['@/views/login'], resolve), + hidden: true + }, + { + path: '/404', + component: (resolve) => require(['@/views/error/404'], resolve), + hidden: true + }, + { + path: '/401', + component: (resolve) => require(['@/views/error/401'], resolve), + hidden: true + }, + { + path: '', + component: Layout, + redirect: 'index', + children: [ + { + path: 'index', + component: (resolve) => require(['@/views/index'], resolve), + name: '首页', + meta: { title: '首页', icon: 'dashboard', noCache: true, affix: true } + } + ] + }, + { + path: '/user', + component: Layout, + hidden: true, + redirect: 'noredirect', + children: [ + { + path: 'profile', + component: (resolve) => require(['@/views/system/user/profile/index'], resolve), + name: 'Profile', + meta: { title: '个人中心', icon: 'user' } + } + ] + }, + { + path: '/user', + component: Layout, + children: [ + { + path: 'users', + component: (resolve) => require(['@/views/system/user/users/index'], resolve), + name: 'Profile', + meta: { title: '个人测试', icon: 'user' } + } + ] + }, + { + path: '/dict', + component: Layout, + hidden: true, + children: [ + { + path: 'type/data/:dictId(\\d+)', + component: (resolve) => require(['@/views/system/dict/data'], resolve), + name: 'Data', + meta: { title: '字典数据', icon: '' } + } + ] + }, + { + path: '/job', + component: Layout, + hidden: true, + children: [ + { + path: 'log', + component: (resolve) => require(['@/views/monitor/job/log'], resolve), + name: 'JobLog', + meta: { title: '调度日志' } + } + ] + }, + { + path: '/gen', + component: Layout, + hidden: true, + children: [ + { + path: 'edit/:tableId(\\d+)', + component: (resolve) => require(['@/views/tool/gen/editTable'], resolve), + name: 'GenEdit', + meta: { title: '修改生成配置' } + } + ] + } +] + +export default new Router({ + mode: 'history', // 去掉url中的# + scrollBehavior: () => ({ y: 0 }), + routes: constantRoutes +}) diff --git a/dvadmin-ui/src/settings.js b/dvadmin-ui/src/settings.js new file mode 100755 index 0000000..40a7f15 --- /dev/null +++ b/dvadmin-ui/src/settings.js @@ -0,0 +1,36 @@ +module.exports = { + title: '若依管理系统', + + /** + * 侧边栏主题 深色主题theme-dark,浅色主题theme-light + */ + sideTheme: 'theme-dark', + + /** + * 是否系统布局配置 + */ + showSettings: false, + + /** + * 是否显示 tagsView + */ + tagsView: true, + + /** + * 是否固定头部 + */ + fixedHeader: false, + + /** + * 是否显示logo + */ + sidebarLogo: true, + + /** + * @type {string | array} 'production' | ['production', 'development'] + * @description Need show err logs component. + * The default is only used in the production env + * If you want to also use it in dev, you can pass ['production', 'development'] + */ + errorLog: 'production' +} diff --git a/dvadmin-ui/src/store/getters.js b/dvadmin-ui/src/store/getters.js new file mode 100755 index 0000000..00c4ebf --- /dev/null +++ b/dvadmin-ui/src/store/getters.js @@ -0,0 +1,16 @@ +const getters = { + sidebar: state => state.app.sidebar, + size: state => state.app.size, + device: state => state.app.device, + visitedViews: state => state.tagsView.visitedViews, + cachedViews: state => state.tagsView.cachedViews, + token: state => state.user.token, + avatar: state => state.user.avatar, + name: state => state.user.name, + introduction: state => state.user.introduction, + roles: state => state.user.roles, + permissions: state => state.user.permissions, + permission_routes: state => state.permission.routes, + sidebarRouters:state => state.permission.sidebarRouters, +} +export default getters diff --git a/dvadmin-ui/src/store/index.js b/dvadmin-ui/src/store/index.js new file mode 100755 index 0000000..53b8437 --- /dev/null +++ b/dvadmin-ui/src/store/index.js @@ -0,0 +1,23 @@ +import Vue from 'vue' +import Vuex from 'vuex' +import app from './modules/app' +import user from './modules/user' +import tagsView from './modules/tagsView' +import permission from './modules/permission' +import settings from './modules/settings' +import getters from './getters' + +Vue.use(Vuex) + +const store = new Vuex.Store({ + modules: { + app, + user, + tagsView, + permission, + settings + }, + getters +}) + +export default store diff --git a/dvadmin-ui/src/store/modules/app.js b/dvadmin-ui/src/store/modules/app.js new file mode 100755 index 0000000..45d89bb --- /dev/null +++ b/dvadmin-ui/src/store/modules/app.js @@ -0,0 +1,56 @@ +import Cookies from 'js-cookie' + +const state = { + sidebar: { + opened: Cookies.get('sidebarStatus') ? !!+Cookies.get('sidebarStatus') : true, + withoutAnimation: false + }, + device: 'desktop', + size: Cookies.get('size') || 'medium' +} + +const mutations = { + TOGGLE_SIDEBAR: state => { + state.sidebar.opened = !state.sidebar.opened + state.sidebar.withoutAnimation = false + if (state.sidebar.opened) { + Cookies.set('sidebarStatus', 1) + } else { + Cookies.set('sidebarStatus', 0) + } + }, + CLOSE_SIDEBAR: (state, withoutAnimation) => { + Cookies.set('sidebarStatus', 0) + state.sidebar.opened = false + state.sidebar.withoutAnimation = withoutAnimation + }, + TOGGLE_DEVICE: (state, device) => { + state.device = device + }, + SET_SIZE: (state, size) => { + state.size = size + Cookies.set('size', size) + } +} + +const actions = { + toggleSideBar({ commit }) { + commit('TOGGLE_SIDEBAR') + }, + closeSideBar({ commit }, { withoutAnimation }) { + commit('CLOSE_SIDEBAR', withoutAnimation) + }, + toggleDevice({ commit }, device) { + commit('TOGGLE_DEVICE', device) + }, + setSize({ commit }, size) { + commit('SET_SIZE', size) + } +} + +export default { + namespaced: true, + state, + mutations, + actions +} diff --git a/dvadmin-ui/src/store/modules/permission.js b/dvadmin-ui/src/store/modules/permission.js new file mode 100755 index 0000000..80015b0 --- /dev/null +++ b/dvadmin-ui/src/store/modules/permission.js @@ -0,0 +1,95 @@ +import { constantRoutes } from '@/router' +import { getRouters } from '@/api/menu' +import Layout from '@/layout/index' +import ParentView from '@/components/ParentView'; + +const permission = { + state: { + routes: [], + addRoutes: [], + sidebarRouters: [] + }, + mutations: { + SET_ROUTES: (state, routes) => { + state.addRoutes = routes + state.routes = constantRoutes.concat(routes) + }, + SET_SIDEBAR_ROUTERS: (state, routers) => { + state.sidebarRouters = constantRoutes.concat(routers) + }, + }, + actions: { + // 生成路由 + GenerateRoutes({ commit }) { + return new Promise(resolve => { + // 向后端请求路由数据 + getRouters().then(res => { + const sdata = JSON.parse(JSON.stringify(res.data)) + const rdata = JSON.parse(JSON.stringify(res.data)) + const sidebarRoutes = filterAsyncRouter(sdata) + const rewriteRoutes = filterAsyncRouter(rdata, false, true) + rewriteRoutes.push({ path: '*', redirect: '/404', hidden: true }) + commit('SET_ROUTES', rewriteRoutes) + commit('SET_SIDEBAR_ROUTERS', sidebarRoutes) + resolve(rewriteRoutes) + }) + }) + } + } +} + +// 遍历后台传来的路由字符串,转换为组件对象 +function filterAsyncRouter(asyncRouterMap, lastRouter = false, type = false) { + return asyncRouterMap.filter(route => { + if (type && route.children) { + route.children = filterChildren(route.children) + } + if (route.component) { + // Layout ParentView 组件特殊处理 + if (route.component === 'Layout') { + route.component = Layout + } else if (route.component === 'ParentView') { + route.component = ParentView + } else { + route.component = loadView(route.component) + } + } + if (route.children != null && route.children && route.children.length) { + route.children = filterAsyncRouter(route.children, route, type) + } else { + delete route['children'] + delete route['redirect'] + } + return true + }) +} + +function filterChildren(childrenMap, lastRouter = false) { + var children = [] + childrenMap.forEach((el, index) => { + if (el.children && el.children.length) { + if (el.component === 'ParentView') { + el.children.forEach(c => { + c.path = el.path + '/' + c.path + if (c.children && c.children.length) { + children = children.concat(filterChildren(c.children, c)) + return + } + children.push(c) + }) + return + } + } + if (lastRouter) { + el.path = lastRouter.path + '/' + el.path + } + children = children.concat(el) + }) + return children +} + +export const loadView = (view) => { // 路由懒加载 + return (resolve) => require([`@/views/${view}`], resolve) +} + +export default permission diff --git a/dvadmin-ui/src/store/modules/settings.js b/dvadmin-ui/src/store/modules/settings.js new file mode 100755 index 0000000..8bd81a3 --- /dev/null +++ b/dvadmin-ui/src/store/modules/settings.js @@ -0,0 +1,35 @@ +import variables from '@/assets/styles/element-variables.scss' +import defaultSettings from '@/settings' + +const { sideTheme, showSettings, tagsView, fixedHeader, sidebarLogo } = defaultSettings + +const state = { + theme: variables.theme, + sideTheme: sideTheme, + showSettings: showSettings, + tagsView: tagsView, + fixedHeader: fixedHeader, + sidebarLogo: sidebarLogo +} + +const mutations = { + CHANGE_SETTING: (state, { key, value }) => { + if (state.hasOwnProperty(key)) { + state[key] = value + } + } +} + +const actions = { + changeSetting({ commit }, data) { + commit('CHANGE_SETTING', data) + } +} + +export default { + namespaced: true, + state, + mutations, + actions +} + diff --git a/dvadmin-ui/src/store/modules/tagsView.js b/dvadmin-ui/src/store/modules/tagsView.js new file mode 100755 index 0000000..2e30d15 --- /dev/null +++ b/dvadmin-ui/src/store/modules/tagsView.js @@ -0,0 +1,159 @@ +const state = { + visitedViews: [], + cachedViews: [] +} + +const mutations = { + ADD_VISITED_VIEW: (state, view) => { + if (state.visitedViews.some(v => v.path === view.path)) return + state.visitedViews.push( + Object.assign({}, view, { + title: view.meta.title || 'no-name' + }) + ) + }, + ADD_CACHED_VIEW: (state, view) => { + if (state.cachedViews.includes(view.name)) return + if (!view.meta.noCache) { + state.cachedViews.push(view.name) + } + }, + + DEL_VISITED_VIEW: (state, view) => { + for (const [i, v] of state.visitedViews.entries()) { + if (v.path === view.path) { + state.visitedViews.splice(i, 1) + break + } + } + }, + DEL_CACHED_VIEW: (state, view) => { + const index = state.cachedViews.indexOf(view.name) + index > -1 && state.cachedViews.splice(index, 1) + }, + + DEL_OTHERS_VISITED_VIEWS: (state, view) => { + state.visitedViews = state.visitedViews.filter(v => { + return v.meta.affix || v.path === view.path + }) + }, + DEL_OTHERS_CACHED_VIEWS: (state, view) => { + const index = state.cachedViews.indexOf(view.name) + if (index > -1) { + state.cachedViews = state.cachedViews.slice(index, index + 1) + } else { + state.cachedViews = [] + } + }, + + DEL_ALL_VISITED_VIEWS: state => { + // keep affix tags + const affixTags = state.visitedViews.filter(tag => tag.meta.affix) + state.visitedViews = affixTags + }, + DEL_ALL_CACHED_VIEWS: state => { + state.cachedViews = [] + }, + + UPDATE_VISITED_VIEW: (state, view) => { + for (let v of state.visitedViews) { + if (v.path === view.path) { + v = Object.assign(v, view) + break + } + } + } +} + +const actions = { + addView({ dispatch }, view) { + dispatch('addVisitedView', view) + dispatch('addCachedView', view) + }, + addVisitedView({ commit }, view) { + commit('ADD_VISITED_VIEW', view) + }, + addCachedView({ commit }, view) { + commit('ADD_CACHED_VIEW', view) + }, + + delView({ dispatch, state }, view) { + return new Promise(resolve => { + dispatch('delVisitedView', view) + dispatch('delCachedView', view) + resolve({ + visitedViews: [...state.visitedViews], + cachedViews: [...state.cachedViews] + }) + }) + }, + delVisitedView({ commit, state }, view) { + return new Promise(resolve => { + commit('DEL_VISITED_VIEW', view) + resolve([...state.visitedViews]) + }) + }, + delCachedView({ commit, state }, view) { + return new Promise(resolve => { + commit('DEL_CACHED_VIEW', view) + resolve([...state.cachedViews]) + }) + }, + + delOthersViews({ dispatch, state }, view) { + return new Promise(resolve => { + dispatch('delOthersVisitedViews', view) + dispatch('delOthersCachedViews', view) + resolve({ + visitedViews: [...state.visitedViews], + cachedViews: [...state.cachedViews] + }) + }) + }, + delOthersVisitedViews({ commit, state }, view) { + return new Promise(resolve => { + commit('DEL_OTHERS_VISITED_VIEWS', view) + resolve([...state.visitedViews]) + }) + }, + delOthersCachedViews({ commit, state }, view) { + return new Promise(resolve => { + commit('DEL_OTHERS_CACHED_VIEWS', view) + resolve([...state.cachedViews]) + }) + }, + + delAllViews({ dispatch, state }, view) { + return new Promise(resolve => { + dispatch('delAllVisitedViews', view) + dispatch('delAllCachedViews', view) + resolve({ + visitedViews: [...state.visitedViews], + cachedViews: [...state.cachedViews] + }) + }) + }, + delAllVisitedViews({ commit, state }) { + return new Promise(resolve => { + commit('DEL_ALL_VISITED_VIEWS') + resolve([...state.visitedViews]) + }) + }, + delAllCachedViews({ commit, state }) { + return new Promise(resolve => { + commit('DEL_ALL_CACHED_VIEWS') + resolve([...state.cachedViews]) + }) + }, + + updateVisitedView({ commit }, view) { + commit('UPDATE_VISITED_VIEW', view) + } +} + +export default { + namespaced: true, + state, + mutations, + actions +} diff --git a/dvadmin-ui/src/store/modules/user.js b/dvadmin-ui/src/store/modules/user.js new file mode 100755 index 0000000..6c1f8d5 --- /dev/null +++ b/dvadmin-ui/src/store/modules/user.js @@ -0,0 +1,96 @@ +import { login, logout, getInfo } from '@/api/login' +import { getToken, setToken, removeToken } from '@/utils/auth' + +const user = { + state: { + token: getToken(), + name: '', + avatar: '', + roles: [], + permissions: [] + }, + + mutations: { + SET_TOKEN: (state, token) => { + state.token = token + }, + SET_NAME: (state, name) => { + state.name = name + }, + SET_AVATAR: (state, avatar) => { + state.avatar = avatar + }, + SET_ROLES: (state, roles) => { + state.roles = roles + }, + SET_PERMISSIONS: (state, permissions) => { + state.permissions = permissions + } + }, + + actions: { + // 登录 + Login({ commit }, userInfo) { + const username = userInfo.username.trim() + const password = userInfo.password + const code = userInfo.code + const uuid = userInfo.uuid + return new Promise((resolve, reject) => { + login(username, password, code, uuid).then(res => { + setToken(res.data.token) + commit('SET_TOKEN', res.data.token) + resolve() + }).catch(error => { + reject(error) + }) + }) + }, + + // 获取用户信息 + GetInfo({ commit, state }) { + return new Promise((resolve, reject) => { + getInfo(state.token).then(res => { + const user = res.user + const avatar = user.avatar == "" ? require("@/assets/images/profile.jpg") : process.env.VUE_APP_BASE_API + user.avatar; + if (res.roles && res.roles.length > 0) { // 验证返回的roles是否是一个非空数组 + commit('SET_ROLES', res.roles) + commit('SET_PERMISSIONS', res.permissions) + } else { + commit('SET_ROLES', ['ROLE_DEFAULT']) + } + commit('SET_NAME', user.userName) + commit('SET_AVATAR', avatar) + resolve(res) + }).catch(error => { + reject(error) + }) + }) + }, + + // 退出系统 + LogOut({ commit, state }) { + return new Promise((resolve, reject) => { + logout(state.token).then(() => { + commit('SET_TOKEN', '') + commit('SET_ROLES', []) + commit('SET_PERMISSIONS', []) + removeToken() + resolve() + }).catch(error => { + reject(error) + }) + }) + }, + + // 前端 登出 + FedLogOut({ commit }) { + return new Promise(resolve => { + commit('SET_TOKEN', '') + removeToken() + resolve() + }) + } + } +} + +export default user diff --git a/dvadmin-ui/src/utils/auth.js b/dvadmin-ui/src/utils/auth.js new file mode 100755 index 0000000..08a43d6 --- /dev/null +++ b/dvadmin-ui/src/utils/auth.js @@ -0,0 +1,15 @@ +import Cookies from 'js-cookie' + +const TokenKey = 'Admin-Token' + +export function getToken() { + return Cookies.get(TokenKey) +} + +export function setToken(token) { + return Cookies.set(TokenKey, token) +} + +export function removeToken() { + return Cookies.remove(TokenKey) +} diff --git a/dvadmin-ui/src/utils/errorCode.js b/dvadmin-ui/src/utils/errorCode.js new file mode 100755 index 0000000..d2111ee --- /dev/null +++ b/dvadmin-ui/src/utils/errorCode.js @@ -0,0 +1,6 @@ +export default { + '401': '认证失败,无法访问系统资源', + '403': '当前操作没有权限', + '404': '访问资源不存在', + 'default': '系统未知错误,请反馈给管理员' +} diff --git a/dvadmin-ui/src/utils/generator/config.js b/dvadmin-ui/src/utils/generator/config.js new file mode 100755 index 0000000..7abf227 --- /dev/null +++ b/dvadmin-ui/src/utils/generator/config.js @@ -0,0 +1,438 @@ +export const formConf = { + formRef: 'elForm', + formModel: 'formData', + size: 'medium', + labelPosition: 'right', + labelWidth: 100, + formRules: 'rules', + gutter: 15, + disabled: false, + span: 24, + formBtns: true +} + +export const inputComponents = [ + { + label: '单行文本', + tag: 'el-input', + tagIcon: 'input', + placeholder: '请输入', + defaultValue: undefined, + span: 24, + labelWidth: null, + style: { width: '100%' }, + clearable: true, + prepend: '', + append: '', + 'prefix-icon': '', + 'suffix-icon': '', + maxlength: null, + 'show-word-limit': false, + readonly: false, + disabled: false, + required: true, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/input' + }, + { + label: '多行文本', + tag: 'el-input', + tagIcon: 'textarea', + type: 'textarea', + placeholder: '请输入', + defaultValue: undefined, + span: 24, + labelWidth: null, + autosize: { + minRows: 4, + maxRows: 4 + }, + style: { width: '100%' }, + maxlength: null, + 'show-word-limit': false, + readonly: false, + disabled: false, + required: true, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/input' + }, + { + label: '密码', + tag: 'el-input', + tagIcon: 'password', + placeholder: '请输入', + defaultValue: undefined, + span: 24, + 'show-password': true, + labelWidth: null, + style: { width: '100%' }, + clearable: true, + prepend: '', + append: '', + 'prefix-icon': '', + 'suffix-icon': '', + maxlength: null, + 'show-word-limit': false, + readonly: false, + disabled: false, + required: true, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/input' + }, + { + label: '计数器', + tag: 'el-input-number', + tagIcon: 'number', + placeholder: '', + defaultValue: undefined, + span: 24, + labelWidth: null, + min: undefined, + max: undefined, + step: undefined, + 'step-strictly': false, + precision: undefined, + 'controls-position': '', + disabled: false, + required: true, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/input-number' + } +] + +export const selectComponents = [ + { + label: '下拉选择', + tag: 'el-select', + tagIcon: 'select', + placeholder: '请选择', + defaultValue: undefined, + span: 24, + labelWidth: null, + style: { width: '100%' }, + clearable: true, + disabled: false, + required: true, + filterable: false, + multiple: false, + options: [{ + label: '选项一', + value: 1 + }, { + label: '选项二', + value: 2 + }], + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/select' + }, + { + label: '级联选择', + tag: 'el-cascader', + tagIcon: 'cascader', + placeholder: '请选择', + defaultValue: [], + span: 24, + labelWidth: null, + style: { width: '100%' }, + props: { + props: { + multiple: false + } + }, + 'show-all-levels': true, + disabled: false, + clearable: true, + filterable: false, + required: true, + options: [{ + id: 1, + value: 1, + label: '选项1', + children: [{ + id: 2, + value: 2, + label: '选项1-1' + }] + }], + dataType: 'dynamic', + labelKey: 'label', + valueKey: 'value', + childrenKey: 'children', + separator: '/', + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/cascader' + }, + { + label: '单选框组', + tag: 'el-radio-group', + tagIcon: 'radio', + defaultValue: undefined, + span: 24, + labelWidth: null, + style: {}, + optionType: 'default', + border: false, + size: 'medium', + disabled: false, + required: true, + options: [{ + label: '选项一', + value: 1 + }, { + label: '选项二', + value: 2 + }], + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/radio' + }, + { + label: '多选框组', + tag: 'el-checkbox-group', + tagIcon: 'checkbox', + defaultValue: [], + span: 24, + labelWidth: null, + style: {}, + optionType: 'default', + border: false, + size: 'medium', + disabled: false, + required: true, + options: [{ + label: '选项一', + value: 1 + }, { + label: '选项二', + value: 2 + }], + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/checkbox' + }, + { + label: '开关', + tag: 'el-switch', + tagIcon: 'switch', + defaultValue: false, + span: 24, + labelWidth: null, + style: {}, + disabled: false, + required: true, + 'active-text': '', + 'inactive-text': '', + 'active-color': null, + 'inactive-color': null, + 'active-value': true, + 'inactive-value': false, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/switch' + }, + { + label: '滑块', + tag: 'el-slider', + tagIcon: 'slider', + defaultValue: null, + span: 24, + labelWidth: null, + disabled: false, + required: true, + min: 0, + max: 100, + step: 1, + 'show-stops': false, + range: false, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/slider' + }, + { + label: '时间选择', + tag: 'el-time-picker', + tagIcon: 'time', + placeholder: '请选择', + defaultValue: null, + span: 24, + labelWidth: null, + style: { width: '100%' }, + disabled: false, + clearable: true, + required: true, + 'picker-options': { + selectableRange: '00:00:00-23:59:59' + }, + format: 'HH:mm:ss', + 'value-format': 'HH:mm:ss', + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/time-picker' + }, + { + label: '时间范围', + tag: 'el-time-picker', + tagIcon: 'time-range', + defaultValue: null, + span: 24, + labelWidth: null, + style: { width: '100%' }, + disabled: false, + clearable: true, + required: true, + 'is-range': true, + 'range-separator': '至', + 'start-placeholder': '开始时间', + 'end-placeholder': '结束时间', + format: 'HH:mm:ss', + 'value-format': 'HH:mm:ss', + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/time-picker' + }, + { + label: '日期选择', + tag: 'el-date-picker', + tagIcon: 'date', + placeholder: '请选择', + defaultValue: null, + type: 'date', + span: 24, + labelWidth: null, + style: { width: '100%' }, + disabled: false, + clearable: true, + required: true, + format: 'yyyy-MM-dd', + 'value-format': 'yyyy-MM-dd', + readonly: false, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/date-picker' + }, + { + label: '日期范围', + tag: 'el-date-picker', + tagIcon: 'date-range', + defaultValue: null, + span: 24, + labelWidth: null, + style: { width: '100%' }, + type: 'daterange', + 'range-separator': '至', + 'start-placeholder': '开始日期', + 'end-placeholder': '结束日期', + disabled: false, + clearable: true, + required: true, + format: 'yyyy-MM-dd', + 'value-format': 'yyyy-MM-dd', + readonly: false, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/date-picker' + }, + { + label: '评分', + tag: 'el-rate', + tagIcon: 'rate', + defaultValue: 0, + span: 24, + labelWidth: null, + style: {}, + max: 5, + 'allow-half': false, + 'show-text': false, + 'show-score': false, + disabled: false, + required: true, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/rate' + }, + { + label: '颜色选择', + tag: 'el-color-picker', + tagIcon: 'color', + defaultValue: null, + labelWidth: null, + 'show-alpha': false, + 'color-format': '', + disabled: false, + required: true, + size: 'medium', + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/color-picker' + }, + { + label: '上传', + tag: 'el-upload', + tagIcon: 'upload', + action: 'https://jsonplaceholder.typicode.com/posts/', + defaultValue: null, + labelWidth: null, + disabled: false, + required: true, + accept: '', + name: 'file', + 'auto-upload': true, + showTip: false, + buttonText: '点击上传', + fileSize: 2, + sizeUnit: 'MB', + 'list-type': 'text', + multiple: false, + regList: [], + changeTag: true, + document: 'https://element.eleme.cn/#/zh-CN/component/upload' + } +] + +export const layoutComponents = [ + { + layout: 'rowFormItem', + tagIcon: 'row', + type: 'default', + justify: 'start', + align: 'top', + label: '行容器', + layoutTree: true, + children: [], + document: 'https://element.eleme.cn/#/zh-CN/component/layout' + }, + { + layout: 'colFormItem', + label: '按钮', + changeTag: true, + labelWidth: null, + tag: 'el-button', + tagIcon: 'button', + span: 24, + default: '主要按钮', + type: 'primary', + icon: 'el-icon-search', + size: 'medium', + disabled: false, + document: 'https://element.eleme.cn/#/zh-CN/component/button' + } +] + +// 组件rule的触发方式,无触发方式的组件不生成rule +export const trigger = { + 'el-input': 'blur', + 'el-input-number': 'blur', + 'el-select': 'change', + 'el-radio-group': 'change', + 'el-checkbox-group': 'change', + 'el-cascader': 'change', + 'el-time-picker': 'change', + 'el-date-picker': 'change', + 'el-rate': 'change' +} diff --git a/dvadmin-ui/src/utils/generator/css.js b/dvadmin-ui/src/utils/generator/css.js new file mode 100755 index 0000000..c1c62e6 --- /dev/null +++ b/dvadmin-ui/src/utils/generator/css.js @@ -0,0 +1,18 @@ +const styles = { + 'el-rate': '.el-rate{display: inline-block; vertical-align: text-top;}', + 'el-upload': '.el-upload__tip{line-height: 1.2;}' +} + +function addCss(cssList, el) { + const css = styles[el.tag] + css && cssList.indexOf(css) === -1 && cssList.push(css) + if (el.children) { + el.children.forEach(el2 => addCss(cssList, el2)) + } +} + +export function makeUpCss(conf) { + const cssList = [] + conf.fields.forEach(el => addCss(cssList, el)) + return cssList.join('\n') +} diff --git a/dvadmin-ui/src/utils/generator/drawingDefalut.js b/dvadmin-ui/src/utils/generator/drawingDefalut.js new file mode 100755 index 0000000..09f133c --- /dev/null +++ b/dvadmin-ui/src/utils/generator/drawingDefalut.js @@ -0,0 +1,29 @@ +export default [ + { + layout: 'colFormItem', + tagIcon: 'input', + label: '手机号', + vModel: 'mobile', + formId: 6, + tag: 'el-input', + placeholder: '请输入手机号', + defaultValue: '', + span: 24, + style: { width: '100%' }, + clearable: true, + prepend: '', + append: '', + 'prefix-icon': 'el-icon-mobile', + 'suffix-icon': '', + maxlength: 11, + 'show-word-limit': true, + readonly: false, + disabled: false, + required: true, + changeTag: true, + regList: [{ + pattern: '/^1(3|4|5|7|8|9)\\d{9}$/', + message: '手机号格式错误' + }] + } +] diff --git a/dvadmin-ui/src/utils/generator/html.js b/dvadmin-ui/src/utils/generator/html.js new file mode 100755 index 0000000..ebf628d --- /dev/null +++ b/dvadmin-ui/src/utils/generator/html.js @@ -0,0 +1,359 @@ +/* eslint-disable max-len */ +import { trigger } from './config' + +let confGlobal +let someSpanIsNot24 + +export function dialogWrapper(str) { + return ` + ${str} +
+ 取消 + 确定 +
+
` +} + +export function vueTemplate(str) { + return `` +} + +export function vueScript(str) { + return `` +} + +export function cssStyle(cssStr) { + return `` +} + +function buildFormTemplate(conf, child, type) { + let labelPosition = '' + if (conf.labelPosition !== 'right') { + labelPosition = `label-position="${conf.labelPosition}"` + } + const disabled = conf.disabled ? `:disabled="${conf.disabled}"` : '' + let str = ` + ${child} + ${buildFromBtns(conf, type)} + ` + if (someSpanIsNot24) { + str = ` + ${str} + ` + } + return str +} + +function buildFromBtns(conf, type) { + let str = '' + if (conf.formBtns && type === 'file') { + str = ` + 提交 + 重置 + ` + if (someSpanIsNot24) { + str = ` + ${str} + ` + } + } + return str +} + +// span不为24的用el-col包裹 +function colWrapper(element, str) { + if (someSpanIsNot24 || element.span !== 24) { + return ` + ${str} + ` + } + return str +} + +const layouts = { + colFormItem(element) { + let labelWidth = '' + if (element.labelWidth && element.labelWidth !== confGlobal.labelWidth) { + labelWidth = `label-width="${element.labelWidth}px"` + } + const required = !trigger[element.tag] && element.required ? 'required' : '' + const tagDom = tags[element.tag] ? tags[element.tag](element) : null + let str = ` + ${tagDom} + ` + str = colWrapper(element, str) + return str + }, + rowFormItem(element) { + const type = element.type === 'default' ? '' : `type="${element.type}"` + const justify = element.type === 'default' ? '' : `justify="${element.justify}"` + const align = element.type === 'default' ? '' : `align="${element.align}"` + const gutter = element.gutter ? `gutter="${element.gutter}"` : '' + const children = element.children.map(el => layouts[el.layout](el)) + let str = ` + ${children.join('\n')} + ` + str = colWrapper(element, str) + return str + } +} + +const tags = { + 'el-button': el => { + const { + tag, disabled + } = attrBuilder(el) + const type = el.type ? `type="${el.type}"` : '' + const icon = el.icon ? `icon="${el.icon}"` : '' + const size = el.size ? `size="${el.size}"` : '' + let child = buildElButtonChild(el) + + if (child) child = `\n${child}\n` // 换行 + return `<${el.tag} ${type} ${icon} ${size} ${disabled}>${child}` + }, + 'el-input': el => { + const { + disabled, vModel, clearable, placeholder, width + } = attrBuilder(el) + const maxlength = el.maxlength ? `:maxlength="${el.maxlength}"` : '' + const showWordLimit = el['show-word-limit'] ? 'show-word-limit' : '' + const readonly = el.readonly ? 'readonly' : '' + const prefixIcon = el['prefix-icon'] ? `prefix-icon='${el['prefix-icon']}'` : '' + const suffixIcon = el['suffix-icon'] ? `suffix-icon='${el['suffix-icon']}'` : '' + const showPassword = el['show-password'] ? 'show-password' : '' + const type = el.type ? `type="${el.type}"` : '' + const autosize = el.autosize && el.autosize.minRows + ? `:autosize="{minRows: ${el.autosize.minRows}, maxRows: ${el.autosize.maxRows}}"` + : '' + let child = buildElInputChild(el) + + if (child) child = `\n${child}\n` // 换行 + return `<${el.tag} ${vModel} ${type} ${placeholder} ${maxlength} ${showWordLimit} ${readonly} ${disabled} ${clearable} ${prefixIcon} ${suffixIcon} ${showPassword} ${autosize} ${width}>${child}` + }, + 'el-input-number': el => { + const { disabled, vModel, placeholder } = attrBuilder(el) + const controlsPosition = el['controls-position'] ? `controls-position=${el['controls-position']}` : '' + const min = el.min ? `:min='${el.min}'` : '' + const max = el.max ? `:max='${el.max}'` : '' + const step = el.step ? `:step='${el.step}'` : '' + const stepStrictly = el['step-strictly'] ? 'step-strictly' : '' + const precision = el.precision ? `:precision='${el.precision}'` : '' + + return `<${el.tag} ${vModel} ${placeholder} ${step} ${stepStrictly} ${precision} ${controlsPosition} ${min} ${max} ${disabled}>` + }, + 'el-select': el => { + const { + disabled, vModel, clearable, placeholder, width + } = attrBuilder(el) + const filterable = el.filterable ? 'filterable' : '' + const multiple = el.multiple ? 'multiple' : '' + let child = buildElSelectChild(el) + + if (child) child = `\n${child}\n` // 换行 + return `<${el.tag} ${vModel} ${placeholder} ${disabled} ${multiple} ${filterable} ${clearable} ${width}>${child}` + }, + 'el-radio-group': el => { + const { disabled, vModel } = attrBuilder(el) + const size = `size="${el.size}"` + let child = buildElRadioGroupChild(el) + + if (child) child = `\n${child}\n` // 换行 + return `<${el.tag} ${vModel} ${size} ${disabled}>${child}` + }, + 'el-checkbox-group': el => { + const { disabled, vModel } = attrBuilder(el) + const size = `size="${el.size}"` + const min = el.min ? `:min="${el.min}"` : '' + const max = el.max ? `:max="${el.max}"` : '' + let child = buildElCheckboxGroupChild(el) + + if (child) child = `\n${child}\n` // 换行 + return `<${el.tag} ${vModel} ${min} ${max} ${size} ${disabled}>${child}` + }, + 'el-switch': el => { + const { disabled, vModel } = attrBuilder(el) + const activeText = el['active-text'] ? `active-text="${el['active-text']}"` : '' + const inactiveText = el['inactive-text'] ? `inactive-text="${el['inactive-text']}"` : '' + const activeColor = el['active-color'] ? `active-color="${el['active-color']}"` : '' + const inactiveColor = el['inactive-color'] ? `inactive-color="${el['inactive-color']}"` : '' + const activeValue = el['active-value'] !== true ? `:active-value='${JSON.stringify(el['active-value'])}'` : '' + const inactiveValue = el['inactive-value'] !== false ? `:inactive-value='${JSON.stringify(el['inactive-value'])}'` : '' + + return `<${el.tag} ${vModel} ${activeText} ${inactiveText} ${activeColor} ${inactiveColor} ${activeValue} ${inactiveValue} ${disabled}>` + }, + 'el-cascader': el => { + const { + disabled, vModel, clearable, placeholder, width + } = attrBuilder(el) + const options = el.options ? `:options="${el.vModel}Options"` : '' + const props = el.props ? `:props="${el.vModel}Props"` : '' + const showAllLevels = el['show-all-levels'] ? '' : ':show-all-levels="false"' + const filterable = el.filterable ? 'filterable' : '' + const separator = el.separator === '/' ? '' : `separator="${el.separator}"` + + return `<${el.tag} ${vModel} ${options} ${props} ${width} ${showAllLevels} ${placeholder} ${separator} ${filterable} ${clearable} ${disabled}>` + }, + 'el-slider': el => { + const { disabled, vModel } = attrBuilder(el) + const min = el.min ? `:min='${el.min}'` : '' + const max = el.max ? `:max='${el.max}'` : '' + const step = el.step ? `:step='${el.step}'` : '' + const range = el.range ? 'range' : '' + const showStops = el['show-stops'] ? `:show-stops="${el['show-stops']}"` : '' + + return `<${el.tag} ${min} ${max} ${step} ${vModel} ${range} ${showStops} ${disabled}>` + }, + 'el-time-picker': el => { + const { + disabled, vModel, clearable, placeholder, width + } = attrBuilder(el) + const startPlaceholder = el['start-placeholder'] ? `start-placeholder="${el['start-placeholder']}"` : '' + const endPlaceholder = el['end-placeholder'] ? `end-placeholder="${el['end-placeholder']}"` : '' + const rangeSeparator = el['range-separator'] ? `range-separator="${el['range-separator']}"` : '' + const isRange = el['is-range'] ? 'is-range' : '' + const format = el.format ? `format="${el.format}"` : '' + const valueFormat = el['value-format'] ? `value-format="${el['value-format']}"` : '' + const pickerOptions = el['picker-options'] ? `:picker-options='${JSON.stringify(el['picker-options'])}'` : '' + + return `<${el.tag} ${vModel} ${isRange} ${format} ${valueFormat} ${pickerOptions} ${width} ${placeholder} ${startPlaceholder} ${endPlaceholder} ${rangeSeparator} ${clearable} ${disabled}>` + }, + 'el-date-picker': el => { + const { + disabled, vModel, clearable, placeholder, width + } = attrBuilder(el) + const startPlaceholder = el['start-placeholder'] ? `start-placeholder="${el['start-placeholder']}"` : '' + const endPlaceholder = el['end-placeholder'] ? `end-placeholder="${el['end-placeholder']}"` : '' + const rangeSeparator = el['range-separator'] ? `range-separator="${el['range-separator']}"` : '' + const format = el.format ? `format="${el.format}"` : '' + const valueFormat = el['value-format'] ? `value-format="${el['value-format']}"` : '' + const type = el.type === 'date' ? '' : `type="${el.type}"` + const readonly = el.readonly ? 'readonly' : '' + + return `<${el.tag} ${type} ${vModel} ${format} ${valueFormat} ${width} ${placeholder} ${startPlaceholder} ${endPlaceholder} ${rangeSeparator} ${clearable} ${readonly} ${disabled}>` + }, + 'el-rate': el => { + const { disabled, vModel } = attrBuilder(el) + const max = el.max ? `:max='${el.max}'` : '' + const allowHalf = el['allow-half'] ? 'allow-half' : '' + const showText = el['show-text'] ? 'show-text' : '' + const showScore = el['show-score'] ? 'show-score' : '' + + return `<${el.tag} ${vModel} ${allowHalf} ${showText} ${showScore} ${disabled}>` + }, + 'el-color-picker': el => { + const { disabled, vModel } = attrBuilder(el) + const size = `size="${el.size}"` + const showAlpha = el['show-alpha'] ? 'show-alpha' : '' + const colorFormat = el['color-format'] ? `color-format="${el['color-format']}"` : '' + + return `<${el.tag} ${vModel} ${size} ${showAlpha} ${colorFormat} ${disabled}>` + }, + 'el-upload': el => { + const disabled = el.disabled ? ':disabled=\'true\'' : '' + const action = el.action ? `:action="${el.vModel}Action"` : '' + const multiple = el.multiple ? 'multiple' : '' + const listType = el['list-type'] !== 'text' ? `list-type="${el['list-type']}"` : '' + const accept = el.accept ? `accept="${el.accept}"` : '' + const name = el.name !== 'file' ? `name="${el.name}"` : '' + const autoUpload = el['auto-upload'] === false ? ':auto-upload="false"' : '' + const beforeUpload = `:before-upload="${el.vModel}BeforeUpload"` + const fileList = `:file-list="${el.vModel}fileList"` + const ref = `ref="${el.vModel}"` + let child = buildElUploadChild(el) + + if (child) child = `\n${child}\n` // 换行 + return `<${el.tag} ${ref} ${fileList} ${action} ${autoUpload} ${multiple} ${beforeUpload} ${listType} ${accept} ${name} ${disabled}>${child}` + } +} + +function attrBuilder(el) { + return { + vModel: `v-model="${confGlobal.formModel}.${el.vModel}"`, + clearable: el.clearable ? 'clearable' : '', + placeholder: el.placeholder ? `placeholder="${el.placeholder}"` : '', + width: el.style && el.style.width ? ':style="{width: \'100%\'}"' : '', + disabled: el.disabled ? ':disabled=\'true\'' : '' + } +} + +// el-buttin 子级 +function buildElButtonChild(conf) { + const children = [] + if (conf.default) { + children.push(conf.default) + } + return children.join('\n') +} + +// el-input innerHTML +function buildElInputChild(conf) { + const children = [] + if (conf.prepend) { + children.push(``) + } + if (conf.append) { + children.push(``) + } + return children.join('\n') +} + +function buildElSelectChild(conf) { + const children = [] + if (conf.options && conf.options.length) { + children.push(``) + } + return children.join('\n') +} + +function buildElRadioGroupChild(conf) { + const children = [] + if (conf.options && conf.options.length) { + const tag = conf.optionType === 'button' ? 'el-radio-button' : 'el-radio' + const border = conf.border ? 'border' : '' + children.push(`<${tag} v-for="(item, index) in ${conf.vModel}Options" :key="index" :label="item.value" :disabled="item.disabled" ${border}>{{item.label}}`) + } + return children.join('\n') +} + +function buildElCheckboxGroupChild(conf) { + const children = [] + if (conf.options && conf.options.length) { + const tag = conf.optionType === 'button' ? 'el-checkbox-button' : 'el-checkbox' + const border = conf.border ? 'border' : '' + children.push(`<${tag} v-for="(item, index) in ${conf.vModel}Options" :key="index" :label="item.value" :disabled="item.disabled" ${border}>{{item.label}}`) + } + return children.join('\n') +} + +function buildElUploadChild(conf) { + const list = [] + if (conf['list-type'] === 'picture-card') list.push('') + else list.push(`${conf.buttonText}`) + if (conf.showTip) list.push(`
只能上传不超过 ${conf.fileSize}${conf.sizeUnit} 的${conf.accept}文件
`) + return list.join('\n') +} + +export function makeUpHtml(conf, type) { + const htmlList = [] + confGlobal = conf + someSpanIsNot24 = conf.fields.some(item => item.span !== 24) + conf.fields.forEach(el => { + htmlList.push(layouts[el.layout](el)) + }) + const htmlStr = htmlList.join('\n') + + let temp = buildFormTemplate(conf, htmlStr, type) + if (type === 'dialog') { + temp = dialogWrapper(temp) + } + confGlobal = null + return temp +} diff --git a/dvadmin-ui/src/utils/generator/icon.json b/dvadmin-ui/src/utils/generator/icon.json new file mode 100755 index 0000000..2d9999a --- /dev/null +++ b/dvadmin-ui/src/utils/generator/icon.json @@ -0,0 +1 @@ +["platform-eleme","eleme","delete-solid","delete","s-tools","setting","user-solid","user","phone","phone-outline","more","more-outline","star-on","star-off","s-goods","goods","warning","warning-outline","question","info","remove","circle-plus","success","error","zoom-in","zoom-out","remove-outline","circle-plus-outline","circle-check","circle-close","s-help","help","minus","plus","check","close","picture","picture-outline","picture-outline-round","upload","upload2","download","camera-solid","camera","video-camera-solid","video-camera","message-solid","bell","s-cooperation","s-order","s-platform","s-fold","s-unfold","s-operation","s-promotion","s-home","s-release","s-ticket","s-management","s-open","s-shop","s-marketing","s-flag","s-comment","s-finance","s-claim","s-custom","s-opportunity","s-data","s-check","s-grid","menu","share","d-caret","caret-left","caret-right","caret-bottom","caret-top","bottom-left","bottom-right","back","right","bottom","top","top-left","top-right","arrow-left","arrow-right","arrow-down","arrow-up","d-arrow-left","d-arrow-right","video-pause","video-play","refresh","refresh-right","refresh-left","finished","sort","sort-up","sort-down","rank","loading","view","c-scale-to-original","date","edit","edit-outline","folder","folder-opened","folder-add","folder-remove","folder-delete","folder-checked","tickets","document-remove","document-delete","document-copy","document-checked","document","document-add","printer","paperclip","takeaway-box","search","monitor","attract","mobile","scissors","umbrella","headset","brush","mouse","coordinate","magic-stick","reading","data-line","data-board","pie-chart","data-analysis","collection-tag","film","suitcase","suitcase-1","receiving","collection","files","notebook-1","notebook-2","toilet-paper","office-building","school","table-lamp","house","no-smoking","smoking","shopping-cart-full","shopping-cart-1","shopping-cart-2","shopping-bag-1","shopping-bag-2","sold-out","sell","present","box","bank-card","money","coin","wallet","discount","price-tag","news","guide","male","female","thumb","cpu","link","connection","open","turn-off","set-up","chat-round","chat-line-round","chat-square","chat-dot-round","chat-dot-square","chat-line-square","message","postcard","position","turn-off-microphone","microphone","close-notification","bangzhu","time","odometer","crop","aim","switch-button","full-screen","copy-document","mic","stopwatch","medal-1","medal","trophy","trophy-1","first-aid-kit","discover","place","location","location-outline","location-information","add-location","delete-location","map-location","alarm-clock","timer","watch-1","watch","lock","unlock","key","service","mobile-phone","bicycle","truck","ship","basketball","football","soccer","baseball","wind-power","light-rain","lightning","heavy-rain","sunrise","sunrise-1","sunset","sunny","cloudy","partly-cloudy","cloudy-and-sunny","moon","moon-night","dish","dish-1","food","chicken","fork-spoon","knife-fork","burger","tableware","sugar","dessert","ice-cream","hot-water","water-cup","coffee-cup","cold-drink","goblet","goblet-full","goblet-square","goblet-square-full","refrigerator","grape","watermelon","cherry","apple","pear","orange","coffee","ice-tea","ice-drink","milk-tea","potato-strips","lollipop","ice-cream-square","ice-cream-round"] \ No newline at end of file diff --git a/dvadmin-ui/src/utils/generator/js.js b/dvadmin-ui/src/utils/generator/js.js new file mode 100755 index 0000000..c6c06a9 --- /dev/null +++ b/dvadmin-ui/src/utils/generator/js.js @@ -0,0 +1,236 @@ +import { isArray } from 'util' +import { exportDefault, titleCase } from '@/utils/index' +import { trigger } from './config' + +const units = { + KB: '1024', + MB: '1024 / 1024', + GB: '1024 / 1024 / 1024' +} +let confGlobal +const inheritAttrs = { + file: '', + dialog: 'inheritAttrs: false,' +} + + +export function makeUpJs(conf, type) { + confGlobal = conf = JSON.parse(JSON.stringify(conf)) + const dataList = [] + const ruleList = [] + const optionsList = [] + const propsList = [] + const methodList = mixinMethod(type) + const uploadVarList = [] + + conf.fields.forEach(el => { + buildAttributes(el, dataList, ruleList, optionsList, methodList, propsList, uploadVarList) + }) + + const script = buildexport( + conf, + type, + dataList.join('\n'), + ruleList.join('\n'), + optionsList.join('\n'), + uploadVarList.join('\n'), + propsList.join('\n'), + methodList.join('\n') + ) + confGlobal = null + return script +} + +function buildAttributes(el, dataList, ruleList, optionsList, methodList, propsList, uploadVarList) { + buildData(el, dataList) + buildRules(el, ruleList) + + if (el.options && el.options.length) { + buildOptions(el, optionsList) + if (el.dataType === 'dynamic') { + const model = `${el.vModel}Options` + const options = titleCase(model) + buildOptionMethod(`get${options}`, model, methodList) + } + } + + if (el.props && el.props.props) { + buildProps(el, propsList) + } + + if (el.action && el.tag === 'el-upload') { + uploadVarList.push( + `${el.vModel}Action: '${el.action}', + ${el.vModel}fileList: [],` + ) + methodList.push(buildBeforeUpload(el)) + if (!el['auto-upload']) { + methodList.push(buildSubmitUpload(el)) + } + } + + if (el.children) { + el.children.forEach(el2 => { + buildAttributes(el2, dataList, ruleList, optionsList, methodList, propsList, uploadVarList) + }) + } +} + +function mixinMethod(type) { + const list = []; const + minxins = { + file: confGlobal.formBtns ? { + submitForm: `submitForm() { + this.$refs['${confGlobal.formRef}'].validate(valid => { + if(!valid) return + // TODO 提交表单 + }) + },`, + resetForm: `resetForm() { + this.$refs['${confGlobal.formRef}'].resetFields() + },` + } : null, + dialog: { + onOpen: 'onOpen() {},', + onClose: `onClose() { + this.$refs['${confGlobal.formRef}'].resetFields() + },`, + close: `close() { + this.$emit('update:visible', false) + },`, + handelConfirm: `handelConfirm() { + this.$refs['${confGlobal.formRef}'].validate(valid => { + if(!valid) return + this.close() + }) + },` + } + } + + const methods = minxins[type] + if (methods) { + Object.keys(methods).forEach(key => { + list.push(methods[key]) + }) + } + + return list +} + +function buildData(conf, dataList) { + if (conf.vModel === undefined) return + let defaultValue + if (typeof (conf.defaultValue) === 'string' && !conf.multiple) { + defaultValue = `'${conf.defaultValue}'` + } else { + defaultValue = `${JSON.stringify(conf.defaultValue)}` + } + dataList.push(`${conf.vModel}: ${defaultValue},`) +} + +function buildRules(conf, ruleList) { + if (conf.vModel === undefined) return + const rules = [] + if (trigger[conf.tag]) { + if (conf.required) { + const type = isArray(conf.defaultValue) ? 'type: \'array\',' : '' + let message = isArray(conf.defaultValue) ? `请至少选择一个${conf.vModel}` : conf.placeholder + if (message === undefined) message = `${conf.label}不能为空` + rules.push(`{ required: true, ${type} message: '${message}', trigger: '${trigger[conf.tag]}' }`) + } + if (conf.regList && isArray(conf.regList)) { + conf.regList.forEach(item => { + if (item.pattern) { + rules.push(`{ pattern: ${eval(item.pattern)}, message: '${item.message}', trigger: '${trigger[conf.tag]}' }`) + } + }) + } + ruleList.push(`${conf.vModel}: [${rules.join(',')}],`) + } +} + +function buildOptions(conf, optionsList) { + if (conf.vModel === undefined) return + if (conf.dataType === 'dynamic') { conf.options = [] } + const str = `${conf.vModel}Options: ${JSON.stringify(conf.options)},` + optionsList.push(str) +} + +function buildProps(conf, propsList) { + if (conf.dataType === 'dynamic') { + conf.valueKey !== 'value' && (conf.props.props.value = conf.valueKey) + conf.labelKey !== 'label' && (conf.props.props.label = conf.labelKey) + conf.childrenKey !== 'children' && (conf.props.props.children = conf.childrenKey) + } + const str = `${conf.vModel}Props: ${JSON.stringify(conf.props.props)},` + propsList.push(str) +} + +function buildBeforeUpload(conf) { + const unitNum = units[conf.sizeUnit]; let rightSizeCode = ''; let acceptCode = ''; const + returnList = [] + if (conf.fileSize) { + rightSizeCode = `let isRightSize = file.size / ${unitNum} < ${conf.fileSize} + if(!isRightSize){ + this.$message.error('文件大小超过 ${conf.fileSize}${conf.sizeUnit}') + }` + returnList.push('isRightSize') + } + if (conf.accept) { + acceptCode = `let isAccept = new RegExp('${conf.accept}').test(file.type) + if(!isAccept){ + this.$message.error('应该选择${conf.accept}类型的文件') + }` + returnList.push('isAccept') + } + const str = `${conf.vModel}BeforeUpload(file) { + ${rightSizeCode} + ${acceptCode} + return ${returnList.join('&&')} + },` + return returnList.length ? str : '' +} + +function buildSubmitUpload(conf) { + const str = `submitUpload() { + this.$refs['${conf.vModel}'].submit() + },` + return str +} + +function buildOptionMethod(methodName, model, methodList) { + const str = `${methodName}() { + // TODO 发起请求获取数据 + this.${model} + },` + methodList.push(str) +} + +function buildexport(conf, type, data, rules, selectOptions, uploadVar, props, methods) { + const str = `${exportDefault}{ + ${inheritAttrs[type]} + components: {}, + props: [], + data () { + return { + ${conf.formModel}: { + ${data} + }, + ${conf.formRules}: { + ${rules} + }, + ${uploadVar} + ${selectOptions} + ${props} + } + }, + computed: {}, + watch: {}, + created () {}, + mounted () {}, + methods: { + ${methods} + } +}` + return str +} diff --git a/dvadmin-ui/src/utils/generator/render.js b/dvadmin-ui/src/utils/generator/render.js new file mode 100755 index 0000000..e8640f0 --- /dev/null +++ b/dvadmin-ui/src/utils/generator/render.js @@ -0,0 +1,126 @@ +import { makeMap } from '@/utils/index' + +// 参考https://github.com/vuejs/vue/blob/v2.6.10/src/platforms/web/server/util.js +const isAttr = makeMap( + 'accept,accept-charset,accesskey,action,align,alt,async,autocomplete,' + + 'autofocus,autoplay,autosave,bgcolor,border,buffered,challenge,charset,' + + 'checked,cite,class,code,codebase,color,cols,colspan,content,http-equiv,' + + 'name,contenteditable,contextmenu,controls,coords,data,datetime,default,' + + 'defer,dir,dirname,disabled,download,draggable,dropzone,enctype,method,for,' + + 'form,formaction,headers,height,hidden,high,href,hreflang,http-equiv,' + + 'icon,id,ismap,itemprop,keytype,kind,label,lang,language,list,loop,low,' + + 'manifest,max,maxlength,media,method,GET,POST,min,multiple,email,file,' + + 'muted,name,novalidate,open,optimum,pattern,ping,placeholder,poster,' + + 'preload,radiogroup,readonly,rel,required,reversed,rows,rowspan,sandbox,' + + 'scope,scoped,seamless,selected,shape,size,type,text,password,sizes,span,' + + 'spellcheck,src,srcdoc,srclang,srcset,start,step,style,summary,tabindex,' + + 'target,title,type,usemap,value,width,wrap' +) + +function vModel(self, dataObject, defaultValue) { + dataObject.props.value = defaultValue + + dataObject.on.input = val => { + self.$emit('input', val) + } +} + +const componentChild = { + 'el-button': { + default(h, conf, key) { + return conf[key] + }, + }, + 'el-input': { + prepend(h, conf, key) { + return + }, + append(h, conf, key) { + return + } + }, + 'el-select': { + options(h, conf, key) { + const list = [] + conf.options.forEach(item => { + list.push() + }) + return list + } + }, + 'el-radio-group': { + options(h, conf, key) { + const list = [] + conf.options.forEach(item => { + if (conf.optionType === 'button') list.push({item.label}) + else list.push({item.label}) + }) + return list + } + }, + 'el-checkbox-group': { + options(h, conf, key) { + const list = [] + conf.options.forEach(item => { + if (conf.optionType === 'button') { + list.push({item.label}) + } else { + list.push({item.label}) + } + }) + return list + } + }, + 'el-upload': { + 'list-type': (h, conf, key) => { + const list = [] + if (conf['list-type'] === 'picture-card') { + list.push() + } else { + list.push({conf.buttonText}) + } + if (conf.showTip) { + list.push(
只能上传不超过 {conf.fileSize}{conf.sizeUnit} 的{conf.accept}文件
) + } + return list + } + } +} + +export default { + render(h) { + const dataObject = { + attrs: {}, + props: {}, + on: {}, + style: {} + } + const confClone = JSON.parse(JSON.stringify(this.conf)) + const children = [] + + const childObjs = componentChild[confClone.tag] + if (childObjs) { + Object.keys(childObjs).forEach(key => { + const childFunc = childObjs[key] + if (confClone[key]) { + children.push(childFunc(h, confClone, key)) + } + }) + } + + Object.keys(confClone).forEach(key => { + const val = confClone[key] + if (key === 'vModel') { + vModel(this, dataObject, confClone.defaultValue) + } else if (dataObject[key]) { + dataObject[key] = val + } else if (!isAttr(key)) { + dataObject.props[key] = val + } else { + dataObject.attrs[key] = val + } + }) + return h(this.conf.tag, dataObject, children) + }, + props: ['conf'] +} diff --git a/dvadmin-ui/src/utils/index.js b/dvadmin-ui/src/utils/index.js new file mode 100755 index 0000000..918580f --- /dev/null +++ b/dvadmin-ui/src/utils/index.js @@ -0,0 +1,390 @@ +import { parseTime } from './ruoyi' + +/** + * 表格时间格式化 + */ +export function formatDate(cellValue) { + if (cellValue == null || cellValue == "") return ""; + var date = new Date(cellValue) + var year = date.getFullYear() + var month = date.getMonth() + 1 < 10 ? '0' + (date.getMonth() + 1) : date.getMonth() + 1 + var day = date.getDate() < 10 ? '0' + date.getDate() : date.getDate() + var hours = date.getHours() < 10 ? '0' + date.getHours() : date.getHours() + var minutes = date.getMinutes() < 10 ? '0' + date.getMinutes() : date.getMinutes() + var seconds = date.getSeconds() < 10 ? '0' + date.getSeconds() : date.getSeconds() + return year + '-' + month + '-' + day + ' ' + hours + ':' + minutes + ':' + seconds +} + +/** + * @param {number} time + * @param {string} option + * @returns {string} + */ +export function formatTime(time, option) { + if (('' + time).length === 10) { + time = parseInt(time) * 1000 + } else { + time = +time + } + const d = new Date(time) + const now = Date.now() + + const diff = (now - d) / 1000 + + if (diff < 30) { + return '刚刚' + } else if (diff < 3600) { + // less 1 hour + return Math.ceil(diff / 60) + '分钟前' + } else if (diff < 3600 * 24) { + return Math.ceil(diff / 3600) + '小时前' + } else if (diff < 3600 * 24 * 2) { + return '1天前' + } + if (option) { + return parseTime(time, option) + } else { + return ( + d.getMonth() + + 1 + + '月' + + d.getDate() + + '日' + + d.getHours() + + '时' + + d.getMinutes() + + '分' + ) + } +} + +/** + * @param {string} url + * @returns {Object} + */ +export function getQueryObject(url) { + url = url == null ? window.location.href : url + const search = url.substring(url.lastIndexOf('?') + 1) + const obj = {} + const reg = /([^?&=]+)=([^?&=]*)/g + search.replace(reg, (rs, $1, $2) => { + const name = decodeURIComponent($1) + let val = decodeURIComponent($2) + val = String(val) + obj[name] = val + return rs + }) + return obj +} + +/** + * @param {string} input value + * @returns {number} output value + */ +export function byteLength(str) { + // returns the byte length of an utf8 string + let s = str.length + for (var i = str.length - 1; i >= 0; i--) { + const code = str.charCodeAt(i) + if (code > 0x7f && code <= 0x7ff) s++ + else if (code > 0x7ff && code <= 0xffff) s += 2 + if (code >= 0xDC00 && code <= 0xDFFF) i-- + } + return s +} + +/** + * @param {Array} actual + * @returns {Array} + */ +export function cleanArray(actual) { + const newArray = [] + for (let i = 0; i < actual.length; i++) { + if (actual[i]) { + newArray.push(actual[i]) + } + } + return newArray +} + +/** + * @param {Object} json + * @returns {Array} + */ +export function param(json) { + if (!json) return '' + return cleanArray( + Object.keys(json).map(key => { + if (json[key] === undefined) return '' + return encodeURIComponent(key) + '=' + encodeURIComponent(json[key]) + }) + ).join('&') +} + +/** + * @param {string} url + * @returns {Object} + */ +export function param2Obj(url) { + const search = decodeURIComponent(url.split('?')[1]).replace(/\+/g, ' ') + if (!search) { + return {} + } + const obj = {} + const searchArr = search.split('&') + searchArr.forEach(v => { + const index = v.indexOf('=') + if (index !== -1) { + const name = v.substring(0, index) + const val = v.substring(index + 1, v.length) + obj[name] = val + } + }) + return obj +} + +/** + * @param {string} val + * @returns {string} + */ +export function html2Text(val) { + const div = document.createElement('div') + div.innerHTML = val + return div.textContent || div.innerText +} + +/** + * Merges two objects, giving the last one precedence + * @param {Object} target + * @param {(Object|Array)} source + * @returns {Object} + */ +export function objectMerge(target, source) { + if (typeof target !== 'object') { + target = {} + } + if (Array.isArray(source)) { + return source.slice() + } + Object.keys(source).forEach(property => { + const sourceProperty = source[property] + if (typeof sourceProperty === 'object') { + target[property] = objectMerge(target[property], sourceProperty) + } else { + target[property] = sourceProperty + } + }) + return target +} + +/** + * @param {HTMLElement} element + * @param {string} className + */ +export function toggleClass(element, className) { + if (!element || !className) { + return + } + let classString = element.className + const nameIndex = classString.indexOf(className) + if (nameIndex === -1) { + classString += '' + className + } else { + classString = + classString.substr(0, nameIndex) + + classString.substr(nameIndex + className.length) + } + element.className = classString +} + +/** + * @param {string} type + * @returns {Date} + */ +export function getTime(type) { + if (type === 'start') { + return new Date().getTime() - 3600 * 1000 * 24 * 90 + } else { + return new Date(new Date().toDateString()) + } +} + +/** + * @param {Function} func + * @param {number} wait + * @param {boolean} immediate + * @return {*} + */ +export function debounce(func, wait, immediate) { + let timeout, args, context, timestamp, result + + const later = function() { + // 据上一次触发时间间隔 + const last = +new Date() - timestamp + + // 上次被包装函数被调用时间间隔 last 小于设定时间间隔 wait + if (last < wait && last > 0) { + timeout = setTimeout(later, wait - last) + } else { + timeout = null + // 如果设定为immediate===true,因为开始边界已经调用过了此处无需调用 + if (!immediate) { + result = func.apply(context, args) + if (!timeout) context = args = null + } + } + } + + return function(...args) { + context = this + timestamp = +new Date() + const callNow = immediate && !timeout + // 如果延时不存在,重新设定延时 + if (!timeout) timeout = setTimeout(later, wait) + if (callNow) { + result = func.apply(context, args) + context = args = null + } + + return result + } +} + +/** + * This is just a simple version of deep copy + * Has a lot of edge cases bug + * If you want to use a perfect deep copy, use lodash's _.cloneDeep + * @param {Object} source + * @returns {Object} + */ +export function deepClone(source) { + if (!source && typeof source !== 'object') { + throw new Error('error arguments', 'deepClone') + } + const targetObj = source.constructor === Array ? [] : {} + Object.keys(source).forEach(keys => { + if (source[keys] && typeof source[keys] === 'object') { + targetObj[keys] = deepClone(source[keys]) + } else { + targetObj[keys] = source[keys] + } + }) + return targetObj +} + +/** + * @param {Array} arr + * @returns {Array} + */ +export function uniqueArr(arr) { + return Array.from(new Set(arr)) +} + +/** + * @returns {string} + */ +export function createUniqueString() { + const timestamp = +new Date() + '' + const randomNum = parseInt((1 + Math.random()) * 65536) + '' + return (+(randomNum + timestamp)).toString(32) +} + +/** + * Check if an element has a class + * @param {HTMLElement} elm + * @param {string} cls + * @returns {boolean} + */ +export function hasClass(ele, cls) { + return !!ele.className.match(new RegExp('(\\s|^)' + cls + '(\\s|$)')) +} + +/** + * Add class to element + * @param {HTMLElement} elm + * @param {string} cls + */ +export function addClass(ele, cls) { + if (!hasClass(ele, cls)) ele.className += ' ' + cls +} + +/** + * Remove class from element + * @param {HTMLElement} elm + * @param {string} cls + */ +export function removeClass(ele, cls) { + if (hasClass(ele, cls)) { + const reg = new RegExp('(\\s|^)' + cls + '(\\s|$)') + ele.className = ele.className.replace(reg, ' ') + } +} + +export function makeMap(str, expectsLowerCase) { + const map = Object.create(null) + const list = str.split(',') + for (let i = 0; i < list.length; i++) { + map[list[i]] = true + } + return expectsLowerCase + ? val => map[val.toLowerCase()] + : val => map[val] +} + +export const exportDefault = 'export default ' + +export const beautifierConf = { + html: { + indent_size: '2', + indent_char: ' ', + max_preserve_newlines: '-1', + preserve_newlines: false, + keep_array_indentation: false, + break_chained_methods: false, + indent_scripts: 'separate', + brace_style: 'end-expand', + space_before_conditional: true, + unescape_strings: false, + jslint_happy: false, + end_with_newline: true, + wrap_line_length: '110', + indent_inner_html: true, + comma_first: false, + e4x: true, + indent_empty_lines: true + }, + js: { + indent_size: '2', + indent_char: ' ', + max_preserve_newlines: '-1', + preserve_newlines: false, + keep_array_indentation: false, + break_chained_methods: false, + indent_scripts: 'normal', + brace_style: 'end-expand', + space_before_conditional: true, + unescape_strings: false, + jslint_happy: true, + end_with_newline: true, + wrap_line_length: '110', + indent_inner_html: true, + comma_first: false, + e4x: true, + indent_empty_lines: true + } +} + +// 首字母大小 +export function titleCase(str) { + return str.replace(/( |^)[a-z]/g, L => L.toUpperCase()) +} + +// 下划转驼峰 +export function camelCase(str) { + return str.replace(/-[a-z]/g, str1 => str1.substr(-1).toUpperCase()) +} + +export function isNumberStr(str) { + return /^[+-]?(0|([1-9]\d*))(\.\d+)?$/g.test(str) +} + diff --git a/dvadmin-ui/src/utils/jsencrypt.js b/dvadmin-ui/src/utils/jsencrypt.js new file mode 100755 index 0000000..78d9523 --- /dev/null +++ b/dvadmin-ui/src/utils/jsencrypt.js @@ -0,0 +1,30 @@ +import JSEncrypt from 'jsencrypt/bin/jsencrypt.min' + +// 密钥对生成 http://web.chacuo.net/netrsakeypair + +const publicKey = 'MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAKoR8mX0rGKLqzcWmOzbfj64K8ZIgOdH\n' + + 'nzkXSOVOZbFu/TJhZ7rFAN+eaGkl3C4buccQd/EjEsj9ir7ijT7h96MCAwEAAQ==' + +const privateKey = 'MIIBVAIBADANBgkqhkiG9w0BAQEFAASCAT4wggE6AgEAAkEAqhHyZfSsYourNxaY\n' + + '7Nt+PrgrxkiA50efORdI5U5lsW79MmFnusUA355oaSXcLhu5xxB38SMSyP2KvuKN\n' + + 'PuH3owIDAQABAkAfoiLyL+Z4lf4Myxk6xUDgLaWGximj20CUf+5BKKnlrK+Ed8gA\n' + + 'kM0HqoTt2UZwA5E2MzS4EI2gjfQhz5X28uqxAiEA3wNFxfrCZlSZHb0gn2zDpWow\n' + + 'cSxQAgiCstxGUoOqlW8CIQDDOerGKH5OmCJ4Z21v+F25WaHYPxCFMvwxpcw99Ecv\n' + + 'DQIgIdhDTIqD2jfYjPTY8Jj3EDGPbH2HHuffvflECt3Ek60CIQCFRlCkHpi7hthh\n' + + 'YhovyloRYsM+IS9h/0BzlEAuO0ktMQIgSPT3aFAgJYwKpqRYKlLDVcflZFCKY7u3\n' + + 'UP8iWi1Qw0Y=' + +// 加密 +export function encrypt(txt) { + const encryptor = new JSEncrypt() + encryptor.setPublicKey(publicKey) // 设置公钥 + return encryptor.encrypt(txt) // 对数据进行加密 +} + +// 解密 +export function decrypt(txt) { + const encryptor = new JSEncrypt() + encryptor.setPrivateKey(privateKey) // 设置私钥 + return encryptor.decrypt(txt) // 对数据进行解密 +} + diff --git a/dvadmin-ui/src/utils/permission.js b/dvadmin-ui/src/utils/permission.js new file mode 100755 index 0000000..1730e33 --- /dev/null +++ b/dvadmin-ui/src/utils/permission.js @@ -0,0 +1,51 @@ +import store from '@/store' + +/** + * 字符权限校验 + * @param {Array} value 校验值 + * @returns {Boolean} + */ +export function checkPermi(value) { + if (value && value instanceof Array && value.length > 0) { + const permissions = store.getters && store.getters.permissions + const permissionDatas = value + const all_permission = "*:*:*"; + + const hasPermission = permissions.some(permission => { + return all_permission === permission || permissionDatas.includes(permission) + }) + + if (!hasPermission) { + return false + } + return true + } else { + console.error(`need roles! Like checkPermi="['system:user:add','system:user:edit']"`) + return false + } +} + +/** + * 角色权限校验 + * @param {Array} value 校验值 + * @returns {Boolean} + */ +export function checkRole(value) { + if (value && value instanceof Array && value.length > 0) { + const roles = store.getters && store.getters.roles + const permissionRoles = value + const super_admin = "admin"; + + const hasRole = roles.some(role => { + return super_admin === role || permissionRoles.includes(role) + }) + + if (!hasRole) { + return false + } + return true + } else { + console.error(`need roles! Like checkRole="['admin','editor']"`) + return false + } +} \ No newline at end of file diff --git a/dvadmin-ui/src/utils/request.js b/dvadmin-ui/src/utils/request.js new file mode 100755 index 0000000..ae89f24 --- /dev/null +++ b/dvadmin-ui/src/utils/request.js @@ -0,0 +1,103 @@ +import axios from 'axios' +import { Notification, MessageBox, Message } from 'element-ui' +import store from '@/store' +import { getToken } from '@/utils/auth' +import errorCode from '@/utils/errorCode' + +axios.defaults.headers['Content-Type'] = 'application/json;charset=utf-8' +// 创建axios实例 +const service = axios.create({ + // axios中请求配置有baseURL选项,表示请求URL公共部分 + baseURL: process.env.VUE_APP_BASE_API, + // 超时 + timeout: 10000 +}) +// request拦截器 +service.interceptors.request.use(config => { + // 是否需要设置 token + const isToken = (config.headers || {}).isToken === false + if (getToken() && !isToken) { + config.headers['Authorization'] = 'Bearer ' + getToken() // 让每个请求携带自定义token 请根据实际情况自行修改 + } + // get请求映射params参数 + if (config.method === 'get' && config.params) { + let url = config.url + '?'; + for (const propName of Object.keys(config.params)) { + const value = config.params[propName]; + var part = encodeURIComponent(propName) + "="; + if (value !== null && typeof(value) !== "undefined") { + if (typeof value === 'object') { + for (const key of Object.keys(value)) { + let params = propName + '[' + key + ']'; + var subPart = encodeURIComponent(params) + "="; + url += subPart + encodeURIComponent(value[key]) + "&"; + } + } else { + url += part + encodeURIComponent(value) + "&"; + } + } + } + url = url.slice(0, -1); + config.params = {}; + config.url = url; + } + return config +}, error => { + console.log(error) + Promise.reject(error) +}) + +// 响应拦截器 +service.interceptors.response.use(res => { + // 未设置状态码则默认成功状态 + const code = res.data.code || 200; + // 获取错误信息 + const msg = errorCode[code] || res.data.msg || errorCode['default'] + if (code === 401) { + MessageBox.confirm('登录状态已过期,您可以继续留在该页面,或者重新登录', '系统提示', { + confirmButtonText: '重新登录', + cancelButtonText: '取消', + type: 'warning' + } + ).then(() => { + store.dispatch('LogOut').then(() => { + location.href = '/index'; + }) + }) + } else if (code === 500) { + Message({ + message: msg, + type: 'error' + }) + return Promise.reject(new Error(msg)) + } else if (code !== 200) { + Notification.error({ + title: msg + }) + return Promise.reject('error') + } else { + return res.data + } + }, + error => { + console.log('err' + error) + let { message } = error; + if (message == "Network Error") { + message = "后端接口连接异常"; + } + else if (message.includes("timeout")) { + message = "系统接口请求超时"; + } + else if (message.includes("Request failed with status code")) { + message = "系统接口" + message.substr(message.length - 3) + "异常"; + } + Message({ + message: message, + type: 'error', + duration: 5 * 1000 + }) + return Promise.reject(error) + } +) + +export default service diff --git a/dvadmin-ui/src/utils/ruoyi.js b/dvadmin-ui/src/utils/ruoyi.js new file mode 100755 index 0000000..81ee9c7 --- /dev/null +++ b/dvadmin-ui/src/utils/ruoyi.js @@ -0,0 +1,152 @@ +/** + * 通用js方法封装处理 + * Copyright (c) 2019 ruoyi + */ + +const baseURL = process.env.VUE_APP_BASE_API + +// 日期格式化 +export function parseTime(time, pattern) { + if (arguments.length === 0 || !time) { + return null + } + const format = pattern || '{y}-{m}-{d} {h}:{i}:{s}' + let date + if (typeof time === 'object') { + date = time + } else { + if ((typeof time === 'string') && (/^[0-9]+$/.test(time))) { + time = parseInt(time) + } else if (typeof time === 'string') { + time = time.replace(new RegExp(/-/gm), '/'); + } + if ((typeof time === 'number') && (time.toString().length === 10)) { + time = time * 1000 + } + date = new Date(time) + } + const formatObj = { + y: date.getFullYear(), + m: date.getMonth() + 1, + d: date.getDate(), + h: date.getHours(), + i: date.getMinutes(), + s: date.getSeconds(), + a: date.getDay() + } + const time_str = format.replace(/{(y|m|d|h|i|s|a)+}/g, (result, key) => { + let value = formatObj[key] + // Note: getDay() returns 0 on Sunday + if (key === 'a') { return ['日', '一', '二', '三', '四', '五', '六'][value] } + if (result.length > 0 && value < 10) { + value = '0' + value + } + return value || 0 + }) + return time_str +} + +// 表单重置 +export function resetForm(refName) { + if (this.$refs[refName]) { + this.$refs[refName].resetFields(); + } +} + +// 添加日期范围 +export function addDateRange(params, dateRange, propName) { + var search = params; + search.params = {}; + if (null != dateRange && '' != dateRange) { + if (typeof(propName) === "undefined") { + search.params["beginTime"] = dateRange[0]; + search.params["endTime"] = dateRange[1]; + } else { + search.params["begin" + propName] = dateRange[0]; + search.params["end" + propName] = dateRange[1]; + } + } + return search; +} + +// 回显数据字典 +export function selectDictLabel(datas, value) { + var actions = []; + Object.keys(datas).some((key) => { + if (datas[key].dictValue == ('' + value)) { + actions.push(datas[key].dictLabel); + return true; + } + }) + return actions.join(''); +} + +// 回显数据字典(字符串数组) +export function selectDictLabels(datas, value, separator) { + var actions = []; + var currentSeparator = undefined === separator ? "," : separator; + var temp = value.split(currentSeparator); + Object.keys(value.split(currentSeparator)).some((val) => { + Object.keys(datas).some((key) => { + if (datas[key].dictValue == ('' + temp[val])) { + actions.push(datas[key].dictLabel + currentSeparator); + } + }) + }) + return actions.join('').substring(0, actions.join('').length - 1); +} + +// 通用下载方法 +export function download(fileName) { + window.location.href = baseURL + "/common/download?fileName=" + encodeURI(fileName) + "&delete=" + true; +} + +// 字符串格式化(%s ) +export function sprintf(str) { + var args = arguments, flag = true, i = 1; + str = str.replace(/%s/g, function () { + var arg = args[i++]; + if (typeof arg === 'undefined') { + flag = false; + return ''; + } + return arg; + }); + return flag ? str : ''; +} + +// 转换字符串,undefined,null等转化为"" +export function praseStrEmpty(str) { + if (!str || str == "undefined" || str == "null") { + return ""; + } + return str; +} + +/** + * 构造树型结构数据 + * @param {*} data 数据源 + * @param {*} id id字段 默认 'id' + * @param {*} parentId 父节点字段 默认 'parentId' + * @param {*} children 孩子节点字段 默认 'children' + * @param {*} rootId 根Id 默认 0 + */ +export function handleTree(data, id, parentId, children, rootId) { + id = id || 'id' + parentId = parentId || 'parentId' + children = children || 'children' + rootId = rootId || Math.min.apply(Math, data.map(item => { return item[parentId] })) || 0 + //对源数据深度克隆 + const cloneData = JSON.parse(JSON.stringify(data)) + //循环所有项 + const treeData = cloneData.filter(father => { + let branchArr = cloneData.filter(child => { + //返回每一项的子级数组 + return father[id] === child[parentId] + }); + branchArr.length > 0 ? father.children = branchArr : ''; + //返回第一层 + return father[parentId] === rootId; + }); + return treeData != '' ? treeData : data; +} diff --git a/dvadmin-ui/src/utils/scroll-to.js b/dvadmin-ui/src/utils/scroll-to.js new file mode 100755 index 0000000..c5d8e04 --- /dev/null +++ b/dvadmin-ui/src/utils/scroll-to.js @@ -0,0 +1,58 @@ +Math.easeInOutQuad = function(t, b, c, d) { + t /= d / 2 + if (t < 1) { + return c / 2 * t * t + b + } + t-- + return -c / 2 * (t * (t - 2) - 1) + b +} + +// requestAnimationFrame for Smart Animating http://goo.gl/sx5sts +var requestAnimFrame = (function() { + return window.requestAnimationFrame || window.webkitRequestAnimationFrame || window.mozRequestAnimationFrame || function(callback) { window.setTimeout(callback, 1000 / 60) } +})() + +/** + * Because it's so fucking difficult to detect the scrolling element, just move them all + * @param {number} amount + */ +function move(amount) { + document.documentElement.scrollTop = amount + document.body.parentNode.scrollTop = amount + document.body.scrollTop = amount +} + +function position() { + return document.documentElement.scrollTop || document.body.parentNode.scrollTop || document.body.scrollTop +} + +/** + * @param {number} to + * @param {number} duration + * @param {Function} callback + */ +export function scrollTo(to, duration, callback) { + const start = position() + const change = to - start + const increment = 20 + let currentTime = 0 + duration = (typeof (duration) === 'undefined') ? 500 : duration + var animateScroll = function() { + // increment the time + currentTime += increment + // find the value with the quadratic in-out easing function + var val = Math.easeInOutQuad(currentTime, start, change, duration) + // move the document.body + move(val) + // do the animation unless its over + if (currentTime < duration) { + requestAnimFrame(animateScroll) + } else { + if (callback && typeof (callback) === 'function') { + // the animation is done so lets callback + callback() + } + } + } + animateScroll() +} diff --git a/dvadmin-ui/src/utils/validate.js b/dvadmin-ui/src/utils/validate.js new file mode 100755 index 0000000..adfa254 --- /dev/null +++ b/dvadmin-ui/src/utils/validate.js @@ -0,0 +1,83 @@ +/** + * @param {string} path + * @returns {Boolean} + */ +export function isExternal(path) { + return /^(https?:|mailto:|tel:)/.test(path) +} + +/** + * @param {string} str + * @returns {Boolean} + */ +export function validUsername(str) { + const valid_map = ['admin', 'editor'] + return valid_map.indexOf(str.trim()) >= 0 +} + +/** + * @param {string} url + * @returns {Boolean} + */ +export function validURL(url) { + const reg = /^(https?|ftp):\/\/([a-zA-Z0-9.-]+(:[a-zA-Z0-9.&%$-]+)*@)*((25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])){3}|([a-zA-Z0-9-]+\.)*[a-zA-Z0-9-]+\.(com|edu|gov|int|mil|net|org|biz|arpa|info|name|pro|aero|coop|museum|[a-zA-Z]{2}))(:[0-9]+)*(\/($|[a-zA-Z0-9.,?'\\+&%$#=~_-]+))*$/ + return reg.test(url) +} + +/** + * @param {string} str + * @returns {Boolean} + */ +export function validLowerCase(str) { + const reg = /^[a-z]+$/ + return reg.test(str) +} + +/** + * @param {string} str + * @returns {Boolean} + */ +export function validUpperCase(str) { + const reg = /^[A-Z]+$/ + return reg.test(str) +} + +/** + * @param {string} str + * @returns {Boolean} + */ +export function validAlphabets(str) { + const reg = /^[A-Za-z]+$/ + return reg.test(str) +} + +/** + * @param {string} email + * @returns {Boolean} + */ +export function validEmail(email) { + const reg = /^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/ + return reg.test(email) +} + +/** + * @param {string} str + * @returns {Boolean} + */ +export function isString(str) { + if (typeof str === 'string' || str instanceof String) { + return true + } + return false +} + +/** + * @param {Array} arg + * @returns {Boolean} + */ +export function isArray(arg) { + if (typeof Array.isArray === 'undefined') { + return Object.prototype.toString.call(arg) === '[object Array]' + } + return Array.isArray(arg) +} diff --git a/dvadmin-ui/src/utils/zipdownload.js b/dvadmin-ui/src/utils/zipdownload.js new file mode 100755 index 0000000..363c45a --- /dev/null +++ b/dvadmin-ui/src/utils/zipdownload.js @@ -0,0 +1,40 @@ +import axios from 'axios' +import { getToken } from '@/utils/auth' + +const mimeMap = { + xlsx: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', + zip: 'application/zip' +} + +const baseUrl = process.env.VUE_APP_BASE_API +export function downLoadZip(str, filename) { + var url = baseUrl + str + axios({ + method: 'get', + url: url, + responseType: 'blob', + headers: { 'Authorization': 'Bearer ' + getToken() } + }).then(res => { + resolveBlob(res, mimeMap.zip) + }) +} +/** + * 解析blob响应内容并下载 + * @param {*} res blob响应内容 + * @param {String} mimeType MIME类型 + */ +export function resolveBlob(res, mimeType) { + const aLink = document.createElement('a') + var blob = new Blob([res.data], { type: mimeType }) + // //从response的headers中获取filename, 后端response.setHeader("Content-disposition", "attachment; filename=xxxx.docx") 设置的文件名; + var patt = new RegExp('filename=([^;]+\\.[^\\.;]+);*') + var contentDisposition = decodeURI(res.headers['content-disposition']) + var result = patt.exec(contentDisposition) + var fileName = result[1] + fileName = fileName.replace(/\"/g, '') + aLink.href = URL.createObjectURL(blob) + aLink.setAttribute('download', fileName) // 设置下载文件名称 + document.body.appendChild(aLink) + aLink.click() + document.body.appendChild(aLink) +} diff --git a/dvadmin-ui/src/views/components/icons/element-icons.js b/dvadmin-ui/src/views/components/icons/element-icons.js new file mode 100755 index 0000000..9ea4d63 --- /dev/null +++ b/dvadmin-ui/src/views/components/icons/element-icons.js @@ -0,0 +1,3 @@ +const elementIcons = ['platform-eleme', 'eleme', 'delete-solid', 'delete', 's-tools', 'setting', 'user-solid', 'user', 'phone', 'phone-outline', 'more', 'more-outline', 'star-on', 'star-off', 's-goods', 'goods', 'warning', 'warning-outline', 'question', 'info', 'remove', 'circle-plus', 'success', 'error', 'zoom-in', 'zoom-out', 'remove-outline', 'circle-plus-outline', 'circle-check', 'circle-close', 's-help', 'help', 'minus', 'plus', 'check', 'close', 'picture', 'picture-outline', 'picture-outline-round', 'upload', 'upload2', 'download', 'camera-solid', 'camera', 'video-camera-solid', 'video-camera', 'message-solid', 'bell', 's-cooperation', 's-order', 's-platform', 's-fold', 's-unfold', 's-operation', 's-promotion', 's-home', 's-release', 's-ticket', 's-management', 's-open', 's-shop', 's-marketing', 's-flag', 's-comment', 's-finance', 's-claim', 's-custom', 's-opportunity', 's-data', 's-check', 's-grid', 'menu', 'share', 'd-caret', 'caret-left', 'caret-right', 'caret-bottom', 'caret-top', 'bottom-left', 'bottom-right', 'back', 'right', 'bottom', 'top', 'top-left', 'top-right', 'arrow-left', 'arrow-right', 'arrow-down', 'arrow-up', 'd-arrow-left', 'd-arrow-right', 'video-pause', 'video-play', 'refresh', 'refresh-right', 'refresh-left', 'finished', 'sort', 'sort-up', 'sort-down', 'rank', 'loading', 'view', 'c-scale-to-original', 'date', 'edit', 'edit-outline', 'folder', 'folder-opened', 'folder-add', 'folder-remove', 'folder-delete', 'folder-checked', 'tickets', 'document-remove', 'document-delete', 'document-copy', 'document-checked', 'document', 'document-add', 'printer', 'paperclip', 'takeaway-box', 'search', 'monitor', 'attract', 'mobile', 'scissors', 'umbrella', 'headset', 'brush', 'mouse', 'coordinate', 'magic-stick', 'reading', 'data-line', 'data-board', 'pie-chart', 'data-analysis', 'collection-tag', 'film', 'suitcase', 'suitcase-1', 'receiving', 'collection', 'files', 'notebook-1', 'notebook-2', 'toilet-paper', 'office-building', 'school', 'table-lamp', 'house', 'no-smoking', 'smoking', 'shopping-cart-full', 'shopping-cart-1', 'shopping-cart-2', 'shopping-bag-1', 'shopping-bag-2', 'sold-out', 'sell', 'present', 'box', 'bank-card', 'money', 'coin', 'wallet', 'discount', 'price-tag', 'news', 'guide', 'male', 'female', 'thumb', 'cpu', 'link', 'connection', 'open', 'turn-off', 'set-up', 'chat-round', 'chat-line-round', 'chat-square', 'chat-dot-round', 'chat-dot-square', 'chat-line-square', 'message', 'postcard', 'position', 'turn-off-microphone', 'microphone', 'close-notification', 'bangzhu', 'time', 'odometer', 'crop', 'aim', 'switch-button', 'full-screen', 'copy-document', 'mic', 'stopwatch', 'medal-1', 'medal', 'trophy', 'trophy-1', 'first-aid-kit', 'discover', 'place', 'location', 'location-outline', 'location-information', 'add-location', 'delete-location', 'map-location', 'alarm-clock', 'timer', 'watch-1', 'watch', 'lock', 'unlock', 'key', 'service', 'mobile-phone', 'bicycle', 'truck', 'ship', 'basketball', 'football', 'soccer', 'baseball', 'wind-power', 'light-rain', 'lightning', 'heavy-rain', 'sunrise', 'sunrise-1', 'sunset', 'sunny', 'cloudy', 'partly-cloudy', 'cloudy-and-sunny', 'moon', 'moon-night', 'dish', 'dish-1', 'food', 'chicken', 'fork-spoon', 'knife-fork', 'burger', 'tableware', 'sugar', 'dessert', 'ice-cream', 'hot-water', 'water-cup', 'coffee-cup', 'cold-drink', 'goblet', 'goblet-full', 'goblet-square', 'goblet-square-full', 'refrigerator', 'grape', 'watermelon', 'cherry', 'apple', 'pear', 'orange', 'coffee', 'ice-tea', 'ice-drink', 'milk-tea', 'potato-strips', 'lollipop', 'ice-cream-square', 'ice-cream-round'] + +export default elementIcons diff --git a/dvadmin-ui/src/views/components/icons/index.vue b/dvadmin-ui/src/views/components/icons/index.vue new file mode 100755 index 0000000..d3c9a71 --- /dev/null +++ b/dvadmin-ui/src/views/components/icons/index.vue @@ -0,0 +1,87 @@ + + + + + diff --git a/dvadmin-ui/src/views/components/icons/svg-icons.js b/dvadmin-ui/src/views/components/icons/svg-icons.js new file mode 100755 index 0000000..724cd8e --- /dev/null +++ b/dvadmin-ui/src/views/components/icons/svg-icons.js @@ -0,0 +1,10 @@ +const req = require.context('../../../assets/icons/svg', false, /\.svg$/) +const requireAll = requireContext => requireContext.keys() + +const re = /\.\/(.*)\.svg/ + +const svgIcons = requireAll(req).map(i => { + return i.match(re)[1] +}) + +export default svgIcons diff --git a/dvadmin-ui/src/views/dashboard/BarChart.vue b/dvadmin-ui/src/views/dashboard/BarChart.vue new file mode 100755 index 0000000..be0af34 --- /dev/null +++ b/dvadmin-ui/src/views/dashboard/BarChart.vue @@ -0,0 +1,102 @@ + + + diff --git a/dvadmin-ui/src/views/dashboard/LineChart.vue b/dvadmin-ui/src/views/dashboard/LineChart.vue new file mode 100755 index 0000000..e654168 --- /dev/null +++ b/dvadmin-ui/src/views/dashboard/LineChart.vue @@ -0,0 +1,135 @@ + + + diff --git a/dvadmin-ui/src/views/dashboard/PanelGroup.vue b/dvadmin-ui/src/views/dashboard/PanelGroup.vue new file mode 100755 index 0000000..1a1081f --- /dev/null +++ b/dvadmin-ui/src/views/dashboard/PanelGroup.vue @@ -0,0 +1,181 @@ + + + + + diff --git a/dvadmin-ui/src/views/dashboard/PieChart.vue b/dvadmin-ui/src/views/dashboard/PieChart.vue new file mode 100755 index 0000000..4d2ef32 --- /dev/null +++ b/dvadmin-ui/src/views/dashboard/PieChart.vue @@ -0,0 +1,79 @@ + + + diff --git a/dvadmin-ui/src/views/dashboard/RaddarChart.vue b/dvadmin-ui/src/views/dashboard/RaddarChart.vue new file mode 100755 index 0000000..6823af3 --- /dev/null +++ b/dvadmin-ui/src/views/dashboard/RaddarChart.vue @@ -0,0 +1,116 @@ + + + diff --git a/dvadmin-ui/src/views/dashboard/mixins/resize.js b/dvadmin-ui/src/views/dashboard/mixins/resize.js new file mode 100755 index 0000000..b1e76e9 --- /dev/null +++ b/dvadmin-ui/src/views/dashboard/mixins/resize.js @@ -0,0 +1,56 @@ +import { debounce } from '@/utils' + +export default { + data() { + return { + $_sidebarElm: null, + $_resizeHandler: null + } + }, + mounted() { + this.initListener() + }, + activated() { + if (!this.$_resizeHandler) { + // avoid duplication init + this.initListener() + } + + // when keep-alive chart activated, auto resize + this.resize() + }, + beforeDestroy() { + this.destroyListener() + }, + deactivated() { + this.destroyListener() + }, + methods: { + // use $_ for mixins properties + // https://vuejs.org/v2/style-guide/index.html#Private-property-names-essential + $_sidebarResizeHandler(e) { + if (e.propertyName === 'width') { + this.$_resizeHandler() + } + }, + initListener() { + this.$_resizeHandler = debounce(() => { + this.resize() + }, 100) + window.addEventListener('resize', this.$_resizeHandler) + + this.$_sidebarElm = document.getElementsByClassName('sidebar-container')[0] + this.$_sidebarElm && this.$_sidebarElm.addEventListener('transitionend', this.$_sidebarResizeHandler) + }, + destroyListener() { + window.removeEventListener('resize', this.$_resizeHandler) + this.$_resizeHandler = null + + this.$_sidebarElm && this.$_sidebarElm.removeEventListener('transitionend', this.$_sidebarResizeHandler) + }, + resize() { + const { chart } = this + chart && chart.resize() + } + } +} diff --git a/dvadmin-ui/src/views/error/401.vue b/dvadmin-ui/src/views/error/401.vue new file mode 100755 index 0000000..448b6ec --- /dev/null +++ b/dvadmin-ui/src/views/error/401.vue @@ -0,0 +1,88 @@ + + + + + diff --git a/dvadmin-ui/src/views/error/404.vue b/dvadmin-ui/src/views/error/404.vue new file mode 100755 index 0000000..96f075c --- /dev/null +++ b/dvadmin-ui/src/views/error/404.vue @@ -0,0 +1,233 @@ + + + + + diff --git a/dvadmin-ui/src/views/index.vue b/dvadmin-ui/src/views/index.vue new file mode 100755 index 0000000..91667f5 --- /dev/null +++ b/dvadmin-ui/src/views/index.vue @@ -0,0 +1,570 @@ + + + + + + diff --git a/dvadmin-ui/src/views/index_v1.vue b/dvadmin-ui/src/views/index_v1.vue new file mode 100755 index 0000000..d2d2ec6 --- /dev/null +++ b/dvadmin-ui/src/views/index_v1.vue @@ -0,0 +1,98 @@ + + + + + diff --git a/dvadmin-ui/src/views/login.vue b/dvadmin-ui/src/views/login.vue new file mode 100755 index 0000000..96d6a80 --- /dev/null +++ b/dvadmin-ui/src/views/login.vue @@ -0,0 +1,203 @@ + + + + + diff --git a/dvadmin-ui/src/views/monitor/cache/index.vue b/dvadmin-ui/src/views/monitor/cache/index.vue new file mode 100755 index 0000000..df0cb1e --- /dev/null +++ b/dvadmin-ui/src/views/monitor/cache/index.vue @@ -0,0 +1,153 @@ + + + diff --git a/dvadmin-ui/src/views/monitor/druid/index.vue b/dvadmin-ui/src/views/monitor/druid/index.vue new file mode 100755 index 0000000..da5b08f --- /dev/null +++ b/dvadmin-ui/src/views/monitor/druid/index.vue @@ -0,0 +1,26 @@ +