@coze-arch/cli 0.0.1-alpha.dffbaa → 0.0.1-alpha.f9be02

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 (56) hide show
  1. package/lib/__templates__/expo/.coze +7 -2
  2. package/lib/__templates__/expo/.cozeproj/scripts/{deploy_build.sh → dev_build.sh} +0 -7
  3. package/lib/__templates__/expo/.cozeproj/scripts/{deploy_run.sh → dev_run.sh} +22 -22
  4. package/lib/__templates__/expo/.cozeproj/scripts/prod_build.sh +47 -0
  5. package/lib/__templates__/expo/.cozeproj/scripts/prod_run.sh +34 -0
  6. package/lib/__templates__/expo/_gitignore +1 -1
  7. package/lib/__templates__/expo/_npmrc +2 -4
  8. package/lib/__templates__/expo/client/app.config.ts +71 -0
  9. package/lib/__templates__/expo/client/metro.config.js +121 -0
  10. package/lib/__templates__/expo/client/package.json +92 -0
  11. package/lib/__templates__/expo/client/src/app/index.ts +1 -0
  12. package/lib/__templates__/expo/client/src/components/ThemedText.tsx +33 -0
  13. package/lib/__templates__/expo/client/src/components/ThemedView.tsx +38 -0
  14. package/lib/__templates__/expo/client/src/constants/theme.ts +850 -0
  15. package/lib/__templates__/expo/client/{hooks → src/hooks}/useTheme.ts +1 -1
  16. package/lib/__templates__/expo/client/{screens → src/screens}/home/index.tsx +0 -1
  17. package/lib/__templates__/expo/client/tsconfig.json +24 -0
  18. package/lib/__templates__/expo/package.json +16 -12
  19. package/lib/__templates__/expo/pnpm-lock.yaml +299 -514
  20. package/lib/__templates__/expo/pnpm-workspace.yaml +3 -0
  21. package/lib/__templates__/expo/server/package.json +17 -0
  22. package/lib/__templates__/expo/tsconfig.json +1 -24
  23. package/lib/__templates__/nextjs/src/app/globals.css +99 -87
  24. package/lib/__templates__/vite/package.json +1 -1
  25. package/lib/__templates__/vite/pnpm-lock.yaml +120 -120
  26. package/lib/cli.js +1 -1
  27. package/package.json +5 -2
  28. package/lib/__templates__/expo/app.json +0 -63
  29. package/lib/__templates__/expo/babel.config.js +0 -9
  30. package/lib/__templates__/expo/client/app/(tabs)/_layout.tsx +0 -43
  31. package/lib/__templates__/expo/client/app/(tabs)/home.tsx +0 -1
  32. package/lib/__templates__/expo/client/app/(tabs)/index.tsx +0 -7
  33. package/lib/__templates__/expo/client/app/+not-found.tsx +0 -79
  34. package/lib/__templates__/expo/client/constants/theme.ts +0 -118
  35. package/lib/__templates__/expo/client/index.js +0 -12
  36. package/lib/__templates__/expo/metro.config.js +0 -53
  37. /package/lib/__templates__/expo/{eslint-formatter-simple.mjs → client/eslint-formatter-simple.mjs} +0 -0
  38. /package/lib/__templates__/expo/{eslint.config.mjs → client/eslint.config.mjs} +0 -0
  39. /package/lib/__templates__/expo/client/{app → src/app}/_layout.tsx +0 -0
  40. /package/lib/__templates__/expo/client/{assets → src/assets}/fonts/SpaceMono-Regular.ttf +0 -0
  41. /package/lib/__templates__/expo/client/{assets → src/assets}/images/adaptive-icon.png +0 -0
  42. /package/lib/__templates__/expo/client/{assets → src/assets}/images/default-avatar.png +0 -0
  43. /package/lib/__templates__/expo/client/{assets → src/assets}/images/favicon.png +0 -0
  44. /package/lib/__templates__/expo/client/{assets → src/assets}/images/icon.png +0 -0
  45. /package/lib/__templates__/expo/client/{assets → src/assets}/images/partial-react-logo.png +0 -0
  46. /package/lib/__templates__/expo/client/{assets → src/assets}/images/react-logo.png +0 -0
  47. /package/lib/__templates__/expo/client/{assets → src/assets}/images/react-logo@2x.png +0 -0
  48. /package/lib/__templates__/expo/client/{assets → src/assets}/images/react-logo@3x.png +0 -0
  49. /package/lib/__templates__/expo/client/{assets → src/assets}/images/splash-icon.png +0 -0
  50. /package/lib/__templates__/expo/client/{components → src/components}/Screen.tsx +0 -0
  51. /package/lib/__templates__/expo/client/{components → src/components}/SmartDateInput.tsx +0 -0
  52. /package/lib/__templates__/expo/client/{contexts → src/contexts}/AuthContext.tsx +0 -0
  53. /package/lib/__templates__/expo/client/{hooks → src/hooks}/useColorScheme.ts +0 -0
  54. /package/lib/__templates__/expo/client/{screens → src/screens}/home/styles.ts +0 -0
  55. /package/lib/__templates__/expo/client/{utils → src/utils}/index.ts +0 -0
  56. /package/lib/__templates__/expo/{src → server/src}/index.ts +0 -0
@@ -3,5 +3,10 @@ entrypoint = "server.js"
3
3
  requires = ["nodejs-24"]
4
4
 
5
5
  [dev]
6
- build = ["bash", ".cozeproj/scripts/deploy_build.sh"]
7
- run = ["bash", ".cozeproj/scripts/deploy_run.sh"]
6
+ build = ["bash", ".cozeproj/scripts/dev_build.sh"]
7
+ run = ["bash", ".cozeproj/scripts/dev_run.sh"]
8
+
9
+ [deploy]
10
+ build = ["bash", ".cozeproj/scripts/prod_build.sh"]
11
+ run = ["bash", ".cozeproj/scripts/prod_run.sh"]
12
+ build_app_dir = "."
@@ -49,13 +49,6 @@ fs.writeSync(fd, record+"\n", null, "utf8");
49
49
  fs.fsyncSync(fd);
50
50
  fs.closeSync(fd);
51
51
  ' "$LOG_FILE" "$level" "$msg"
52
-
53
- case "$level" in
54
- INFO) info "$msg" ;;
55
- WARN) warn "$msg" ;;
56
- ERROR) echo -e "\033[31m[ERROR] $msg\033[0m" ;;
57
- *) info "$msg" ;;
58
- esac
59
52
  }
60
53
 
61
54
  # ==================== 前置检查 ====================
@@ -1,4 +1,5 @@
1
1
  ROOT_DIR="$(cd "$(dirname "$0")" && pwd)"
2
+ PREVIEW_DIR="${COZE_PREVIEW_DIR:-$ROOT_DIR}"
2
3
  LOG_DIR="${COZE_LOG_DIR:-$ROOT_DIR/logs}"
3
4
  LOG_FILE="$LOG_DIR/app.log"
4
5
  mkdir -p "$LOG_DIR"
@@ -10,7 +11,7 @@ SERVER_PORT="9091"
10
11
  # Expo 项目配置
11
12
  EXPO_HOST="0.0.0.0"
12
13
  EXPO_DIR="expo"
13
- EXPO_PORT="9090"
14
+ EXPO_PORT="5000"
14
15
  WEB_URL="${COZE_PROJECT_DOMAIN_DEFAULT:-http://127.0.0.1:${SERVER_PORT}}"
15
16
  ASSUME_YES="1"
16
17
  EXPO_PUBLIC_BACKEND_BASE_URL="${EXPO_PUBLIC_BACKEND_BASE_URL:-$WEB_URL}"
@@ -116,18 +117,12 @@ fs.writeSync(fd, record+"\n", null, "utf8");
116
117
  fs.fsyncSync(fd);
117
118
  fs.closeSync(fd);
118
119
  ' "$LOG_FILE" "$level" "$msg"
119
- case "$level" in
120
- INFO) echo -e "\033[32m[INFO] $msg\033[0m" ;;
121
- WARN) echo -e "\033[33m[WARN] $msg\033[0m" ;;
122
- ERROR) echo -e "\033[31m[ERROR] $msg\033[0m" ;;
123
- *) echo -e "\033[32m[INFO] $msg\033[0m" ;;
124
- esac
125
120
  }
126
121
 
127
122
  wait_port_connectable() {
128
123
  local host=$1 port=$2 retries=${3:-10}
129
124
  for _ in $(seq 1 "$retries"); do
130
- nc -z "$host" "$port" && return 0
125
+ nc -z -w 1 "$host" "$port" >/dev/null 2>&1 && return 0
131
126
  sleep 1
132
127
  done
133
128
  return 1
@@ -135,13 +130,18 @@ wait_port_connectable() {
135
130
 
136
131
  start_expo() {
137
132
  local offline="${1:-0}"
133
+
134
+ pushd ./client
135
+
138
136
  if [ "$offline" = "1" ]; then
139
- EXPO_OFFLINE=1 EXPO_NO_DOCTOR=1 EXPO_PUBLIC_BACKEND_BASE_URL="$EXPO_PUBLIC_BACKEND_BASE_URL" EXPO_PACKAGER_PROXY_URL="$EXPO_PACKAGER_PROXY_URL" nohup npx expo start --port "$EXPO_PORT" > logs/expo.log 2>&1 &
137
+ EXPO_OFFLINE=1 EXPO_NO_DOCTOR=1 EXPO_PUBLIC_BACKEND_BASE_URL="$EXPO_PUBLIC_BACKEND_BASE_URL" EXPO_PACKAGER_PROXY_URL="$EXPO_PACKAGER_PROXY_URL" nohup npx expo start --port "$EXPO_PORT" > ../logs/expo.log 2>&1 &
140
138
  else
141
- EXPO_NO_DOCTOR=1 EXPO_PUBLIC_BACKEND_BASE_URL="$EXPO_PUBLIC_BACKEND_BASE_URL" EXPO_PACKAGER_PROXY_URL="$EXPO_PACKAGER_PROXY_URL" nohup npx expo start --port "$EXPO_PORT" > logs/expo.log 2>&1 &
139
+ EXPO_NO_DOCTOR=1 EXPO_PUBLIC_BACKEND_BASE_URL="$EXPO_PUBLIC_BACKEND_BASE_URL" EXPO_PACKAGER_PROXY_URL="$EXPO_PACKAGER_PROXY_URL" nohup npx expo start --port "$EXPO_PORT" > ../logs/expo.log 2>&1 &
142
140
  fi
143
141
  EXPO_PID=$!
144
142
  echo "$EXPO_PID"
143
+
144
+ popd
145
145
  }
146
146
 
147
147
  detect_expo_fetch_failed() {
@@ -159,6 +159,15 @@ detect_expo_fetch_failed() {
159
159
  }
160
160
 
161
161
  # ==================== 前置检查 ====================
162
+ # 关掉nginx进程
163
+ ps -ef | grep nginx | grep -v grep | awk '{print $2}' | xargs -r kill -9
164
+
165
+ write_log "INFO" "检查根目录 pre_install.py"
166
+ if [ -f "$PREVIEW_DIR/pre_install.py" ]; then
167
+ write_log "INFO" "执行:python $PREVIEW_DIR/pre_install.py"
168
+ python "$PREVIEW_DIR/pre_install.py" || write_log "ERROR" "pre_install.py 执行失败"
169
+ fi
170
+
162
171
  write_log "INFO" "==================== 开始启动 ===================="
163
172
  write_log "INFO" "开始执行服务启动脚本(start_dev.sh)..."
164
173
  write_log "INFO" "正在检查依赖命令和目录是否存在..."
@@ -176,17 +185,9 @@ ensure_port SERVER_PORT "$SERVER_PORT"
176
185
  ensure_port EXPO_PORT "$EXPO_PORT"
177
186
 
178
187
  # ==================== 启动 Server 服务 ====================
179
- write_log "INFO" "检查 Nginx 端口 (5000)..."
180
- if is_port_free 5000; then
181
- write_log "INFO" "端口 5000 未被占用,正在启动 Nginx..."
182
- service nginx start
183
- else
184
- write_log "INFO" "端口 5000 已被占用,跳过 Nginx 启动"
185
- fi
186
-
187
188
  write_log "INFO" "==================== 启动 server 服务 ===================="
188
- write_log "INFO" "正在执行:npm run server"
189
- PORT="$SERVER_PORT" nohup npm run server > logs/app.log 2>&1 &
189
+ write_log "INFO" "正在执行:npm run dev"
190
+ PORT="$SERVER_PORT" nohup npm run dev --prefix ./server > logs/app.log 2>&1 &
190
191
  SERVER_PID=$!
191
192
  if [ -z "${SERVER_PID}" ]; then
192
193
  write_log "ERROR" "无法获取 server 后台进程 PID"
@@ -194,8 +195,7 @@ fi
194
195
  write_log "INFO" "server 服务已启动,进程 ID:${SERVER_PID:-unknown}"
195
196
 
196
197
  write_log "INFO" "==================== 启动 Expo 项目 ===================="
197
- write_log "INFO" "正在执行:npx expo start --port ${EXPO_PORT}"
198
- write_log "INFO" "开始启动 Expo 服务"
198
+ write_log "INFO" "开始启动 Expo 服务,端口 ${EXPO_PORT}"
199
199
  EXPO_PID=$(start_expo 0)
200
200
  if detect_expo_fetch_failed 8; then
201
201
  write_log "WARN" "Expo 启动检测到网络错误:TypeError: fetch failed,启用离线模式重试"
@@ -0,0 +1,47 @@
1
+ #!/bin/bash
2
+ if [ -z "${BASH_VERSION:-}" ]; then exec /usr/bin/env bash "$0" "$@"; fi
3
+ set -euo pipefail
4
+ ROOT_DIR="$(pwd)"
5
+
6
+ # ==================== 工具函数 ====================
7
+ info() {
8
+ echo "[INFO] $1"
9
+ }
10
+ warn() {
11
+ echo "[WARN] $1"
12
+ }
13
+ error() {
14
+ echo "[ERROR] $1"
15
+ exit 1
16
+ }
17
+ check_command() {
18
+ if ! command -v "$1" &> /dev/null; then
19
+ error "命令 $1 未找到,请先安装"
20
+ fi
21
+ }
22
+
23
+ info "==================== 开始构建 ===================="
24
+ info "开始执行构建脚本(build_prod.sh)..."
25
+ info "正在检查依赖命令是否存在..."
26
+ # 检查核心命令
27
+ check_command "pnpm"
28
+ check_command "npm"
29
+
30
+ # ==================== 安装 Node 依赖 ====================
31
+ info "==================== 安装 Node 依赖 ===================="
32
+ info "开始安装 Node 依赖"
33
+ if [ -f "$ROOT_DIR/package.json" ]; then
34
+ info "进入目录:$ROOT_DIR"
35
+ info "正在执行:pnpm install"
36
+ (cd "$ROOT_DIR" && pnpm install --registry=https://registry.npmmirror.com) || error "Node 依赖安装失败"
37
+ else
38
+ warn "未找到 $ROOT_DIR/package.json 文件,请检查路径是否正确"
39
+ fi
40
+ info "==================== 依赖安装完成!====================\n"
41
+
42
+ info "==================== dist打包 ===================="
43
+ info "开始执行:npm run build --prefix server"
44
+ (cd "$ROOT_DIR" && npm run build --prefix ./server) || error "dist打包失败"
45
+ info "==================== dist打包完成!====================\n"
46
+
47
+ info "下一步:执行 ./prod_run.sh 启动服务"
@@ -0,0 +1,34 @@
1
+ #!/usr/bin/env bash
2
+ # 产物部署使用
3
+ set -euo pipefail
4
+
5
+ ROOT_DIR="$(pwd)"
6
+
7
+ HOST="${HOST:-0.0.0.0}"
8
+ PORT="${PORT:-5000}"
9
+
10
+ # ==================== 工具函数 ====================
11
+ info() {
12
+ echo "[INFO] $1"
13
+ }
14
+ warn() {
15
+ echo "[WARN] $1"
16
+ }
17
+ error() {
18
+ echo "[ERROR] $1"
19
+ exit 1
20
+ }
21
+ check_command() {
22
+ if ! command -v "$1" &> /dev/null; then
23
+ error "命令 $1 未找到,请先安装"
24
+ fi
25
+ }
26
+
27
+ # ============== 启动服务 ======================
28
+ # 检查核心命令
29
+ check_command "pnpm"
30
+ check_command "npm"
31
+
32
+ info "开始执行:npm run start"
33
+ (cd "$ROOT_DIR" && PORT="$PORT" npm run start --prefix ./server) || error "服务启动失败"
34
+ info "服务启动完成!\n"
@@ -8,4 +8,4 @@ expo/expo-env.d.ts
8
8
  expo-env.d.ts
9
9
  .DS_Store
10
10
  logs/
11
- tsconfig.tsbuildinfo
11
+ *.tsbuildinfo
@@ -8,15 +8,13 @@ network-concurrency=16
8
8
  fetch-retries=3
9
9
  fetch-timeout=60000
10
10
 
11
- # 严格使用 peer dependencies
11
+ # peerDependencies
12
12
  strict-peer-dependencies=false
13
-
14
- # 自动生成 lockfile
15
13
  auto-install-peers=true
16
14
 
17
15
  # lockfile 配置
18
16
  lockfile=true
19
17
  prefer-frozen-lockfile=true
20
18
 
21
- # 如果 lockfile 存在但过期,更新而不是失败
19
+ # semver 选择最高版本
22
20
  resolution-mode=highest
@@ -0,0 +1,71 @@
1
+ import { ExpoConfig, ConfigContext } from 'expo/config';
2
+
3
+ export default ({ config }: ConfigContext): ExpoConfig => {
4
+ return {
5
+ ...config,
6
+ "name": "${app_name}",
7
+ "slug": "${slug}",
8
+ "version": "1.0.0",
9
+ "orientation": "portrait",
10
+ "icon": "./assets/images/icon.png",
11
+ "scheme": "myapp",
12
+ "userInterfaceStyle": "automatic",
13
+ "newArchEnabled": true,
14
+ "ios": {
15
+ "supportsTablet": true
16
+ },
17
+ "android": {
18
+ "adaptiveIcon": {
19
+ "foregroundImage": "./assets/images/adaptive-icon.png",
20
+ "backgroundColor": "#ffffff"
21
+ }
22
+ },
23
+ "web": {
24
+ "bundler": "metro",
25
+ "output": "single",
26
+ "favicon": "./assets/images/favicon.png"
27
+ },
28
+ "plugins": [
29
+ process.env.EXPO_PUBLIC_BACKEND_BASE_URL ? [
30
+ "expo-router",
31
+ {
32
+ "origin": process.env.EXPO_PUBLIC_BACKEND_BASE_URL
33
+ }
34
+ ] : 'expo-router',
35
+ [
36
+ "expo-splash-screen",
37
+ {
38
+ "image": "./assets/images/splash-icon.png",
39
+ "imageWidth": 200,
40
+ "resizeMode": "contain",
41
+ "backgroundColor": "#ffffff"
42
+ }
43
+ ],
44
+ [
45
+ "expo-image-picker",
46
+ {
47
+ "photosPermission": "允许${app_name}访问您的相册,以便您上传或保存图片。",
48
+ "cameraPermission": "允许${app_name}使用您的相机,以便您直接拍摄照片上传。",
49
+ "microphonePermission": "允许${app_name}访问您的麦克风,以便您拍摄带有声音的视频。"
50
+ }
51
+ ],
52
+ [
53
+ "expo-location",
54
+ {
55
+ "locationWhenInUsePermission": "${app_name}需要访问您的位置以提供周边服务及导航功能。"
56
+ }
57
+ ],
58
+ [
59
+ "expo-camera",
60
+ {
61
+ "cameraPermission": "${app_name}需要访问相机以拍摄照片和视频。",
62
+ "microphonePermission": "${app_name}需要访问麦克风以录制视频声音。",
63
+ "recordAudioAndroid": true
64
+ }
65
+ ]
66
+ ],
67
+ "experiments": {
68
+ "typedRoutes": true
69
+ }
70
+ }
71
+ }
@@ -0,0 +1,121 @@
1
+ const { getDefaultConfig } = require('expo/metro-config');
2
+ const { createProxyMiddleware } = require('http-proxy-middleware');
3
+ const connect = require('connect');
4
+
5
+ const config = getDefaultConfig(__dirname);
6
+
7
+ // 安全地获取 Expo 的默认排除列表
8
+ const existingBlockList = [].concat(config.resolver.blockList || []);
9
+
10
+ config.resolver.blockList = [
11
+ ...existingBlockList,
12
+ /.*\/\.expo\/.*/, // Expo 的缓存和构建产物目录
13
+
14
+ // 1. 原生代码 (Java/C++/Objective-C)
15
+ /.*\/react-native\/ReactAndroid\/.*/,
16
+ /.*\/react-native\/ReactCommon\/.*/,
17
+
18
+ // 2. 纯开发和调试工具
19
+ // 这些工具只在开发电脑上运行,不会被打包到应用中
20
+ /.*\/@typescript-eslint\/eslint-plugin\/.*/,
21
+
22
+ // 3. 构建时数据
23
+ // 这个数据库只在打包过程中使用,应用运行时不需要
24
+ /.*\/caniuse-lite\/data\/.*/,
25
+
26
+ // 4. 通用规则
27
+ /.*\/__tests__\/.*/, // 排除所有测试目录
28
+ /.*\.git\/.*/, // 排除 Git 目录
29
+ ];
30
+
31
+ const BACKEND_TARGET = 'http://localhost:9091';
32
+
33
+ const apiProxy = createProxyMiddleware({
34
+ target: BACKEND_TARGET,
35
+ changeOrigin: true,
36
+ logLevel: 'debug',
37
+ proxyTimeout: 86400000,
38
+ onProxyReq: (proxyReq, req) => {
39
+ const accept = req.headers.accept || '';
40
+ if (accept.includes('text/event-stream')) {
41
+ proxyReq.setHeader('accept-encoding', 'identity');
42
+ }
43
+ },
44
+ onProxyRes: (proxyRes, req, res) => {
45
+ const contentType = proxyRes.headers['content-type'] || '';
46
+ if (contentType.includes('text/event-stream') || contentType.includes('application/stream')) {
47
+ res.setHeader('Cache-Control', 'no-cache');
48
+ res.setHeader('Connection', 'keep-alive');
49
+ res.setHeader('X-Accel-Buffering', 'no');
50
+ if (typeof res.flushHeaders === 'function') {
51
+ try { res.flushHeaders(); } catch {}
52
+ }
53
+ }
54
+ },
55
+ });
56
+
57
+ const streamProxy = createProxyMiddleware({
58
+ target: BACKEND_TARGET,
59
+ changeOrigin: true,
60
+ logLevel: 'debug',
61
+ ws: true,
62
+ proxyTimeout: 86400000,
63
+ onProxyReq: (proxyReq, req) => {
64
+ const upgrade = req.headers.upgrade;
65
+ const accept = req.headers.accept || '';
66
+ if (upgrade && upgrade.toLowerCase() === 'websocket') {
67
+ proxyReq.setHeader('Connection', 'upgrade');
68
+ proxyReq.setHeader('Upgrade', req.headers.upgrade);
69
+ } else if (accept.includes('text/event-stream')) {
70
+ proxyReq.setHeader('accept-encoding', 'identity');
71
+ proxyReq.setHeader('Connection', 'keep-alive');
72
+ }
73
+ },
74
+ onProxyRes: (proxyRes, req, res) => {
75
+ const contentType = proxyRes.headers['content-type'] || '';
76
+ if (contentType.includes('text/event-stream') || contentType.includes('application/stream')) {
77
+ res.setHeader('Cache-Control', 'no-cache');
78
+ res.setHeader('Connection', 'keep-alive');
79
+ res.setHeader('X-Accel-Buffering', 'no');
80
+ if (typeof res.flushHeaders === 'function') {
81
+ try { res.flushHeaders(); } catch {}
82
+ }
83
+ }
84
+ },
85
+ });
86
+
87
+ const shouldProxyToBackend = (url) => {
88
+ if (!url) return false;
89
+ if (/^\/api\/v\d+\//.test(url)) {
90
+ return true;
91
+ }
92
+ return false;
93
+ };
94
+
95
+ const isWebSocketRequest = (req) =>
96
+ !!(req.headers.upgrade && req.headers.upgrade.toLowerCase() === 'websocket');
97
+ const isSSERequest = (req) => {
98
+ const accept = req.headers.accept || '';
99
+ return accept.includes('text/event-stream');
100
+ };
101
+
102
+ config.server = {
103
+ ...config.server,
104
+ enhanceMiddleware: (metroMiddleware, metroServer) => {
105
+ return connect()
106
+ .use((req, res, next) => {
107
+ if (shouldProxyToBackend(req.url)) {
108
+ console.log(`[Metro Proxy] Forwarding ${req.method} ${req.url}`);
109
+
110
+ if (isWebSocketRequest(req) || isSSERequest(req)) {
111
+ return streamProxy(req, res, next);
112
+ }
113
+ return apiProxy(req, res, next);
114
+ }
115
+ next();
116
+ })
117
+ .use(metroMiddleware);
118
+ },
119
+ };
120
+
121
+ module.exports = config;
@@ -0,0 +1,92 @@
1
+ {
2
+ "name": "expo-app",
3
+ "description": "<%= appName %>",
4
+ "main": "expo-router/entry",
5
+ "private": true,
6
+ "scripts": {
7
+ "check-deps": "npx depcheck",
8
+ "postinstall": "npm run install-missing",
9
+ "install-missing": "node ./scripts/install-missing-deps.js",
10
+ "lint": "expo lint",
11
+ "start": "expo start --web --clear",
12
+ "test": "jest --watchAll"
13
+ },
14
+ "jest": {
15
+ "preset": "jest-expo"
16
+ },
17
+ "dependencies": {
18
+ "@expo/metro-runtime": "^6.1.2",
19
+ "@expo/vector-icons": "^15.0.0",
20
+ "@react-native-async-storage/async-storage": "^2.2.0",
21
+ "@react-native-community/datetimepicker": "^8.5.0",
22
+ "@react-native-community/slider": "^5.0.1",
23
+ "@react-native-masked-view/masked-view": "^0.3.2",
24
+ "@react-native-picker/picker": "^2.11.0",
25
+ "@react-navigation/bottom-tabs": "^7.2.0",
26
+ "@react-navigation/native": "^7.0.14",
27
+ "babel-plugin-module-resolver": "^5.0.2",
28
+ "babel-preset-expo": "^54.0.9",
29
+ "dayjs": "^1.11.19",
30
+ "expo": "^54.0.7",
31
+ "expo-auth-session": "^7.0.9",
32
+ "expo-av": "~16.0.6",
33
+ "expo-blur": "~15.0.6",
34
+ "expo-camera": "~17.0.10",
35
+ "expo-constants": "~18.0.8",
36
+ "expo-crypto": "^15.0.7",
37
+ "expo-font": "~14.0.7",
38
+ "expo-haptics": "~15.0.6",
39
+ "expo-image-picker": "~17.0.7",
40
+ "expo-linear-gradient": "~15.0.6",
41
+ "expo-linking": "~8.0.7",
42
+ "expo-location": "~19.0.7",
43
+ "expo-router": "~6.0.0",
44
+ "expo-splash-screen": "~31.0.8",
45
+ "expo-status-bar": "~3.0.7",
46
+ "expo-symbols": "~1.0.6",
47
+ "expo-system-ui": "~6.0.9",
48
+ "expo-web-browser": "~15.0.10",
49
+ "react": "19.1.0",
50
+ "react-dom": "19.1.0",
51
+ "react-native": "0.81.5",
52
+ "react-native-chart-kit": "^6.12.0",
53
+ "react-native-gesture-handler": "~2.28.0",
54
+ "react-native-keyboard-aware-scroll-view": "^0.9.5",
55
+ "react-native-modal-datetime-picker": "18.0.0",
56
+ "react-native-reanimated": "~4.1.0",
57
+ "react-native-safe-area-context": "~5.6.0",
58
+ "react-native-screens": "~4.16.0",
59
+ "react-native-svg": "15.15.0",
60
+ "react-native-toast-message": "^2.3.3",
61
+ "react-native-web": "^0.21.2",
62
+ "react-native-webview": "~13.15.0",
63
+ "react-native-worklets": "0.5.1",
64
+ "zod": "^4.2.1"
65
+ },
66
+ "devDependencies": {
67
+ "@babel/core": "^7.25.2",
68
+ "@eslint/js": "^9.27.0",
69
+ "@types/jest": "^29.5.12",
70
+ "@types/react": "~19.1.0",
71
+ "@types/react-test-renderer": "19.1.0",
72
+ "chalk": "^4.1.2",
73
+ "depcheck": "^1.4.7",
74
+ "esbuild": "0.27.2",
75
+ "eslint": "^9.39.2",
76
+ "eslint-formatter-compact": "^9.0.1",
77
+ "eslint-import-resolver-typescript": "^4.4.4",
78
+ "eslint-plugin-import": "^2.32.0",
79
+ "eslint-plugin-react": "^7.37.5",
80
+ "eslint-plugin-react-hooks": "^7.0.1",
81
+ "eslint-plugin-regexp": "^2.10.0",
82
+ "globals": "^16.1.0",
83
+ "jest": "^29.2.1",
84
+ "jest-expo": "~54.0.10",
85
+ "react-test-renderer": "19.1.0",
86
+ "tsx": "^4.21.0",
87
+ "typescript": "^5.8.3",
88
+ "typescript-eslint": "^8.32.1",
89
+ "connect": "^3.7.0",
90
+ "http-proxy-middleware": "^3.0.5"
91
+ }
92
+ }
@@ -0,0 +1 @@
1
+ export { default } from '@/screens/home'
@@ -0,0 +1,33 @@
1
+ import React from 'react';
2
+ import { Text, TextProps, TextStyle } from 'react-native';
3
+ import { useTheme } from '@/hooks/useTheme';
4
+ import { Typography } from '@/constants/theme';
5
+
6
+ type TypographyVariant = keyof typeof Typography;
7
+
8
+ interface ThemedTextProps extends TextProps {
9
+ variant?: TypographyVariant;
10
+ color?: string;
11
+ }
12
+
13
+ export function ThemedText({
14
+ variant = 'body',
15
+ color,
16
+ style,
17
+ children,
18
+ ...props
19
+ }: ThemedTextProps) {
20
+ const { theme } = useTheme();
21
+ const typographyStyle = Typography[variant];
22
+
23
+ const textStyle: TextStyle = {
24
+ ...typographyStyle,
25
+ color: color ?? theme.textPrimary,
26
+ };
27
+
28
+ return (
29
+ <Text style={[textStyle, style]} {...props}>
30
+ {children}
31
+ </Text>
32
+ );
33
+ }
@@ -0,0 +1,38 @@
1
+ import React from 'react';
2
+ import { View, ViewProps, ViewStyle } from 'react-native';
3
+ import { useTheme } from '@/hooks/useTheme';
4
+
5
+ type BackgroundLevel = 'root' | 'default' | 'secondary' | 'tertiary';
6
+
7
+ interface ThemedViewProps extends ViewProps {
8
+ level?: BackgroundLevel;
9
+ backgroundColor?: string;
10
+ }
11
+
12
+ const backgroundMap: Record<BackgroundLevel, string> = {
13
+ root: 'backgroundRoot',
14
+ default: 'backgroundDefault',
15
+ secondary: 'backgroundSecondary',
16
+ tertiary: 'backgroundTertiary',
17
+ };
18
+
19
+ export function ThemedView({
20
+ level = 'root',
21
+ backgroundColor,
22
+ style,
23
+ children,
24
+ ...props
25
+ }: ThemedViewProps) {
26
+ const { theme } = useTheme();
27
+ const bgColor = backgroundColor ?? (theme as any)[backgroundMap[level]];
28
+
29
+ const viewStyle: ViewStyle = {
30
+ backgroundColor: bgColor,
31
+ };
32
+
33
+ return (
34
+ <View style={[viewStyle, style]} {...props}>
35
+ {children}
36
+ </View>
37
+ );
38
+ }