feat: 新增source目录用于存放配套资源与多端代码

This commit is contained in:
augushong
2026-02-01 10:51:01 +08:00
parent 5bbf69125c
commit ae6b3f1b67
23 changed files with 14683 additions and 0 deletions

29
source/README.md Normal file
View File

@@ -0,0 +1,29 @@
# source/
本目录用于收纳“配套资源与多端代码”,不影响现有 PHP/ThinkPHP 主工程运行与发布。根目录仍是后端工程的唯一入口。
## 目录约定
- `source/assets/`:通用静态资源(模板、字体、设计稿导出等)
- `source/assets/templates/`
- `source/assets/fonts/`
- `source/docker/`:附属服务/可选服务编排与开发辅助(与根目录 `docker/` 区分)
- `source/clients/`:各端前端/客户端工程
- `source/clients/uniapp/`uni-appVue3 + Vite + JavaScript
- `source/clients/vue-screen/`:预留“大屏端”目录
## 命名规则
- 统一小写,使用 `-` 连接(如 `vue-screen`
- 目录按 `clients / assets / docker` 分类
## 依赖与构建产物
- 每个子工程自带依赖清单(例如 `package.json`
- 禁止提交依赖目录与构建产物(例如 `node_modules/``dist/`
## 安全约束
- 禁止在该目录提交任何密钥/证书/私有 token
- 如需环境变量,提交 `.env.example`,不要提交真实 `.env`
-

1
source/assets/.gitkeep Normal file
View File

@@ -0,0 +1 @@

21
source/clients/uniapp/.gitignore vendored Normal file
View File

@@ -0,0 +1,21 @@
# Logs
logs
*.log
npm-debug.log*
yarn-debug.log*
yarn-error.log*
pnpm-debug.log*
lerna-debug.log*
node_modules
.DS_Store
dist
*.local
# Editor directories and files
.idea
*.suo
*.ntvs*
*.njsproj
*.sln
*.sw?

View File

@@ -0,0 +1,20 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<script>
var coverSupport = 'CSS' in window && typeof CSS.supports === 'function' && (CSS.supports('top: env(a)') ||
CSS.supports('top: constant(a)'))
document.write(
'<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0' +
(coverSupport ? ', viewport-fit=cover' : '') + '" />')
</script>
<title></title>
<!--preload-links-->
<!--app-context-->
</head>
<body>
<div id="app"><!--app-html--></div>
<script type="module" src="/src/main.js"></script>
</body>
</html>

13915
source/clients/uniapp/package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,70 @@
{
"name": "uni-preset-vue",
"version": "0.0.0",
"scripts": {
"dev:custom": "uni -p",
"dev:h5": "uni",
"dev:h5:ssr": "uni --ssr",
"dev:mp-alipay": "uni -p mp-alipay",
"dev:mp-baidu": "uni -p mp-baidu",
"dev:mp-jd": "uni -p mp-jd",
"dev:mp-kuaishou": "uni -p mp-kuaishou",
"dev:mp-lark": "uni -p mp-lark",
"dev:mp-qq": "uni -p mp-qq",
"dev:mp-toutiao": "uni -p mp-toutiao",
"dev:mp-harmony": "uni -p mp-harmony",
"dev:mp-weixin": "uni -p mp-weixin",
"dev:mp-xhs": "uni -p mp-xhs",
"dev:quickapp-webview": "uni -p quickapp-webview",
"dev:quickapp-webview-huawei": "uni -p quickapp-webview-huawei",
"dev:quickapp-webview-union": "uni -p quickapp-webview-union",
"build:custom": "uni build -p",
"build:h5": "uni build",
"build:h5:ssr": "uni build --ssr",
"build:mp-alipay": "uni build -p mp-alipay",
"build:mp-baidu": "uni build -p mp-baidu",
"build:mp-jd": "uni build -p mp-jd",
"build:mp-kuaishou": "uni build -p mp-kuaishou",
"build:mp-lark": "uni build -p mp-lark",
"build:mp-qq": "uni build -p mp-qq",
"build:mp-toutiao": "uni build -p mp-toutiao",
"build:mp-harmony": "uni build -p mp-harmony",
"build:mp-weixin": "uni build -p mp-weixin",
"build:mp-xhs": "uni build -p mp-xhs",
"build:quickapp-webview": "uni build -p quickapp-webview",
"build:quickapp-webview-huawei": "uni build -p quickapp-webview-huawei",
"build:quickapp-webview-union": "uni build -p quickapp-webview-union"
},
"dependencies": {
"@dcloudio/uni-app": "3.0.0-4080720251210001",
"@dcloudio/uni-app-harmony": "3.0.0-4080720251210001",
"@dcloudio/uni-app-plus": "3.0.0-4080720251210001",
"@dcloudio/uni-components": "3.0.0-4080720251210001",
"@dcloudio/uni-h5": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-alipay": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-baidu": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-harmony": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-jd": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-kuaishou": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-lark": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-qq": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-toutiao": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-weixin": "3.0.0-4080720251210001",
"@dcloudio/uni-mp-xhs": "3.0.0-4080720251210001",
"@dcloudio/uni-quickapp-webview": "3.0.0-4080720251210001",
"uview-plus": "^3.3.65",
"ului.css": "0.0.1",
"vue": "^3.4.21",
"vue-i18n": "^9.1.9"
},
"devDependencies": {
"@dcloudio/types": "^3.4.8",
"@dcloudio/uni-automator": "3.0.0-4080720251210001",
"@dcloudio/uni-cli-shared": "3.0.0-4080720251210001",
"@dcloudio/uni-stacktracey": "3.0.0-4080720251210001",
"@dcloudio/vite-plugin-uni": "3.0.0-4080720251210001",
"@vue/runtime-core": "^3.4.21",
"sass": "^1.77.8",
"vite": "5.2.8"
}
}

10
source/clients/uniapp/shims-uni.d.ts vendored Normal file
View File

@@ -0,0 +1,10 @@
/// <reference types='@dcloudio/types' />
import 'vue'
declare module '@vue/runtime-core' {
type Hooks = App.AppInstance & Page.PageInstance;
interface ComponentCustomOptions extends Hooks {
}
}

View File

@@ -0,0 +1,17 @@
<script>
export default {
onLaunch: function () {
console.log('App Launch')
},
onShow: function () {
console.log('App Show')
},
onHide: function () {
console.log('App Hide')
},
}
</script>
<style lang="scss">
@import "ului.css/public/cdn/ului.css";
</style>

View File

@@ -0,0 +1,13 @@
import {
createSSRApp
} from "vue";
import App from "./App.vue";
import uviewPlus from 'uview-plus'
import './styles/uview.scss'
export function createApp() {
const app = createSSRApp(App);
app.use(uviewPlus);
return {
app,
};
}

View File

@@ -0,0 +1,72 @@
{
"name" : "",
"appid" : "",
"description" : "",
"versionName" : "1.0.0",
"versionCode" : "100",
"transformPx" : false,
/* 5+App */
"app-plus" : {
"usingComponents" : true,
"nvueStyleCompiler" : "uni-app",
"compilerVersion" : 3,
"splashscreen" : {
"alwaysShowBeforeRender" : true,
"waiting" : true,
"autoclose" : true,
"delay" : 0
},
/* */
"modules" : {},
/* */
"distribute" : {
/* android */
"android" : {
"permissions" : [
"<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>",
"<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>",
"<uses-permission android:name=\"android.permission.VIBRATE\"/>",
"<uses-permission android:name=\"android.permission.READ_LOGS\"/>",
"<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>",
"<uses-feature android:name=\"android.hardware.camera.autofocus\"/>",
"<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>",
"<uses-permission android:name=\"android.permission.CAMERA\"/>",
"<uses-permission android:name=\"android.permission.GET_ACCOUNTS\"/>",
"<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>",
"<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>",
"<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>",
"<uses-permission android:name=\"android.permission.FLASHLIGHT\"/>",
"<uses-feature android:name=\"android.hardware.camera\"/>",
"<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>"
]
},
/* ios */
"ios" : {},
/* SDK */
"sdkConfigs" : {}
}
},
/* */
"quickapp" : {},
/* */
"mp-weixin" : {
"appid" : "",
"setting" : {
"urlCheck" : false
},
"usingComponents" : true
},
"mp-alipay" : {
"usingComponents" : true
},
"mp-baidu" : {
"usingComponents" : true
},
"mp-toutiao" : {
"usingComponents" : true
},
"uniStatistics": {
"enable": false
},
"vueVersion" : "3"
}

View File

@@ -0,0 +1,51 @@
{
"easycom": {
"autoscan": true,
"custom": {
"^u--(.*)": "uview-plus/components/u-$1/u-$1.vue",
"^up-(.*)": "uview-plus/components/u-$1/u-$1.vue",
"^u-([^-].*)": "uview-plus/components/u-$1/u-$1.vue"
}
},
"pages": [
{
"path": "pages/index/index",
"style": {
"navigationBarTitleText": "首页"
}
},
{
"path": "pages/mine/index",
"style": {
"navigationBarTitleText": "个人中心"
}
},
{
"path": "pages/login/index",
"style": {
"navigationBarTitleText": "登录"
}
}
],
"globalStyle": {
"navigationBarTextStyle": "black",
"navigationBarTitleText": "Ulthon",
"navigationBarBackgroundColor": "#F8F8F8",
"backgroundColor": "#F8F8F8"
},
"tabBar": {
"color": "#7A7E83",
"selectedColor": "#3c9cff",
"backgroundColor": "#FFFFFF",
"list": [
{
"pagePath": "pages/index/index",
"text": "首页"
},
{
"pagePath": "pages/mine/index",
"text": "我的"
}
]
}
}

View File

@@ -0,0 +1,79 @@
<template>
<view class="page">
<view class="header">
<image class="logo" src="/static/logo.png"></image>
<view class="title">Ulthon Admin</view>
</view>
<view class="card">
<view class="row">
<view class="label">说明</view>
<view class="value">uni-appVue3 + Vite + JS</view>
</view>
<view class="row">
<view class="label">样式</view>
<view class="value">uview-plus + ULUI</view>
</view>
</view>
<up-button type="primary" text="去个人中心" @click="goMine" />
</view>
</template>
<script>
export default {
methods: {
goMine() {
uni.switchTab({ url: '/pages/mine/index' })
},
},
}
</script>
<style>
.page {
padding: 24rpx;
}
.header {
display: flex;
align-items: center;
gap: 16rpx;
margin-bottom: 24rpx;
}
.logo {
height: 72rpx;
width: 72rpx;
}
.title {
font-size: 36rpx;
font-weight: 600;
color: #222;
}
.card {
padding: 20rpx;
background: #fff;
border-radius: 12rpx;
margin-bottom: 24rpx;
}
.row {
display: flex;
align-items: center;
justify-content: space-between;
padding: 12rpx 0;
}
.label {
color: #666;
font-size: 26rpx;
}
.value {
color: #222;
font-size: 26rpx;
}
</style>

View File

@@ -0,0 +1,94 @@
<template>
<view class="page">
<view class="card">
<up-form :model="form" labelPosition="top">
<up-form-item label="用户名">
<up-input v-model="form.username" placeholder="请输入用户名" />
</up-form-item>
<up-form-item label="密码">
<up-input v-model="form.password" type="password" placeholder="请输入密码" />
</up-form-item>
<up-form-item label="验证码">
<up-input v-model="form.captcha" placeholder="如启用验证码请填写" />
</up-form-item>
<up-form-item label="保持登录">
<up-switch v-model="form.keep_login" />
</up-form-item>
</up-form>
</view>
<up-button type="primary" :loading="loading" text="登录" @click="submit" />
</view>
</template>
<script>
import { setToken } from '../../utils/auth'
import { request } from '../../utils/request'
export default {
data() {
return {
loading: false,
form: {
username: '',
password: '',
captcha: '',
keep_login: true,
},
}
},
methods: {
async submit() {
if (!this.form.username || !this.form.password) {
uni.showToast({ title: '请输入用户名和密码', icon: 'none' })
return
}
this.loading = true
try {
const res = await request({
url: '/admin/login/index',
method: 'POST',
data: {
username: this.form.username,
password: this.form.password,
captcha: this.form.captcha,
keep_login: this.form.keep_login ? 1 : 0,
},
})
const token = (res && res.data && res.data.token) || ''
if (!token) {
uni.showToast({ title: '未获取到 token', icon: 'none' })
return
}
setToken(token)
uni.showToast({ title: '登录成功', icon: 'none' })
uni.navigateBack({
delta: 1,
fail: () => {
uni.switchTab({ url: '/pages/mine/index' })
},
})
} finally {
this.loading = false
}
},
},
}
</script>
<style>
.page {
padding: 24rpx;
}
.card {
padding: 20rpx;
background: #fff;
border-radius: 12rpx;
margin-bottom: 24rpx;
}
</style>

View File

@@ -0,0 +1,93 @@
<template>
<view class="page">
<view class="card">
<view class="row">
<view class="label">登录状态</view>
<view class="value">{{ token ? '已登录' : '未登录' }}</view>
</view>
<view class="row">
<view class="label">Token</view>
<view class="value token">{{ token || '-' }}</view>
</view>
</view>
<up-button type="primary" text="重新登录" @click="goLogin" />
<view style="height: 16rpx" />
<up-button type="error" text="退出登录" @click="logout" />
</view>
</template>
<script>
import { clearToken, getToken } from '../../utils/auth'
import { request } from '../../utils/request'
export default {
data() {
return {
token: '',
}
},
onShow() {
this.token = getToken()
if (!this.token) {
this.goLogin()
}
},
methods: {
goLogin() {
uni.navigateTo({ url: '/pages/login/index' })
},
async logout() {
if (!this.token) {
uni.showToast({ title: '当前未登录', icon: 'none' })
return
}
try {
await request({
url: '/admin/login/out',
method: 'POST',
})
} finally {
clearToken()
this.token = ''
uni.showToast({ title: '已退出', icon: 'none' })
uni.switchTab({ url: '/pages/index/index' })
}
},
},
}
</script>
<style>
.page {
padding: 24rpx;
}
.card {
padding: 20rpx;
background: #fff;
border-radius: 12rpx;
margin-bottom: 24rpx;
}
.row {
padding: 12rpx 0;
}
.label {
color: #666;
font-size: 26rpx;
margin-bottom: 8rpx;
}
.value {
color: #222;
font-size: 26rpx;
word-break: break-all;
}
.token {
color: #3c9cff;
}
</style>

View File

@@ -0,0 +1,6 @@
export {};
declare module "vue" {
type Hooks = App.AppInstance & Page.PageInstance;
interface ComponentCustomOptions extends Hooks {}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.9 KiB

View File

@@ -0,0 +1,3 @@
@import "uview-plus/theme.scss";
@import "uview-plus/index.scss";

View File

@@ -0,0 +1,78 @@
/**
* 这里是uni-app内置的常用样式变量
*
* uni-app 官方扩展插件及插件市场https://ext.dcloud.net.cn上很多三方插件均使用了这些样式变量
* 如果你是插件开发者建议你使用scss预处理并在插件代码中直接使用这些变量无需 import 这个文件方便用户通过搭积木的方式开发整体风格一致的App
*
*/
/**
* 如果你是App开发者插件使用者你可以通过修改这些变量来定制自己的插件主题实现自定义主题功能
*
* 如果你的项目同样使用了scss预处理你也可以直接在你的 scss 代码中使用如下变量,同时无需 import 这个文件
*/
/* 颜色变量 */
/* 行为相关颜色 */
$uni-color-primary: #007aff;
$uni-color-success: #4cd964;
$uni-color-warning: #f0ad4e;
$uni-color-error: #dd524d;
/* 文字基本颜色 */
$uni-text-color: #333; // 基本色
$uni-text-color-inverse: #fff; // 反色
$uni-text-color-grey: #999; // 辅助灰色,如加载更多的提示信息
$uni-text-color-placeholder: #808080;
$uni-text-color-disable: #c0c0c0;
/* 背景颜色 */
$uni-bg-color: #fff;
$uni-bg-color-grey: #f8f8f8;
$uni-bg-color-hover: #f1f1f1; // 点击状态颜色
$uni-bg-color-mask: rgba(0, 0, 0, 0.4); // 遮罩颜色
/* 边框颜色 */
$uni-border-color: #c8c7cc;
/* 尺寸变量 */
/* 文字尺寸 */
$uni-font-size-sm: 12px;
$uni-font-size-base: 14px;
$uni-font-size-lg: 16;
/* 图片尺寸 */
$uni-img-size-sm: 20px;
$uni-img-size-base: 26px;
$uni-img-size-lg: 40px;
/* Border Radius */
$uni-border-radius-sm: 2px;
$uni-border-radius-base: 3px;
$uni-border-radius-lg: 6px;
$uni-border-radius-circle: 50%;
/* 水平间距 */
$uni-spacing-row-sm: 5px;
$uni-spacing-row-base: 10px;
$uni-spacing-row-lg: 15px;
/* 垂直间距 */
$uni-spacing-col-sm: 4px;
$uni-spacing-col-base: 8px;
$uni-spacing-col-lg: 12px;
/* 透明度 */
$uni-opacity-disabled: 0.3; // 组件禁用态的透明度
/* 文章场景相关 */
$uni-color-title: #2c405a; // 文章标题颜色
$uni-font-size-title: 20px;
$uni-color-subtitle: #555; // 二级标题颜色
$uni-font-size-subtitle: 18px;
$uni-color-paragraph: #3f536e; // 文章段落颜色
$uni-font-size-paragraph: 15px;
@import "uview-plus/theme.scss";

View File

@@ -0,0 +1,18 @@
const TOKEN_KEY = 'ULTHON_ADMIN_TOKEN'
export function getToken() {
return uni.getStorageSync(TOKEN_KEY) || ''
}
export function setToken(token) {
uni.setStorageSync(TOKEN_KEY, token || '')
}
export function clearToken() {
uni.removeStorageSync(TOKEN_KEY)
}
export function isLoggedIn() {
return !!getToken()
}

View File

@@ -0,0 +1,83 @@
import { clearToken, getToken } from './auth'
const BASE_URL = ''
function buildUrl(url) {
if (!url) return ''
if (/^https?:\/\//i.test(url)) return url
if (url.startsWith('/')) return `${BASE_URL}${url}`
return `${BASE_URL}/${url}`
}
function isLoginError(resData) {
const msg = (resData && resData.msg) || ''
const url = (resData && resData.url) || ''
return msg.includes('请先登录') || msg.includes('登录已过期') || url.includes('/admin/login')
}
export function request(options) {
const url = buildUrl(options.url)
const method = (options.method || 'GET').toUpperCase()
const header = {
Accept: 'application/json',
...(options.header || {}),
}
const token = getToken()
if (token && !header.Authorization) {
header.Authorization = `Bearer ${token}`
}
if (!header['content-type'] && method !== 'GET') {
header['content-type'] = 'application/x-www-form-urlencoded'
}
return new Promise((resolve, reject) => {
uni.request({
url,
method,
data: options.data || {},
header,
timeout: options.timeout || 60000,
success: (res) => {
const resData = res.data
if (!res || typeof res !== 'object') {
uni.showToast({ title: '请求失败', icon: 'none' })
reject(new Error('Invalid response'))
return
}
if (res.statusCode && res.statusCode >= 400) {
uni.showToast({ title: `请求错误(${res.statusCode})`, icon: 'none' })
reject(new Error(`HTTP ${res.statusCode}`))
return
}
if (resData && typeof resData === 'object' && 'code' in resData) {
if (resData.code === 0) {
resolve(resData)
return
}
if (isLoginError(resData)) {
clearToken()
uni.navigateTo({ url: '/pages/login/index' })
}
uni.showToast({ title: resData.msg || '请求失败', icon: 'none' })
reject(resData)
return
}
resolve(resData)
},
fail: (err) => {
uni.showToast({ title: '网络异常', icon: 'none' })
reject(err)
},
})
})
}

View File

@@ -0,0 +1,8 @@
import { defineConfig } from 'vite'
import uni from '@dcloudio/vite-plugin-uni'
// https://vitejs.dev/config/
export default defineConfig({
plugins: [
uni(),
],
})

View File

@@ -0,0 +1 @@

1
source/docker/.gitkeep Normal file
View File

@@ -0,0 +1 @@