@coze-arch/cli 0.0.1-alpha.9f719c → 0.0.1-alpha.a166f2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (136) hide show
  1. package/lib/__templates__/expo/.coze +3 -3
  2. package/lib/__templates__/expo/.cozeproj/scripts/dev_build.sh +46 -0
  3. package/lib/__templates__/expo/.cozeproj/scripts/dev_run.sh +229 -0
  4. package/lib/__templates__/expo/.cozeproj/scripts/prod_build.sh +2 -2
  5. package/lib/__templates__/expo/.cozeproj/scripts/prod_run.sh +3 -4
  6. package/lib/__templates__/expo/.cozeproj/scripts/server_dev_run.sh +46 -0
  7. package/lib/__templates__/expo/README.md +68 -7
  8. package/lib/__templates__/expo/_gitignore +1 -1
  9. package/lib/__templates__/expo/_npmrc +2 -4
  10. package/lib/__templates__/expo/client/app/+not-found.tsx +15 -64
  11. package/lib/__templates__/expo/client/app/_layout.tsx +15 -12
  12. package/lib/__templates__/expo/client/app/index.tsx +1 -0
  13. package/lib/__templates__/expo/client/app.config.ts +76 -0
  14. package/lib/__templates__/expo/client/components/Screen.tsx +1 -17
  15. package/lib/__templates__/expo/client/components/ThemedText.tsx +33 -0
  16. package/lib/__templates__/expo/client/components/ThemedView.tsx +37 -0
  17. package/lib/__templates__/expo/client/constants/theme.ts +116 -67
  18. package/lib/__templates__/expo/client/declarations.d.ts +5 -0
  19. package/lib/__templates__/expo/{eslint.config.mjs → client/eslint.config.mjs} +33 -10
  20. package/lib/__templates__/expo/client/hooks/useColorScheme.tsx +48 -0
  21. package/lib/__templates__/expo/client/hooks/useSafeRouter.ts +152 -0
  22. package/lib/__templates__/expo/client/hooks/useTheme.ts +26 -6
  23. package/lib/__templates__/expo/client/metro.config.js +124 -0
  24. package/lib/__templates__/expo/client/package.json +95 -0
  25. package/lib/__templates__/expo/client/screens/demo/index.tsx +25 -0
  26. package/lib/__templates__/expo/client/screens/demo/styles.ts +28 -0
  27. package/lib/__templates__/expo/client/scripts/install-missing-deps.js +1 -0
  28. package/lib/__templates__/expo/client/tsconfig.json +24 -0
  29. package/lib/__templates__/expo/client/utils/index.ts +22 -0
  30. package/lib/__templates__/expo/eslint-plugins/fontawesome6/index.js +9 -0
  31. package/lib/__templates__/expo/eslint-plugins/fontawesome6/names.js +1889 -0
  32. package/lib/__templates__/expo/eslint-plugins/fontawesome6/rule.js +174 -0
  33. package/lib/__templates__/expo/eslint-plugins/fontawesome6/v5-only-names.js +388 -0
  34. package/lib/__templates__/expo/eslint-plugins/react-native/index.js +9 -0
  35. package/lib/__templates__/expo/eslint-plugins/react-native/rule.js +64 -0
  36. package/lib/__templates__/expo/eslint-plugins/reanimated/index.js +9 -0
  37. package/lib/__templates__/expo/eslint-plugins/reanimated/rule.js +88 -0
  38. package/lib/__templates__/expo/package.json +16 -107
  39. package/lib/__templates__/expo/patches/expo@54.0.33.patch +45 -0
  40. package/lib/__templates__/expo/pnpm-lock.yaml +1437 -3171
  41. package/lib/__templates__/expo/pnpm-workspace.yaml +3 -0
  42. package/lib/__templates__/expo/server/build.js +21 -0
  43. package/lib/__templates__/expo/server/package.json +34 -0
  44. package/lib/__templates__/expo/server/src/index.ts +20 -0
  45. package/lib/__templates__/expo/server/tsconfig.json +24 -0
  46. package/lib/__templates__/expo/template.config.js +57 -0
  47. package/lib/__templates__/expo/tsconfig.json +1 -24
  48. package/lib/__templates__/native-static/.coze +11 -0
  49. package/lib/__templates__/native-static/index.html +33 -0
  50. package/lib/__templates__/native-static/styles/main.css +136 -0
  51. package/lib/__templates__/native-static/template.config.js +22 -0
  52. package/lib/__templates__/nextjs/.babelrc +15 -0
  53. package/lib/__templates__/nextjs/.coze +1 -0
  54. package/lib/__templates__/nextjs/_npmrc +1 -0
  55. package/lib/__templates__/nextjs/next.config.ts +12 -0
  56. package/lib/__templates__/nextjs/package.json +13 -2
  57. package/lib/__templates__/nextjs/pnpm-lock.yaml +2682 -1786
  58. package/lib/__templates__/nextjs/scripts/prepare.sh +9 -0
  59. package/lib/__templates__/nextjs/src/app/globals.css +109 -89
  60. package/lib/__templates__/nextjs/src/app/layout.tsx +5 -14
  61. package/lib/__templates__/nextjs/src/app/page.tsx +18 -48
  62. package/lib/__templates__/nextjs/src/components/ui/resizable.tsx +29 -22
  63. package/lib/__templates__/nextjs/src/components/ui/sidebar.tsx +228 -230
  64. package/lib/__templates__/nextjs/template.config.js +67 -2
  65. package/lib/__templates__/taro/.coze +14 -0
  66. package/lib/__templates__/taro/.cozeproj/scripts/deploy_build.sh +19 -0
  67. package/lib/__templates__/taro/.cozeproj/scripts/deploy_run.sh +14 -0
  68. package/lib/__templates__/taro/.cozeproj/scripts/dev_build.sh +2 -0
  69. package/lib/__templates__/taro/.cozeproj/scripts/dev_run.sh +151 -0
  70. package/lib/__templates__/taro/.cozeproj/scripts/init_env.sh +5 -0
  71. package/lib/__templates__/taro/.cozeproj/scripts/pack.sh +24 -0
  72. package/lib/__templates__/taro/README.md +749 -0
  73. package/lib/__templates__/taro/_gitignore +40 -0
  74. package/lib/__templates__/taro/_npmrc +18 -0
  75. package/lib/__templates__/taro/babel.config.js +12 -0
  76. package/lib/__templates__/taro/config/dev.ts +9 -0
  77. package/lib/__templates__/taro/config/index.ts +173 -0
  78. package/lib/__templates__/taro/config/prod.ts +35 -0
  79. package/lib/__templates__/taro/eslint.config.mjs +79 -0
  80. package/lib/__templates__/taro/key/private.appid.key +0 -0
  81. package/lib/__templates__/taro/package.json +97 -0
  82. package/lib/__templates__/taro/pnpm-lock.yaml +22694 -0
  83. package/lib/__templates__/taro/pnpm-workspace.yaml +2 -0
  84. package/lib/__templates__/taro/project.config.json +15 -0
  85. package/lib/__templates__/taro/server/nest-cli.json +10 -0
  86. package/lib/__templates__/taro/server/package.json +40 -0
  87. package/lib/__templates__/taro/server/src/app.controller.ts +23 -0
  88. package/lib/__templates__/taro/server/src/app.module.ts +10 -0
  89. package/lib/__templates__/taro/server/src/app.service.ts +8 -0
  90. package/lib/__templates__/taro/server/src/interceptors/http-status.interceptor.ts +23 -0
  91. package/lib/__templates__/taro/server/src/main.ts +49 -0
  92. package/lib/__templates__/taro/server/tsconfig.json +24 -0
  93. package/lib/__templates__/taro/src/app.config.ts +11 -0
  94. package/lib/__templates__/taro/src/app.css +52 -0
  95. package/lib/__templates__/taro/src/app.tsx +9 -0
  96. package/lib/__templates__/taro/src/index.html +39 -0
  97. package/lib/__templates__/taro/src/network.ts +39 -0
  98. package/lib/__templates__/taro/src/pages/index/index.config.ts +3 -0
  99. package/lib/__templates__/taro/src/pages/index/index.css +1 -0
  100. package/lib/__templates__/taro/src/pages/index/index.tsx +33 -0
  101. package/lib/__templates__/taro/src/presets/h5-navbar.tsx +171 -0
  102. package/lib/__templates__/taro/src/presets/h5-styles.ts +33 -0
  103. package/lib/__templates__/taro/src/presets/index.tsx +18 -0
  104. package/lib/__templates__/taro/src/presets/wx-debug.ts +23 -0
  105. package/lib/__templates__/taro/stylelint.config.mjs +4 -0
  106. package/lib/__templates__/taro/template.config.js +68 -0
  107. package/lib/__templates__/taro/tsconfig.json +29 -0
  108. package/lib/__templates__/taro/types/global.d.ts +32 -0
  109. package/lib/__templates__/templates.json +104 -43
  110. package/lib/__templates__/vite/.coze +1 -0
  111. package/lib/__templates__/vite/_npmrc +1 -0
  112. package/lib/__templates__/vite/eslint.config.mjs +9 -0
  113. package/lib/__templates__/vite/package.json +14 -1
  114. package/lib/__templates__/vite/pnpm-lock.yaml +1581 -105
  115. package/lib/__templates__/vite/scripts/prepare.sh +9 -0
  116. package/lib/__templates__/vite/src/main.ts +17 -48
  117. package/lib/__templates__/vite/template.config.js +67 -6
  118. package/lib/__templates__/vite/vite.config.ts +1 -0
  119. package/lib/cli.js +1006 -154
  120. package/package.json +9 -3
  121. package/lib/__templates__/expo/.cozeproj/scripts/deploy_build.sh +0 -109
  122. package/lib/__templates__/expo/.cozeproj/scripts/deploy_run.sh +0 -232
  123. package/lib/__templates__/expo/app.json +0 -63
  124. package/lib/__templates__/expo/babel.config.js +0 -19
  125. package/lib/__templates__/expo/client/app/(tabs)/_layout.tsx +0 -43
  126. package/lib/__templates__/expo/client/app/(tabs)/home.tsx +0 -1
  127. package/lib/__templates__/expo/client/app/(tabs)/index.tsx +0 -7
  128. package/lib/__templates__/expo/client/hooks/useColorScheme.ts +0 -1
  129. package/lib/__templates__/expo/client/index.js +0 -12
  130. package/lib/__templates__/expo/client/screens/home/index.tsx +0 -51
  131. package/lib/__templates__/expo/client/screens/home/styles.ts +0 -60
  132. package/lib/__templates__/expo/metro.config.js +0 -51
  133. package/lib/__templates__/expo/src/index.ts +0 -12
  134. package/lib/__templates__/nextjs/.vscode/settings.json +0 -121
  135. package/lib/__templates__/vite/.vscode/settings.json +0 -7
  136. /package/lib/__templates__/expo/{eslint-formatter-simple.mjs → client/eslint-formatter-simple.mjs} +0 -0
@@ -0,0 +1,2 @@
1
+ packages:
2
+ - 'server'
@@ -0,0 +1,15 @@
1
+ {
2
+ "miniprogramRoot": "./dist",
3
+ "projectname": "<%= appName %>",
4
+ "description": "test",
5
+ "appid": "touristappid",
6
+ "setting": {
7
+ "urlCheck": true,
8
+ "es6": false,
9
+ "enhance": false,
10
+ "compileHotReLoad": false,
11
+ "postcss": false,
12
+ "minified": false
13
+ },
14
+ "compileType": "miniprogram"
15
+ }
@@ -0,0 +1,10 @@
1
+ {
2
+ "$schema": "https://json.schemastore.org/nest-cli",
3
+ "collection": "@nestjs/schematics",
4
+ "sourceRoot": "src",
5
+ "compilerOptions": {
6
+ "deleteOutDir": true,
7
+ "exclude": ["node_modules", "dist", ".git", "../dist", "../src"]
8
+ },
9
+ "webpack": true
10
+ }
@@ -0,0 +1,40 @@
1
+ {
2
+ "name": "server",
3
+ "version": "1.0.0",
4
+ "private": true,
5
+ "description": "NestJS server application",
6
+ "scripts": {
7
+ "build": "nest build",
8
+ "dev": "nest start --watch",
9
+ "start": "nest start",
10
+ "start:debug": "nest start --debug --watch",
11
+ "start:prod": "node dist/main"
12
+ },
13
+ "dependencies": {
14
+ "@aws-sdk/client-s3": "^3.958.0",
15
+ "@aws-sdk/lib-storage": "^3.958.0",
16
+ "@nestjs/common": "^10.4.15",
17
+ "@nestjs/core": "^10.4.15",
18
+ "@nestjs/platform-express": "^10.4.15",
19
+ "@supabase/supabase-js": "2.95.3",
20
+ "better-sqlite3": "^11.9.1",
21
+ "coze-coding-dev-sdk": "^0.7.16",
22
+ "dotenv": "^17.2.3",
23
+ "drizzle-kit": "^0.31.8",
24
+ "drizzle-orm": "^0.45.1",
25
+ "drizzle-zod": "^0.8.3",
26
+ "express": "5.2.1",
27
+ "pg": "^8.16.3",
28
+ "rxjs": "^7.8.1",
29
+ "zod": "^4.3.5"
30
+ },
31
+ "devDependencies": {
32
+ "@nestjs/cli": "^10.4.9",
33
+ "@nestjs/schematics": "^10.2.3",
34
+ "@types/better-sqlite3": "^7.6.13",
35
+ "@types/express": "5.0.6",
36
+ "@types/node": "^22.10.2",
37
+ "drizzle-kit": "^0.31.8",
38
+ "typescript": "^5.7.2"
39
+ }
40
+ }
@@ -0,0 +1,23 @@
1
+ import { Controller, Get } from '@nestjs/common';
2
+ import { AppService } from '@/app.service';
3
+
4
+ @Controller()
5
+ export class AppController {
6
+ constructor(private readonly appService: AppService) {}
7
+
8
+ @Get('hello')
9
+ getHello(): { status: string; data: string } {
10
+ return {
11
+ status: 'success',
12
+ data: this.appService.getHello()
13
+ };
14
+ }
15
+
16
+ @Get('health')
17
+ getHealth(): { status: string; data: string } {
18
+ return {
19
+ status: 'success',
20
+ data: new Date().toISOString(),
21
+ };
22
+ }
23
+ }
@@ -0,0 +1,10 @@
1
+ import { Module } from '@nestjs/common';
2
+ import { AppController } from '@/app.controller';
3
+ import { AppService } from '@/app.service';
4
+
5
+ @Module({
6
+ imports: [],
7
+ controllers: [AppController],
8
+ providers: [AppService],
9
+ })
10
+ export class AppModule {}
@@ -0,0 +1,8 @@
1
+ import { Injectable } from '@nestjs/common';
2
+
3
+ @Injectable()
4
+ export class AppService {
5
+ getHello(): string {
6
+ return 'Hello, welcome to coze coding mini-program server!';
7
+ }
8
+ }
@@ -0,0 +1,23 @@
1
+ import {
2
+ Injectable,
3
+ NestInterceptor,
4
+ ExecutionContext,
5
+ CallHandler,
6
+ } from '@nestjs/common';
7
+ import { Observable } from 'rxjs';
8
+ import { map } from 'rxjs/operators';
9
+
10
+ @Injectable()
11
+ export class HttpStatusInterceptor implements NestInterceptor {
12
+ intercept(context: ExecutionContext, next: CallHandler): Observable<any> {
13
+ const request = context.switchToHttp().getRequest();
14
+ const response = context.switchToHttp().getResponse();
15
+
16
+ // 如果是 POST 请求且状态码为 201,改为 200
17
+ if (request.method === 'POST' && response.statusCode === 201) {
18
+ response.status(200);
19
+ }
20
+
21
+ return next.handle();
22
+ }
23
+ }
@@ -0,0 +1,49 @@
1
+ import { NestFactory } from '@nestjs/core';
2
+ import { AppModule } from '@/app.module';
3
+ import * as express from 'express';
4
+ import { HttpStatusInterceptor } from '@/interceptors/http-status.interceptor';
5
+
6
+ function parsePort(): number {
7
+ const args = process.argv.slice(2);
8
+ const portIndex = args.indexOf('-p');
9
+ if (portIndex !== -1 && args[portIndex + 1]) {
10
+ const port = parseInt(args[portIndex + 1], 10);
11
+ if (!isNaN(port) && port > 0 && port < 65536) {
12
+ return port;
13
+ }
14
+ }
15
+ return <%= serverPort %>;
16
+ }
17
+
18
+ async function bootstrap() {
19
+ const app = await NestFactory.create(AppModule);
20
+
21
+ app.enableCors({
22
+ origin: true,
23
+ credentials: true,
24
+ });
25
+ app.setGlobalPrefix('api');
26
+ app.use(express.json({ limit: '50mb' }));
27
+ app.use(express.urlencoded({ limit: '50mb', extended: true }));
28
+
29
+ // 全局拦截器:统一将 POST 请求的 201 状态码改为 200
30
+ app.useGlobalInterceptors(new HttpStatusInterceptor());
31
+ // 1. 开启优雅关闭 Hooks (关键!)
32
+ app.enableShutdownHooks();
33
+
34
+ // 2. 解析端口
35
+ const port = parsePort();
36
+ try {
37
+ await app.listen(port);
38
+ console.log(`Server running on http://localhost:${port}`);
39
+ } catch (err) {
40
+ if (err.code === 'EADDRINUSE') {
41
+ console.error(`❌ 端口 \({port} 被占用! 请运行 'npx kill-port \){port}' 然后重试。`);
42
+ process.exit(1);
43
+ } else {
44
+ throw err;
45
+ }
46
+ }
47
+ console.log(`Application is running on: http://localhost:<%= serverPort %>`);
48
+ }
49
+ bootstrap();
@@ -0,0 +1,24 @@
1
+ {
2
+ "compilerOptions": {
3
+ "module": "commonjs",
4
+ "declaration": true,
5
+ "removeComments": true,
6
+ "emitDecoratorMetadata": true,
7
+ "experimentalDecorators": true,
8
+ "allowSyntheticDefaultImports": true,
9
+ "target": "ES2021",
10
+ "sourceMap": true,
11
+ "outDir": "./dist",
12
+ "baseUrl": "./",
13
+ "incremental": true,
14
+ "skipLibCheck": true,
15
+ "strictNullChecks": true,
16
+ "noImplicitAny": false,
17
+ "strictBindCallApply": false,
18
+ "forceConsistentCasingInFileNames": false,
19
+ "noFallthroughCasesInSwitch": false,
20
+ "paths": {
21
+ "@/*": ["./src/*"]
22
+ }
23
+ }
24
+ }
@@ -0,0 +1,11 @@
1
+ export default defineAppConfig({
2
+ pages: [
3
+ 'pages/index/index'
4
+ ],
5
+ window: {
6
+ backgroundTextStyle: 'light',
7
+ navigationBarBackgroundColor: '#fff',
8
+ navigationBarTitleText: 'WeChat',
9
+ navigationBarTextStyle: 'black'
10
+ }
11
+ })
@@ -0,0 +1,52 @@
1
+ @import url('tailwindcss');
2
+
3
+ /*
4
+ * H5 端 rem 适配:与小程序 rpx 缩放一致
5
+ * 375px 屏幕:1rem = 16px,小程序 32rpx = 16px
6
+ */
7
+ html {
8
+ font-size: 4vw !important;
9
+ }
10
+
11
+ /* 小程序页面容器高度设置,确保垂直居中生效 */
12
+ /* stylelint-disable-next-line selector-type-no-unknown */
13
+ page {
14
+ height: 100%;
15
+ }
16
+
17
+ /* H5 端组件默认样式修复 */
18
+ taro-view-core {
19
+ display: block;
20
+ }
21
+
22
+ taro-text-core {
23
+ display: inline;
24
+ }
25
+
26
+ taro-input-core {
27
+ display: block;
28
+ width: 100%;
29
+ }
30
+
31
+ taro-input-core input {
32
+ width: 100%;
33
+ background: transparent;
34
+ border: none;
35
+ outline: none;
36
+ }
37
+
38
+ /* 全局按钮样式重置 */
39
+ taro-button-core,
40
+ button {
41
+ margin: 0 !important;
42
+ padding: 0 !important;
43
+ line-height: inherit;
44
+ display: flex;
45
+ align-items: center;
46
+ justify-content: center;
47
+ }
48
+
49
+ taro-button-core::after,
50
+ button::after {
51
+ border: none;
52
+ }
@@ -0,0 +1,9 @@
1
+ import { PropsWithChildren } from 'react';
2
+ import '@/app.css';
3
+ import { Preset } from './presets';
4
+
5
+ const App = ({ children }: PropsWithChildren) => {
6
+ return <Preset>{children}</Preset>;
7
+ };
8
+
9
+ export default App;
@@ -0,0 +1,39 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+
4
+ <head>
5
+ <meta content="text/html; charset=utf-8" http-equiv="Content-Type">
6
+ <meta content="width=device-width,initial-scale=1,user-scalable=no" name="viewport">
7
+ <meta name="apple-mobile-web-app-capable" content="yes">
8
+ <meta name="apple-touch-fullscreen" content="yes">
9
+ <meta name="format-detection" content="telephone=no,address=no">
10
+ <meta name="apple-mobile-web-app-status-bar-style" content="white">
11
+ <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
12
+ <title><%= appName %></title>
13
+ <script><%%= htmlWebpackPlugin.options.script %></script>
14
+ </head>
15
+
16
+ <body>
17
+ <div id="app">
18
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="lab(2.86037 0.455312 0.568903)"
19
+ stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="
20
+ position: fixed;
21
+ inset: 0;
22
+ margin: auto;
23
+ width: 24px;
24
+ height: 24px;
25
+ animation: __app-loading-spin 1s linear infinite;
26
+ ">
27
+ <path d="M21 12a9 9 0 1 1-6.219-8.56" />
28
+ </svg>
29
+ <style>
30
+ @keyframes __app-loading-spin {
31
+ to {
32
+ transform: rotate(360deg);
33
+ }
34
+ }
35
+ </style>
36
+ </div>
37
+ </body>
38
+
39
+ </html>
@@ -0,0 +1,39 @@
1
+ import Taro from '@tarojs/taro'
2
+
3
+ /**
4
+ * 网络请求模块
5
+ * 封装 Taro.request、Taro.uploadFile、Taro.downloadFile,自动添加项目域名前缀
6
+ * 如果请求的 url 以 http:// 或 https:// 开头,则不会添加域名前缀
7
+ *
8
+ * IMPORTANT: 项目已经全局注入 PROJECT_DOMAIN
9
+ * IMPORTANT: 除非你需要添加全局参数,如给所有请求加上 header,否则不能修改此文件
10
+ */
11
+ export namespace Network {
12
+ const createUrl = (url: string): string => {
13
+ if (url.startsWith('http://') || url.startsWith('https://')) {
14
+ return url
15
+ }
16
+ return `${PROJECT_DOMAIN}${url}`
17
+ }
18
+
19
+ export const request: typeof Taro.request = option => {
20
+ return Taro.request({
21
+ ...option,
22
+ url: createUrl(option.url),
23
+ })
24
+ }
25
+
26
+ export const uploadFile: typeof Taro.uploadFile = option => {
27
+ return Taro.uploadFile({
28
+ ...option,
29
+ url: createUrl(option.url),
30
+ })
31
+ }
32
+
33
+ export const downloadFile: typeof Taro.downloadFile = option => {
34
+ return Taro.downloadFile({
35
+ ...option,
36
+ url: createUrl(option.url),
37
+ })
38
+ }
39
+ }
@@ -0,0 +1,3 @@
1
+ export default definePageConfig({
2
+ navigationBarTitleText: '首页'
3
+ })
@@ -0,0 +1 @@
1
+ /* 优先使用 tailwindcss,如无必要请不要使用css */
@@ -0,0 +1,33 @@
1
+ import { View, Text, Image } from '@tarojs/components';
2
+ import { useLoad } from '@tarojs/taro';
3
+ import { Network } from '@/network';
4
+ import './index.css';
5
+
6
+ /**
7
+ * 默认首页,直接覆盖本页内容
8
+ */
9
+ const IndexPage = () => {
10
+ useLoad(async () => {
11
+ const res = await Network.request({ url: '/api/hello' });
12
+ console.log(res.data);
13
+ });
14
+
15
+ return (
16
+ <View className="w-full h-full flex flex-col justify-center items-center gap-1">
17
+ <Image
18
+ className="w-32 h-28"
19
+ src="https://lf-coze-web-cdn.coze.cn/obj/eden-cn/lm-lgvj/ljhwZthlaukjlkulzlp/coze-coding/icon/coze-coding.gif"
20
+ />
21
+ <View className="self-stretch flex flex-col justify-start items-start gap-2">
22
+ <Text className="self-stretch text-center justify-start text-base-accent-foreground text-base font-bold">
23
+ 应用开发中
24
+ </Text>
25
+ <Text className="self-stretch text-center justify-start text-base-muted-foreground text-sm font-normal">
26
+ 请稍候,界面即将呈现
27
+ </Text>
28
+ </View>
29
+ </View>
30
+ );
31
+ };
32
+
33
+ export default IndexPage;
@@ -0,0 +1,171 @@
1
+ import { View, Text } from '@tarojs/components';
2
+ import Taro, { useDidShow, usePageScroll } from '@tarojs/taro';
3
+ import { useState, useEffect, PropsWithChildren, useCallback } from 'react';
4
+ import { ChevronLeft, House } from 'lucide-react-taro';
5
+
6
+ interface NavConfig {
7
+ navigationBarTitleText?: string;
8
+ navigationBarBackgroundColor?: string;
9
+ navigationBarTextStyle?: 'black' | 'white';
10
+ navigationStyle?: 'default' | 'custom';
11
+ transparentTitle?: 'none' | 'always' | 'auto';
12
+ }
13
+
14
+ enum LeftIcon {
15
+ Back = 'back',
16
+ Home = 'home',
17
+ None = 'none',
18
+ }
19
+
20
+ interface NavState {
21
+ visible: boolean;
22
+ title: string;
23
+ bgColor: string;
24
+ textStyle: 'black' | 'white';
25
+ navStyle: 'default' | 'custom';
26
+ transparent: 'none' | 'always' | 'auto';
27
+ leftIcon: LeftIcon;
28
+ }
29
+
30
+ const DEFAULT_NAV_STATE: NavState = {
31
+ visible: false,
32
+ title: '',
33
+ bgColor: '#ffffff',
34
+ textStyle: 'black',
35
+ navStyle: 'default',
36
+ transparent: 'none',
37
+ leftIcon: LeftIcon.None,
38
+ };
39
+
40
+ const getTabBarPages = (): Set<string> => {
41
+ const tabBar = Taro.getApp()?.config?.tabBar;
42
+ return new Set(
43
+ tabBar?.list?.map((item: { pagePath: string }) => item.pagePath) || [],
44
+ );
45
+ };
46
+
47
+ const computeLeftIcon = (
48
+ route: string,
49
+ tabBarPages: Set<string>,
50
+ historyLength: number,
51
+ ): LeftIcon => {
52
+ if (!route) return LeftIcon.None;
53
+
54
+ const isHomePage =
55
+ route === 'pages/index/index' || route === '/pages/index/index';
56
+ const isTabBarPage = tabBarPages.has(route);
57
+ const hasHistory = historyLength > 1;
58
+
59
+ if (isTabBarPage || isHomePage) return LeftIcon.None;
60
+ if (hasHistory) return LeftIcon.Back;
61
+ return LeftIcon.Home;
62
+ };
63
+
64
+ export const H5NavBar = ({ children }: PropsWithChildren) => {
65
+ const [navState, setNavState] = useState<NavState>(DEFAULT_NAV_STATE);
66
+ const [scrollOpacity, setScrollOpacity] = useState(0);
67
+
68
+ const updateNavState = useCallback(() => {
69
+ const pages = Taro.getCurrentPages();
70
+ const currentPage = pages[pages.length - 1];
71
+ const route = currentPage?.route || '';
72
+ const config: NavConfig = (currentPage as any)?.config || {};
73
+ const tabBarPages = getTabBarPages();
74
+
75
+ const isSinglePageApp = tabBarPages.size <= 1 && pages.length <= 1;
76
+
77
+ setNavState({
78
+ visible: !isSinglePageApp,
79
+ title: config.navigationBarTitleText || '',
80
+ bgColor: config.navigationBarBackgroundColor || '#ffffff',
81
+ textStyle: config.navigationBarTextStyle || 'black',
82
+ navStyle: config.navigationStyle || 'default',
83
+ transparent: config.transparentTitle || 'none',
84
+ leftIcon: isSinglePageApp
85
+ ? LeftIcon.None
86
+ : computeLeftIcon(route, tabBarPages, pages.length),
87
+ });
88
+ }, []);
89
+
90
+ useDidShow(() => {
91
+ updateNavState();
92
+ });
93
+
94
+ usePageScroll(({ scrollTop }) => {
95
+ if (navState.transparent === 'auto') {
96
+ setScrollOpacity(Math.min(scrollTop / 100, 1));
97
+ }
98
+ });
99
+
100
+ useEffect(() => {
101
+ if (TARO_ENV !== 'h5') return;
102
+
103
+ const titleEl = document.querySelector('title') || document.head;
104
+ const observer = new MutationObserver(() => updateNavState());
105
+ observer.observe(titleEl, {
106
+ subtree: true,
107
+ childList: true,
108
+ characterData: true,
109
+ });
110
+
111
+ return () => observer.disconnect();
112
+ }, [updateNavState]);
113
+
114
+ if (
115
+ TARO_ENV !== 'h5' ||
116
+ navState.navStyle === 'custom' ||
117
+ !navState.visible
118
+ ) {
119
+ return <>{children}</>;
120
+ }
121
+
122
+ const iconColor = navState.textStyle === 'white' ? '#fff' : '#333';
123
+ const textColorClass =
124
+ navState.textStyle === 'white' ? 'text-white' : 'text-gray-800';
125
+
126
+ const getBgStyle = () => {
127
+ if (navState.transparent === 'always') {
128
+ return { backgroundColor: 'transparent' };
129
+ }
130
+ if (navState.transparent === 'auto') {
131
+ return { backgroundColor: navState.bgColor, opacity: scrollOpacity };
132
+ }
133
+ return { backgroundColor: navState.bgColor };
134
+ };
135
+
136
+ const handleBack = () => Taro.navigateBack();
137
+ const handleGoHome = () => Taro.switchTab({ url: '/pages/index/index' });
138
+
139
+ return (
140
+ <View className="flex flex-col h-full">
141
+ <View
142
+ className={`fixed top-0 left-0 right-0 h-11 flex items-center justify-center z-1000 ${navState.transparent === 'none' ? 'border-b border-gray-200' : ''}`}
143
+ style={getBgStyle()}
144
+ >
145
+ {navState.leftIcon === LeftIcon.Back && (
146
+ <View
147
+ className="absolute left-2 top-1/2 -translate-y-1/2 p-1 flex items-center justify-center"
148
+ onClick={handleBack}
149
+ >
150
+ <ChevronLeft size={24} color={iconColor} />
151
+ </View>
152
+ )}
153
+ {navState.leftIcon === LeftIcon.Home && (
154
+ <View
155
+ className="absolute left-2 top-1/2 -translate-y-1/2 p-1 flex items-center justify-center"
156
+ onClick={handleGoHome}
157
+ >
158
+ <House size={22} color={iconColor} />
159
+ </View>
160
+ )}
161
+ <Text
162
+ className={`text-base font-medium max-w-3/5 truncate ${textColorClass}`}
163
+ >
164
+ {navState.title}
165
+ </Text>
166
+ </View>
167
+ <View className="h-11 shrink-0" />
168
+ <View className="pb-11 h-full">{children}</View>
169
+ </View>
170
+ );
171
+ };
@@ -0,0 +1,33 @@
1
+ /**
2
+ * H5 端特殊样式注入
3
+ * 如无必要,请勿修改本文件
4
+ */
5
+ export function injectH5Styles() {
6
+ if (TARO_ENV !== 'h5') return;
7
+
8
+ const style = document.createElement('style');
9
+ style.innerHTML = `
10
+ /* H5 端隐藏 TabBar 空图标(只隐藏没有 src 的图标) */
11
+ .weui-tabbar__icon:not([src]),
12
+ .weui-tabbar__icon[src=''] {
13
+ display: none !important;
14
+ }
15
+
16
+ .weui-tabbar__item:has(.weui-tabbar__icon:not([src])) .weui-tabbar__label,
17
+ .weui-tabbar__item:has(.weui-tabbar__icon[src='']) .weui-tabbar__label {
18
+ margin-top: 0 !important;
19
+ }
20
+
21
+ /* Vite 错误覆盖层无法选择文本的问题 */
22
+ vite-error-overlay {
23
+ /* stylelint-disable-next-line property-no-vendor-prefix */
24
+ -webkit-user-select: text !important;
25
+ }
26
+
27
+ vite-error-overlay::part(window) {
28
+ max-width: 90vw;
29
+ padding: 10px;
30
+ }
31
+ `;
32
+ document.head.appendChild(style);
33
+ }
@@ -0,0 +1,18 @@
1
+ import { useLaunch } from '@tarojs/taro';
2
+ import { PropsWithChildren } from 'react';
3
+ import { H5NavBar } from './h5-navbar';
4
+ import { injectH5Styles } from './h5-styles';
5
+ import { enableWxDebugIfNeeded } from './wx-debug';
6
+
7
+ export const Preset = ({ children }: PropsWithChildren) => {
8
+ useLaunch(() => {
9
+ enableWxDebugIfNeeded();
10
+ injectH5Styles();
11
+ });
12
+
13
+ if (TARO_ENV === 'h5') {
14
+ return <H5NavBar>{children}</H5NavBar>;
15
+ }
16
+
17
+ return <>{children}</>;
18
+ };
@@ -0,0 +1,23 @@
1
+ import Taro from '@tarojs/taro'
2
+
3
+ /**
4
+ * 微信小程序调试工具
5
+ * 在开发版/体验版自动开启调试模式
6
+ */
7
+ export function enableWxDebugIfNeeded() {
8
+ // 仅在微信小程序环境执行
9
+ if (Taro.getEnv() === Taro.ENV_TYPE.WEAPP) {
10
+ try {
11
+ const accountInfo = Taro.getAccountInfoSync()
12
+ const envVersion = accountInfo.miniProgram.envVersion
13
+ console.log('[Debug] envVersion:', envVersion)
14
+
15
+ // 开发版/体验版自动开启调试
16
+ if (envVersion !== 'release') {
17
+ Taro.setEnableDebug({ enableDebug: true })
18
+ }
19
+ } catch (error) {
20
+ console.error('[Debug] 开启调试模式失败:', error)
21
+ }
22
+ }
23
+ }
@@ -0,0 +1,4 @@
1
+ /** @type {import('stylelint').Config} */
2
+ export default {
3
+ extends: "stylelint-config-standard",
4
+ };