diff --git a/.eslintrc.js b/.eslintrc.js
index 84fed4d..a91d224 100644
--- a/.eslintrc.js
+++ b/.eslintrc.js
@@ -1,15 +1,22 @@
+// 参考:https://eslint.bootcss.com/docs/rules/
+// 参考:https://blog.csdn.net/x550392236/article/details/89497202
+// 参考:https://blog.csdn.net/brokenkay/article/details/111106266
+
module.exports = {
+ root: true,
env: {
- browser: true,
- es2021: true,
+ node: true, //允许运行在node环境下
+ browser: true, //允许运行在浏览器环境下
+ es2021: true, //允许运行es2021环境下语法
},
parser: 'vue-eslint-parser',
- extends: ['plugin:@typescript-eslint/recommended', 'plugin:vue/vue3-essential', 'prettier'],
parserOptions: {
ecmaVersion: 'latest',
- parser: '@typescript-eslint/parser',
sourceType: 'module',
+ parser: '@typescript-eslint/parser',
},
+ extends: ['plugin:vue/vue3-essential', 'eslint:recommended', 'plugin:@typescript-eslint/recommended', 'prettier'],
+ plugins: ['vue', '@typescript-eslint', 'prettier'],
settings: {
'import/resolver': {
alias: {
@@ -18,16 +25,25 @@ module.exports = {
},
},
},
- plugins: ['vue', '@typescript-eslint', 'prettier'],
+ globals: {
+ //可以定义全局中的变量的权限(只读,可读可写)
+ defineProps: 'readonly',
+ defineEmits: 'readonly',
+ defineExpose: 'readonly',
+ withDefaults: 'readonly',
+ uni: 'readonly',
+ },
rules: {
'no-console': process.env.NODE_ENV === 'production' ? 'warn' : 'off',
'no-debugger': process.env.NODE_ENV === 'production' ? 'warn' : 'off',
- 'no-var': 'error',
+ 'no-var': 'error', //要求使用 let 或 const 而不是 var
'prettier/prettier': 'error',
'vue/no-multiple-template-root': 'off',
'no-mutating-props': 'off',
'vue/no-v-html': 'off',
- // @fixable 必须使用单引号,禁止使用双引号
+ camelcase: 'error', // 双峰驼命名格式
+ // indent: ['error', 4], //代码缩进4个空格 (switch时与prettier发生冲突)
+ eqeqeq: ['error', 'always', { null: 'ignore' }], //比较时强制使用 === 或者 !==,但对null作比较时可以不用全等
quotes: [
'error',
'single',
@@ -35,7 +51,7 @@ module.exports = {
avoidEscape: true,
allowTemplateLiterals: true,
},
- ],
+ ], // @fixable 必须使用单引号,禁止使用双引号
// 结尾必须有分号;
semi: [
'error',
@@ -43,7 +59,7 @@ module.exports = {
{
omitLastInOneLineBlock: true,
},
- ],
+ ], // 结尾必须有分号;
'vue/script-setup-uses-vars': 'error',
'@typescript-eslint/ban-ts-ignore': 'off',
'@typescript-eslint/explicit-function-return-type': 'off',
@@ -83,11 +99,4 @@ module.exports = {
'vue/attribute-hyphenation': 'off',
'vue/require-default-prop': 'off',
},
- globals: {
- defineProps: 'readonly',
- defineEmits: 'readonly',
- defineExpose: 'readonly',
- withDefaults: 'readonly',
- uni: 'readonly',
- },
};
diff --git a/.gitignore b/.gitignore
index 2f46287..6347ba2 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,11 +1,34 @@
-node_modules
-dist
-.eslintcache
-.pnpm-lock.yaml
-*.local
+# 日志
+logs
+*.log
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+pnpm-debug.log*
+lerna-debug.log*
+node_modules
+.DS_Store
+dist
+dist-ssr
+coverage
+*.local
+.eslintcache
+
+/cypress/videos/
+/cypress/screenshots/
# 编辑器目录和文件
+.vscode/*
+.hbuilderx/*
+!.vscode/extensions.json
+!.vscode/settings.json
.idea
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
+*.sw?
+.eslintcache
.hbuilderx
pnpm-lock.yaml
\ No newline at end of file
diff --git a/.vscode/extensions.json b/.vscode/extensions.json
index 9771df2..8cff2f2 100644
--- a/.vscode/extensions.json
+++ b/.vscode/extensions.json
@@ -1,9 +1,10 @@
{
"recommendations": [
- "CodeInChinese.EnglishChineseDictionary", // 翻译(英汉词典)
- "kisstkondoros.vscode-gutter-preview", // Image 预览
- "ritwickdey.LiveServer", // 为静态和动态页面启动具有实时重载功能的本地开发服务器
+ "Vue.volar", // Vue语言支持扩展
+ "Vue.vscode-typescript-vue-plugin", // 一个TS服务器插件,使TS服务器知道*.vue文件。
+ "esbenp.prettier-vscode", // 代码格式化
+ "dbaeumer.vscode-eslint", //代码质量检查
"antfu.unocss", // UnoCSS 原子化css
- "esbenp.prettier-vscode" // 格式化
+ "mrmlnc.vscode-autoprefixer" //将less/scss/css文件自动添加浏览器兼容前缀
]
}
diff --git a/.vscode/settings.json b/.vscode/settings.json
index b68d597..8ea3fa5 100644
--- a/.vscode/settings.json
+++ b/.vscode/settings.json
@@ -4,10 +4,11 @@
"editor.defaultFormatter": "esbenp.prettier-vscode", // 定义一个默认格式化程序, 该格式化程序优先于所有其他格式化程序设置。必须是提供格式化程序的扩展的标识符。
"editor.formatOnSave": true, //在保存时格式化文件。格式化程序必须可用,延迟后文件不能保存,并且编辑器不能关闭。
"editor.detectIndentation": false, // 控制在基于文件内容打开文件时是否自动检测 #editor.tabSize# 和 #editor.insertSpaces#。
- "editor.tabSize": 4,
+ "editor.tabSize": 4, // 一个制表符等于的空格数。当 #editor.detectIndentation# 打开时,将根据文件内容替代此设置。
"editor.codeActionsOnSave": {
"source.fixAll": true, // 控制是否应在文件保存时运行自动修复操作。
"source.fixAll.eslint": true,
"source.fixAll.stylelint": true
- } // 要在保存时运行的代码操作种类。
+ }, // 要在保存时运行的代码操作种类。
+ "files.eol": "\n" //行尾字符与.prettierrc.cjs配置一致
}
diff --git a/index.html b/index.html
index 1948eec..d2edea7 100644
--- a/index.html
+++ b/index.html
@@ -1,21 +1,24 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/package.json b/package.json
index 0f62709..9eccce5 100644
--- a/package.json
+++ b/package.json
@@ -88,8 +88,7 @@
"unocss": "^0.46.5",
"unocss-preset-weapp": "^0.2.5",
"unplugin-vue-components": "^0.22.12",
- "vite": "^4.1.4",
- "vite-plugin-eslint": "^1.8.1"
+ "vite": "^4.1.4"
},
"husky": {
"hooks": {
diff --git a/src/services/api/user.ts b/src/services/api/user.ts
index 2be05ab..7f785e1 100644
--- a/src/services/api/user.ts
+++ b/src/services/api/user.ts
@@ -1 +1,2 @@
-import { request } from '@/utils/http';
+// import { request } from '@/utils/http';
+export {};
diff --git a/src/types/router/route.d.ts b/src/types/router/route.d.ts
index f2aaa86..502697f 100644
--- a/src/types/router/route.d.ts
+++ b/src/types/router/route.d.ts
@@ -1,5 +1,5 @@
-import { types } from 'sass';
-import Boolean = types.Boolean;
+// import { types } from 'sass';
+// import Boolean = types.Boolean;
export interface Route extends Record {
path: string;
diff --git a/src/utils/platform.ts b/src/utils/platform.ts
index 366a148..23be0ef 100644
--- a/src/utils/platform.ts
+++ b/src/utils/platform.ts
@@ -20,7 +20,7 @@ export function judgePlatform(target: PLATFORMS): boolean {
let isMpLark = false;
let isMpQq = false;
let isMpKuaishou = false;
- let isMpJd = false;
+ // let isMpJd = false;
let isMp360 = false;
let isQuickAppWebView = false;
let isQuickAppWebViewUnion = false;
@@ -32,111 +32,93 @@ export function judgePlatform(target: PLATFORMS): boolean {
isVue3 = true;
/* #endif */
return isVue3;
- break;
case PLATFORMS.APP_PLUS:
/* #ifdef APP-PLUS */
isAppPlus = true;
/* #endif */
return isAppPlus;
- break;
case PLATFORMS.APP_PLUS_NVUE:
/* #ifdef APP-PLUS-NVUE */
isAppPlusNvue = true;
/* #endif */
return isAppPlusNvue;
- break;
case PLATFORMS.APP_NVUE:
/* #ifdef APP-NVUE */
isAppNvue = true;
/* #endif */
return isAppNvue;
- break;
case PLATFORMS.H5:
/* #ifdef H5 */
isH5 = true;
/* #endif */
return isH5;
- break;
case PLATFORMS.MP:
/* #ifdef MP */
isMp = true;
/* #endif */
return isMp;
- break;
case PLATFORMS.MP_WEIXIN:
/* #ifdef MP-WEIXIN */
isMpWeinxin = true;
/* #endif */
return isMpWeinxin;
- break;
case PLATFORMS.MP_ALIPAY:
/* #ifdef MP-ALIPAY */
isMpAlipay = true;
/* #endif */
return isMpAlipay;
- break;
case PLATFORMS.MP_BAIDU:
/* #ifdef MP_BAIDU */
isMpBaidu = true;
/* #endif */
return isMpBaidu;
- break;
case PLATFORMS.MP_TOUTIAO:
/* #ifdef MP-TOUTIAO */
isMpToutiao = true;
/* #endif */
return isMpToutiao;
- break;
case PLATFORMS.MP_LARK:
/* #ifdef MP-LARK */
isMpLark = true;
/* #endif */
return isMpLark;
- break;
case PLATFORMS.MP_QQ:
/* #ifdef MP-QQ */
isMpQq = true;
/* #endif */
return isMpQq;
- break;
case PLATFORMS.MP_KUAISHOU:
/* #ifdef MP-KUAISHOU */
isMpKuaishou = true;
/* #endif */
return isMpKuaishou;
- break;
- case PLATFORMS.MP_JD:
- /* #ifdef MP-JD */
- isMpJd = true;
- /* #endif */
- return (isMpJd = true);
- break;
+ // case PLATFORMS.MP_JD:
+ // /* #ifdef MP-JD */
+ // isMpJd = true;
+ // /* #endif */
+ // return (isMpJd = true);
+ // break;
case PLATFORMS.MP_360:
/* #ifdef MP-360 */
isMp360 = true;
/* #endif */
return isMp360;
- break;
case PLATFORMS.QUICKAPP_WEBVIEW:
/* #ifdef QUICKAPP-WEBVIEW */
isQuickAppWebView = true;
/* #endif */
return isQuickAppWebView;
- break;
case PLATFORMS.QUICKAPP_WEBVIEW_UNION:
/* #ifdef QUICKAPP-WEBVIEW-UNION */
isQuickAppWebViewUnion = true;
/* #endif */
return isQuickAppWebViewUnion;
- break;
case PLATFORMS.QUICKAPP_WEBVIEW_HUAWEI:
/* #ifdef QUICKAPP-WEBVIEW-HUAWEI */
isQuickAppWebViewHuawei = true;
/* #endif */
return isQuickAppWebViewHuawei;
- break;
default:
return false;
}
- return false;
}
diff --git a/tsconfig.json b/tsconfig.json
index 65cfb69..aee314d 100644
--- a/tsconfig.json
+++ b/tsconfig.json
@@ -1,54 +1,25 @@
{
- /* 根选项 */
- "include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"], // 指定被编译文件所在的目录
- "exclude": [], // 指定不需要被编译的目录
- //使用小技巧:在填写路径时 ** 表示任意目录, * 表示任意文件。
-
- /* 项目选项 */
"compilerOptions": {
- "target": "esnext", // 目标语言的版本
- "useDefineForClassFields": true, //发出符合ecmascript标准的类字段
- "module": "esnext", // 生成代码的模板标准
- "moduleResolution": "node", //指定TypeScript如何从给定的模块说明符中查找文件
- "jsx": "preserve", //指定生成什么JSX代码。
- "sourceMap": true, //为发出的JavaScript文件创建源映射文件。
- "esModuleInterop": true, // 允许export=导出,由import from 导入
- "lib": ["esnext", "dom"], // TS需要引用的库
- "types": ["@dcloudio/types"],
- // "allowJs": true, // 允许编译器编译JS,JSX文件
- // "checkJs": false, // 允许在JS文件中报错,通常与allowJS一起使用
- "removeComments": true, // 删除注释
- "paths": {
- "@/*": ["./src/*"]
- }, //指定一组条目,它们将导入重新映射到其他查找位置。
-
- /* 严格检查选项 */
+ "target": "ESNext", // 目标语言的版本
+ "useDefineForClassFields": true, //发出符合ECMAScript标准的类字段
+ "module": "ESNext", // 生成代码的模板标准
+ "moduleResolution": "Node", //指定TypeScript如何从给定的模块说明符中查找文件
"strict": true, // 开启所有严格的类型检查
- "alwaysStrict": true, // 在代码中注入'use strict'
- "noImplicitAny": true, // 不允许隐式的any类型
- "noImplicitThis": true, // 不允许this有隐式的any类型
- "strictNullChecks": true, // 在进行类型检查时,请考虑null和undefined。
- "strictBindCallApply": true, // 检查bind、call和apply方法的参数是否与原始函数匹配。
- "strictFunctionTypes": true, // 在给函数赋值时,要确保参数和返回值是子类型兼容的。
- "strictPropertyInitialization": true, // 类的实例属性必须初始化
-
- /* 额外检查 */
- "noUnusedLocals": true, //是否检查未使用的局部变量
- "noUnusedParameters": true, //是否检查未使用的参数
- "noImplicitReturns": true, //检查函数是否不含有隐式返回值
- "noImplicitOverride": true, //是否检查子类继承自基类时,其重载的函数命名与基类的函数不同步问题
- "noFallthroughCasesInSwitch": true, //检查switch中是否含有case没有使用break跳出
- "noUncheckedIndexedAccess": true, //是否通过索引签名来描述对象上有未知键但已知值的对象
- "noPropertyAccessFromIndexSignature": false, //是否通过" . “(obj.key) 语法访问字段和"索引”( obj[“key”]), 以及在类型中声明属性的方式之间的一致性
-
- /* 实验选项 */
- "experimentalDecorators": true, //是否启用对装饰器的实验性支持,装饰器是一种语言特性,还没有完全被 JavaScript 规范批准
- "emitDecoratorMetadata": true, //为装饰器启用对发出类型元数据的实验性支持
-
- /* 高级选项 */
- "forceConsistentCasingInFileNames": true, //是否区分文件系统大小写规则
- "extendedDiagnostics": false, //是否查看 TS 在编译时花费的时间
+ "jsx": "preserve", //指定生成什么JSX代码。
+ "resolveJsonModule": true, //是否解析 JSON 模块
+ "isolatedModules": true, // 确保每个文件都可以安全地转译,而不依赖于其他导入。
+ "esModuleInterop": true, // 发出额外的JavaScript以简化对导入Common/S模块的支持。这使得allowsyntheticdefaulultimports类型兼容。
+ "removeComments": true, // 删除注释
+ "types": ["@dcloudio/types"],
+ "paths": { "@/*": ["./src/*"] }, //指定一组条目,它们将导入重新映射到其他查找位置。
+ "lib": ["ESNext", "DOM"], // TS需要引用的库
+ "skipLibCheck": true, // 跳过所有.d.ts文件的类型检查
+ "noEmit": true, // 从编译中禁用发射文件。
"noEmitOnError": true, //有错误时不进行编译
- "resolveJsonModule": true //是否解析 JSON 模块
- }
+ "forceConsistentCasingInFileNames": true, //是否区分文件系统大小写规则
+ "experimentalDecorators": true, //是否启用对装饰器的实验性支持,装饰器是一种语言特性,还没有完全被 JavaScript 规范批准
+ "emitDecoratorMetadata": true //为装饰器启用对发出类型元数据的实验性支持
+ },
+ "include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"],
+ "references": [{ "path": "./tsconfig.node.json" }] // 引用的项目。要求TypeScript 3.0或更高版本。
}
diff --git a/tsconfig.node.json b/tsconfig.node.json
new file mode 100644
index 0000000..1a7ab1b
--- /dev/null
+++ b/tsconfig.node.json
@@ -0,0 +1,9 @@
+{
+ "compilerOptions": {
+ "composite": true,
+ "module": "ESNext",
+ "moduleResolution": "Node",
+ "allowSyntheticDefaultImports": true
+ },
+ "include": ["vite.config.*", "vitest.config.*", "cypress.config.*", "playwright.config.*"]
+}
diff --git a/vite.config.ts b/vite.config.ts
index b4e77f2..6d17f3f 100644
--- a/vite.config.ts
+++ b/vite.config.ts
@@ -1,67 +1,26 @@
-import { ConfigEnv, UserConfig } from 'vite';
-import uni from '@dcloudio/vite-plugin-uni';
-import eslintPlugin from 'vite-plugin-eslint';
+// Vite中文网:https://vitejs.cn/config/
+import { ConfigEnv, loadEnv, UserConfig } from 'vite';
import { resolve } from 'path';
-import { loadEnv } from 'vite';
+import uni from '@dcloudio/vite-plugin-uni';
import Unocss from 'unocss/vite';
-//发布时动态修改 manifest.json
-// if (process.env.NODE_ENV === 'production') {
-// // 读取 manifest.json ,修改后重新写入
-// const fs = require('fs');
-// const manifestPath = './src/manifest.json';
-// let Manifest = fs.readFileSync(manifestPath, { encoding: 'utf-8' });
-// function replaceManifest(path: string, value: any) {
-// const arr = path.split('.');
-// const len = arr.length;
-// const lastItem = arr[len - 1];
-
-// let i = 0;
-// let ManifestArr = Manifest.split(/\n/);
-
-// for (let index = 0; index < ManifestArr.length; index++) {
-// const item = ManifestArr[index];
-// if (new RegExp(`"${arr[i]}"`).test(item)) ++i;
-// if (i === len) {
-// const hasComma = /,/.test(item);
-// ManifestArr[index] = item.replace(new RegExp(`"${lastItem}"[\\s\\S]*:[\\s\\S]*`), `"${lastItem}": ${value}${hasComma ? ',' : ''}`);
-// break;
-// }
-// }
-
-// Manifest = ManifestArr.join('\n');
-// }
-// let Data1 = new Date().toLocaleDateString();
-// let Data2 = new Date().toLocaleTimeString();
-// let Data_ = Data1.replace(/\//g, '-') + ' ' + Data2;
-// // 使用
-// replaceManifest('description', JSON.stringify(`app平台-${Data_}`));
-// replaceManifest(
-// 'versionName',
-// JSON.stringify(
-// String(Number(JSON.parse(Manifest).versionCode) + 1)
-// .split('')
-// .join('.')
-// )
-// );
-// replaceManifest('versionCode', JSON.stringify(String(Number(JSON.parse(Manifest).versionCode) + 1)));
-// fs.writeFileSync(manifestPath, Manifest, { flag: 'w' });
-// }
-
-// https://vitejs.cn/config/
export default ({ mode }: ConfigEnv): UserConfig => {
const root = process.cwd();
const env = loadEnv(mode, root);
return {
base: './',
+ // 设置路径别名
resolve: {
alias: {
'@': resolve('./src'),
},
+ extensions: ['.js', '.json', '.ts', '.vue'], // 使用路径别名时想要省略的后缀名,可以自己 增减
},
+ // 自定义全局变量
define: {
'process.env': {},
},
+ // 开发服务器配置
server: {
host: true,
// open: true,
@@ -79,21 +38,24 @@ export default ({ mode }: ConfigEnv): UserConfig => {
},
},
},
- plugins: [
- uni(),
- Unocss(),
- // eslintPlugin({
- // include: ['src/**/*.js', 'src/**/*.vue', 'src/**/*.ts'],
- // exclude: ['./node_modules/**'],
- // cache: false,
- // }),
- ],
- css: {
- preprocessorOptions: {
- scss: {
- // additionalData: '@import "@/assets/style/main.scss";',
+ // 构建配置
+ build: {
+ outDir: 'dist',
+ chunkSizeWarningLimit: 1500,
+ rollupOptions: {
+ output: {
+ entryFileNames: `assets/[name].${new Date().getTime()}.js`,
+ chunkFileNames: `assets/[name].${new Date().getTime()}.js`,
+ assetFileNames: `assets/[name].${new Date().getTime()}.[ext]`,
+ compact: true,
+ // manualChunks: {
+ // vue: ['vue', 'vue-router', 'vuex'],
+ // echarts: ['echarts'],
+ // },
},
},
},
+ // 插件
+ plugins: [uni(), Unocss()],
};
};