Compare commits
34 Commits
| Author | SHA1 | Date |
|---|---|---|
|
|
7ec3aa156b | |
|
|
acb6578b0a | |
|
|
644b6f2b15 | |
|
|
ad22c7f4fd | |
|
|
f55007d2d8 | |
|
|
8bddc3b06d | |
|
|
8346c96908 | |
|
|
32e8c8c741 | |
|
|
32cc10ffd5 | |
|
|
f8b47bdf5a | |
|
|
adc7389916 | |
|
|
917f45ec91 | |
|
|
598cb33027 | |
|
|
9345654cc1 | |
|
|
fc49d3b6d7 | |
|
|
8c79fd8ef6 | |
|
|
7ce6057fd5 | |
|
|
d44cfe511a | |
|
|
d31340a05a | |
|
|
1c4f8d1a66 | |
|
|
81dbdf490a | |
|
|
115a003afe | |
|
|
f06257c2d9 | |
|
|
a0e0b75472 | |
|
|
4c8d5606a8 | |
|
|
f5bd20406c | |
|
|
eae75615c0 | |
|
|
937f6994eb | |
|
|
945c7f8530 | |
|
|
74bafd3739 | |
|
|
cc94096dfe | |
|
|
f2439273e5 | |
|
|
58162183e8 | |
|
|
9e92afb24b |
|
|
@ -0,0 +1,120 @@
|
|||
{
|
||||
"permissions": {
|
||||
"allow": [
|
||||
"Bash(python3:*)",
|
||||
"Read(//Users/jettcheng1018/Downloads/**)",
|
||||
"Read(//tmp/**)",
|
||||
"Bash(comm:*)",
|
||||
"Bash(git commit:*)",
|
||||
"Bash(./check_consistency.sh:*)",
|
||||
"Bash(chmod:*)",
|
||||
"Bash(./scripts/maintenance/create_issue.sh:*)",
|
||||
"Bash(./scripts/maintenance/check_issues.sh:*)",
|
||||
"Bash(./check_issues.sh)",
|
||||
"Bash(cat:*)",
|
||||
"Bash(./drama issue)",
|
||||
"Bash(./drama report \"UI設計缺漏嚴重性評估\")",
|
||||
"Bash(./drama compliance)",
|
||||
"Bash(./drama report analysis \"文檔分類組織結構優化\")",
|
||||
"Bash(git push:*)",
|
||||
"Bash(flutter:*)",
|
||||
"Bash(mkdir:*)",
|
||||
"Bash(echo $PATH)",
|
||||
"Bash(/Users/jettcheng1018/flutter/flutter_3.24.5/bin/flutter --version)",
|
||||
"Read(//Users/jettcheng1018/flutter/**)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter --version)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter pub get)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter doctor)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter run -d chrome --web-port=8080)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter clean)",
|
||||
"Read(//Users/jettcheng1018/**)",
|
||||
"Bash(./drama:*)",
|
||||
"Bash(./dl)",
|
||||
"Bash(./dl project list)",
|
||||
"Bash(./dl status)",
|
||||
"Bash(./dl project types)",
|
||||
"Bash(./dl phase status)",
|
||||
"Bash(./dl project help)",
|
||||
"Bash(./dl phase help)",
|
||||
"Bash(./dl phase list)",
|
||||
"Bash(rm:*)",
|
||||
"Bash(git add:*)",
|
||||
"Bash(git rm:*)",
|
||||
"Bash(java:*)",
|
||||
"Bash(brew install:*)",
|
||||
"Bash(sudo ln:*)",
|
||||
"Bash(export:*)",
|
||||
"Bash(./dl issue)",
|
||||
"Bash(xcode-select:*)",
|
||||
"Read(//Applications/**)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter doctor -v)",
|
||||
"Bash(mas account:*)",
|
||||
"Bash(killall:*)",
|
||||
"Bash(open:*)",
|
||||
"Bash($ANDROID_HOME/emulator/emulator:*)",
|
||||
"Bash(curl:*)",
|
||||
"Bash(unzip:*)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter create --platforms android,ios .)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter devices)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter build apk --debug)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter analyze --no-fatal-infos)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter build apk --debug --target-platform android-arm64)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter build apk --debug --target-platform android-arm64 --dry-run)",
|
||||
"Bash(./gradlew:*)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter emulators)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter run -d emulator-5554)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter run -d apple_ios_simulator)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter run -d AF3D76B9-F005-4880-BE7D-25EA8ECD1E4D)",
|
||||
"Bash(./dl report analysis \"詞彙學習關卡系統設計分析\")",
|
||||
"Bash(brew cleanup:*)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter emulators --launch apple_ios_simulator)",
|
||||
"Bash(for ui in \"UI_Cost_Confirm_Popup\" \"UI_Insufficient_Resources\" \"UI_LevelResult_SuccessResult\" \"UI_LifePoints_Display\" \"UI_Shop_Item_Confirm\" \"UI_Subscription_Success\" \"UI_TimeWarp_Cards\")",
|
||||
"Bash(do echo -n \"$ui: \")",
|
||||
"Bash(if grep -q \"$ui\" /tmp/system_ui_list.txt)",
|
||||
"Bash(fi)",
|
||||
"Bash(done)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter run -d 148D878C-62EB-4B60-9C04-2173EC0248BF)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter run -d Medium_Phone_API_36.0)",
|
||||
"Bash(/Users/jettcheng1018/flutter/bin/flutter emulators --launch Medium_Phone_API_36.0)",
|
||||
"Bash(dotnet run:*)",
|
||||
"Bash(dotnet --version)",
|
||||
"Bash(npm install)",
|
||||
"Bash(npm install:*)",
|
||||
"Bash(npm run dev:*)",
|
||||
"Bash(find:*)",
|
||||
"Bash(/bashes)",
|
||||
"Bash(pkill:*)",
|
||||
"Bash(say:*)",
|
||||
"Bash(./dl list)",
|
||||
"Bash(./dl task)",
|
||||
"Bash(./scripts/file_version_manager.sh:*)",
|
||||
"Bash(./scripts/archive_file.sh:*)",
|
||||
"Bash(./scripts/view_archives.sh:*)",
|
||||
"Bash(tree:*)",
|
||||
"Bash(./sop/scripts/archive_file.sh:*)",
|
||||
"Bash(mv:*)",
|
||||
"Bash(./dl report analysis \"CLAUDE.md文件品質改善分析\")",
|
||||
"Bash(./dl report analysis \"文檔結構模板規格設計\")",
|
||||
"Bash(./dl:*)",
|
||||
"Bash(npm run type-check:*)",
|
||||
"Bash(timeout 10 curl -s http://localhost:3000/)",
|
||||
"Bash(./sop/scripts/sop_consistency_check.sh:*)",
|
||||
"Bash(timeout 30 npm run type-check:*)",
|
||||
"Bash(timeout 10 curl -s -I http://localhost:3000/)",
|
||||
"Bash(npm init:*)",
|
||||
"Bash(timeout 5 curl -s -I http://localhost:3000/)",
|
||||
"Bash(lsof:*)",
|
||||
"Bash(npm run build:*)",
|
||||
"Bash(npm run preview:*)",
|
||||
"Bash(sort:*)",
|
||||
"Bash(for:*)",
|
||||
"Bash(grep:*)",
|
||||
"Bash(sed:*)",
|
||||
"Bash(do sed -i '' 's/u2190 u8fd4u56deu5c0eu822a/← 返回導航/g' \"$file\")",
|
||||
"Bash(do)",
|
||||
"Bash(tar:*)"
|
||||
],
|
||||
"deny": [],
|
||||
"ask": []
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,58 @@
|
|||
---
|
||||
name: 🐛 Bug Report
|
||||
about: Create a report to help us improve
|
||||
title: '[BUG] '
|
||||
labels: ['bug', 'needs-triage']
|
||||
assignees: ''
|
||||
---
|
||||
|
||||
## 🐛 Bug Description
|
||||
<!-- A clear and concise description of what the bug is -->
|
||||
|
||||
## 🔄 Steps to Reproduce
|
||||
1. Go to '...'
|
||||
2. Click on '....'
|
||||
3. Scroll down to '....'
|
||||
4. See error
|
||||
|
||||
## ✅ Expected Behavior
|
||||
<!-- A clear and concise description of what you expected to happen -->
|
||||
|
||||
## ❌ Actual Behavior
|
||||
<!-- A clear and concise description of what actually happened -->
|
||||
|
||||
## 📱 Environment
|
||||
### Mobile App (Flutter)
|
||||
- **Platform**: iOS / Android
|
||||
- **Device**: [e.g. iPhone 12, Samsung Galaxy S21]
|
||||
- **OS Version**: [e.g. iOS 15.0, Android 12]
|
||||
- **App Version**: [e.g. 1.2.0]
|
||||
|
||||
### Backend (.NET Core)
|
||||
- **Environment**: Development / Staging / Production
|
||||
- **Server OS**: [if known]
|
||||
- **Database**: [PostgreSQL version]
|
||||
|
||||
## 📸 Screenshots
|
||||
<!-- If applicable, add screenshots to help explain your problem -->
|
||||
|
||||
## 📋 Additional Context
|
||||
<!-- Add any other context about the problem here -->
|
||||
|
||||
## 🔍 Error Logs
|
||||
<!-- If available, add any relevant error logs or stack traces -->
|
||||
|
||||
```
|
||||
Paste error logs here
|
||||
```
|
||||
|
||||
## 🎯 Priority
|
||||
<!-- Mark the priority level -->
|
||||
- [ ] 🔴 Critical (System down, data loss)
|
||||
- [ ] 🟠 High (Major feature broken)
|
||||
- [ ] 🟡 Medium (Minor feature issue)
|
||||
- [ ] 🟢 Low (Cosmetic issue)
|
||||
|
||||
## 🏷️ Labels
|
||||
<!-- The following labels will be automatically applied -->
|
||||
<!-- bug, needs-triage -->
|
||||
|
|
@ -0,0 +1,66 @@
|
|||
---
|
||||
name: ✨ Feature Request
|
||||
about: Suggest an idea for this project
|
||||
title: '[FEATURE] '
|
||||
labels: ['enhancement', 'needs-triage']
|
||||
assignees: ''
|
||||
---
|
||||
|
||||
## 💡 Feature Description
|
||||
<!-- A clear and concise description of the feature you want to see -->
|
||||
|
||||
## 🎯 Problem Statement
|
||||
<!-- What problem does this feature solve? -->
|
||||
**Is your feature request related to a problem?**
|
||||
A clear description of what the problem is. Ex. I'm always frustrated when [...]
|
||||
|
||||
## 🛠 Proposed Solution
|
||||
<!-- Describe the solution you'd like to see -->
|
||||
|
||||
## 🔄 User Stories
|
||||
<!-- Describe how users would interact with this feature -->
|
||||
- As a [user type], I want [goal] so that [benefit]
|
||||
- As a [user type], I want [goal] so that [benefit]
|
||||
|
||||
## 📱 Platform
|
||||
<!-- Which parts of the system would be affected? -->
|
||||
- [ ] 📱 Mobile App (Flutter)
|
||||
- [ ] 🔧 Backend API (.NET Core)
|
||||
- [ ] 🗄️ Database Schema
|
||||
- [ ] 🎮 Gamification System
|
||||
- [ ] 🤖 AI Analysis Engine
|
||||
- [ ] 📊 Analytics/Reporting
|
||||
- [ ] 🔐 Authentication/Security
|
||||
|
||||
## 🎨 UI/UX Considerations
|
||||
<!-- If this affects the UI, describe the expected user experience -->
|
||||
|
||||
## 🔧 Technical Considerations
|
||||
<!-- Any technical implementation details or constraints -->
|
||||
|
||||
## 📈 Success Metrics
|
||||
<!-- How would we measure the success of this feature? -->
|
||||
|
||||
## 🚧 Alternative Solutions
|
||||
<!-- Describe alternatives you've considered -->
|
||||
|
||||
## 📋 Additional Context
|
||||
<!-- Add any other context, mockups, or examples -->
|
||||
|
||||
## 🎯 Priority
|
||||
<!-- Mark the priority level -->
|
||||
- [ ] 🔴 Critical (Essential for launch)
|
||||
- [ ] 🟠 High (Important for user experience)
|
||||
- [ ] 🟡 Medium (Nice to have)
|
||||
- [ ] 🟢 Low (Future consideration)
|
||||
|
||||
## 📅 Timeline
|
||||
<!-- When would you like to see this feature? -->
|
||||
- [ ] Next release
|
||||
- [ ] Within 3 months
|
||||
- [ ] Within 6 months
|
||||
- [ ] Future roadmap
|
||||
|
||||
## 🏷️ Labels
|
||||
<!-- The following labels will be automatically applied -->
|
||||
<!-- enhancement, needs-triage -->
|
||||
|
|
@ -0,0 +1,88 @@
|
|||
# Pull Request
|
||||
|
||||
## 📋 Summary
|
||||
<!-- Provide a brief description of the changes in this PR -->
|
||||
|
||||
## 🎯 Type of Change
|
||||
<!-- Mark the relevant option with an 'x' -->
|
||||
- [ ] 🐛 Bug fix (non-breaking change which fixes an issue)
|
||||
- [ ] ✨ New feature (non-breaking change which adds functionality)
|
||||
- [ ] 💥 Breaking change (fix or feature that would cause existing functionality to not work as expected)
|
||||
- [ ] 📚 Documentation update
|
||||
- [ ] 🏗️ Infrastructure/build changes
|
||||
- [ ] 🧹 Code cleanup/refactoring
|
||||
- [ ] 🧪 Tests only
|
||||
|
||||
## 🔗 Related Issues
|
||||
<!-- Link any related issues using "Fixes #123" or "Related to #123" -->
|
||||
|
||||
## 🛠 Changes Made
|
||||
<!-- Describe the changes made in detail -->
|
||||
|
||||
### Frontend (Flutter)
|
||||
- [ ] UI components updated
|
||||
- [ ] State management changes
|
||||
- [ ] Navigation changes
|
||||
- [ ] New screens/widgets added
|
||||
|
||||
### Backend (.NET Core)
|
||||
- [ ] API endpoints added/modified
|
||||
- [ ] Database schema changes
|
||||
- [ ] Business logic updates
|
||||
- [ ] Authentication/authorization changes
|
||||
|
||||
## 🧪 Testing
|
||||
<!-- Describe the testing done for this change -->
|
||||
|
||||
### Flutter Testing
|
||||
- [ ] Unit tests added/updated
|
||||
- [ ] Widget tests added/updated
|
||||
- [ ] Integration tests added/updated
|
||||
- [ ] Manual testing completed on iOS
|
||||
- [ ] Manual testing completed on Android
|
||||
|
||||
### .NET Testing
|
||||
- [ ] Unit tests added/updated
|
||||
- [ ] Integration tests added/updated
|
||||
- [ ] API testing completed
|
||||
- [ ] Database migration tested
|
||||
|
||||
## 📱 Screenshots/Videos
|
||||
<!-- Add screenshots or videos if applicable -->
|
||||
|
||||
## 📝 Additional Notes
|
||||
<!-- Any additional context, warnings, or notes for reviewers -->
|
||||
|
||||
## ✅ Checklist
|
||||
<!-- Mark completed items with an 'x' -->
|
||||
|
||||
### Code Quality
|
||||
- [ ] Code follows the established coding standards
|
||||
- [ ] Self-review of the code completed
|
||||
- [ ] Code is properly commented (especially complex logic)
|
||||
- [ ] No debugging code or console logs left in
|
||||
- [ ] Error handling is appropriate
|
||||
|
||||
### Documentation
|
||||
- [ ] Documentation updated (if needed)
|
||||
- [ ] API documentation updated (if applicable)
|
||||
- [ ] README updated (if needed)
|
||||
|
||||
### Security & Performance
|
||||
- [ ] No sensitive data exposed in code
|
||||
- [ ] Performance impact considered
|
||||
- [ ] Security implications reviewed
|
||||
- [ ] Accessibility guidelines followed (for UI changes)
|
||||
|
||||
### Testing & Deployment
|
||||
- [ ] All tests pass locally
|
||||
- [ ] CI/CD pipeline passes
|
||||
- [ ] Database migrations work (if applicable)
|
||||
- [ ] Feature works in staging environment
|
||||
|
||||
## 👥 Reviewers
|
||||
<!-- Tag specific people if needed -->
|
||||
@team-leads @senior-developers
|
||||
|
||||
---
|
||||
**Note**: Please ensure all checkboxes are marked before requesting review.
|
||||
|
|
@ -0,0 +1,137 @@
|
|||
name: CI/CD Pipeline
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [ main, develop ]
|
||||
pull_request:
|
||||
branches: [ main, develop ]
|
||||
|
||||
jobs:
|
||||
# Flutter Mobile App CI
|
||||
flutter-test:
|
||||
name: Flutter Tests
|
||||
runs-on: ubuntu-latest
|
||||
defaults:
|
||||
run:
|
||||
working-directory: ./mobile
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Setup Flutter
|
||||
uses: subosito/flutter-action@v2
|
||||
with:
|
||||
flutter-version: '3.16.0'
|
||||
channel: 'stable'
|
||||
|
||||
- name: Get dependencies
|
||||
run: flutter pub get
|
||||
|
||||
- name: Analyze code
|
||||
run: flutter analyze
|
||||
|
||||
- name: Run tests
|
||||
run: flutter test
|
||||
|
||||
- name: Generate coverage
|
||||
run: flutter test --coverage
|
||||
|
||||
- name: Upload coverage to Codecov
|
||||
uses: codecov/codecov-action@v3
|
||||
with:
|
||||
file: ./mobile/coverage/lcov.info
|
||||
|
||||
# .NET Backend API CI
|
||||
dotnet-test:
|
||||
name: .NET Tests
|
||||
runs-on: ubuntu-latest
|
||||
defaults:
|
||||
run:
|
||||
working-directory: ./backend
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Setup .NET
|
||||
uses: actions/setup-dotnet@v3
|
||||
with:
|
||||
dotnet-version: '8.0.x'
|
||||
|
||||
- name: Restore dependencies
|
||||
run: dotnet restore
|
||||
|
||||
- name: Build
|
||||
run: dotnet build --no-restore --configuration Release
|
||||
|
||||
- name: Test
|
||||
run: dotnet test --no-build --configuration Release --verbosity normal --collect:"XPlat Code Coverage"
|
||||
|
||||
- name: Upload coverage to Codecov
|
||||
uses: codecov/codecov-action@v3
|
||||
with:
|
||||
file: ./backend/TestResults/*/coverage.cobertura.xml
|
||||
|
||||
# Security and Quality Checks
|
||||
security-scan:
|
||||
name: Security Scan
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Run Trivy vulnerability scanner
|
||||
uses: aquasecurity/trivy-action@master
|
||||
with:
|
||||
scan-type: 'fs'
|
||||
scan-ref: '.'
|
||||
format: 'sarif'
|
||||
output: 'trivy-results.sarif'
|
||||
|
||||
- name: Upload Trivy scan results to GitHub Security tab
|
||||
uses: github/codeql-action/upload-sarif@v2
|
||||
with:
|
||||
sarif_file: 'trivy-results.sarif'
|
||||
|
||||
# Build and Deploy to Staging (develop branch only)
|
||||
deploy-staging:
|
||||
name: Deploy to Staging
|
||||
needs: [flutter-test, dotnet-test]
|
||||
runs-on: ubuntu-latest
|
||||
if: github.ref == 'refs/heads/develop' && github.event_name == 'push'
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Configure AWS credentials
|
||||
uses: aws-actions/configure-aws-credentials@v2
|
||||
with:
|
||||
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
||||
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
|
||||
aws-region: us-west-2
|
||||
|
||||
- name: Deploy to staging
|
||||
run: |
|
||||
echo "Deploying to staging environment"
|
||||
# Add actual deployment commands here
|
||||
|
||||
# Build and Deploy to Production (main branch only)
|
||||
deploy-production:
|
||||
name: Deploy to Production
|
||||
needs: [flutter-test, dotnet-test, security-scan]
|
||||
runs-on: ubuntu-latest
|
||||
if: github.ref == 'refs/heads/main' && github.event_name == 'push'
|
||||
environment: production
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Configure AWS credentials
|
||||
uses: aws-actions/configure-aws-credentials@v2
|
||||
with:
|
||||
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
|
||||
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
|
||||
aws-region: us-west-2
|
||||
|
||||
- name: Deploy to production
|
||||
run: |
|
||||
echo "Deploying to production environment"
|
||||
# Add actual deployment commands here
|
||||
|
|
@ -483,4 +483,4 @@ secrets/
|
|||
*.pem
|
||||
*.p12
|
||||
*.p8
|
||||
*.mobileprovision
|
||||
*.mobileprovisiondocs/05_views/
|
||||
|
|
|
|||
|
|
@ -0,0 +1,143 @@
|
|||
# 🎯 Drama Ling 前端架構重構執行摘要
|
||||
|
||||
**建立日期**: 2025-09-10
|
||||
**專案負責**: Claude Code
|
||||
**執行狀態**: ⏳ 準備執行
|
||||
**預估工期**: 3-4週
|
||||
|
||||
## 📊 重構決策總覽
|
||||
|
||||
### 🔄 **架構轉換**
|
||||
```
|
||||
Vue 3 + Quasar Framework → 原生 HTML + CSS + JavaScript
|
||||
```
|
||||
|
||||
### 🎯 **核心目標**
|
||||
1. **設計精確度**: 85% → 100%
|
||||
2. **Claude Code相容性**: 80% → 95%
|
||||
3. **載入性能**: 2s → 0.8s
|
||||
4. **Bundle大小**: 800KB → 150KB
|
||||
|
||||
## 🚀 四階段執行計劃
|
||||
|
||||
### **第一階段 (週1) - 基礎架構**
|
||||
```bash
|
||||
apps/web-native/
|
||||
├── assets/css/main.css # 設計系統
|
||||
├── assets/js/app.js # 核心JavaScript
|
||||
└── docs/ARCHITECTURE.md # 架構文檔
|
||||
```
|
||||
|
||||
### **第二階段 (週1) - 核心頁面**
|
||||
```bash
|
||||
pages/
|
||||
├── index.html # 首頁
|
||||
├── auth/login.html # 認證
|
||||
├── vocabulary/index.html # 詞彙學習
|
||||
└── profile/index.html # 個人檔案
|
||||
```
|
||||
|
||||
### **第三階段 (週1) - 功能頁面**
|
||||
```bash
|
||||
pages/vocabulary/
|
||||
├── practice.html # 練習頁面
|
||||
├── review.html # 複習頁面
|
||||
└── analytics.html # 分析儀表板
|
||||
```
|
||||
|
||||
### **第四階段 (週1) - 整合優化**
|
||||
- API整合 + 測試 + 部署
|
||||
|
||||
## 📋 已完成的準備工作
|
||||
|
||||
### ✅ **SOP標準流程執行**
|
||||
- [x] 歸檔舊版技術文檔 (`sop/archive/20250910142112_README.md`)
|
||||
- [x] 創建重構專案文檔 (`projects/native-html-migration.md`)
|
||||
- [x] 更新任務管理系統 (`TASKS.md`)
|
||||
- [x] 更新技術文檔 (`docs/04_technical/README.md`)
|
||||
- [x] 更新功能規格說明 (`docs/02_design/function-specs/web/README.md`)
|
||||
- [x] 產生正式分析報告 (`sop/tools/reports/analysis/2025-09-10_analysis-analysis.md`)
|
||||
|
||||
### 📄 **關鍵文檔建立**
|
||||
| 文檔類型 | 檔案路徑 | 狀態 |
|
||||
|---------|----------|------|
|
||||
| **專案規劃** | `projects/native-html-migration.md` | ✅ 已完成 |
|
||||
| **技術架構** | `docs/04_technical/README.md` | ✅ 已更新 |
|
||||
| **任務管理** | `TASKS.md` | ✅ 已更新 |
|
||||
| **功能規格** | `docs/02_design/function-specs/web/README.md` | ✅ 已更新 |
|
||||
| **分析報告** | `sop/tools/reports/analysis/2025-09-10_analysis-analysis.md` | ✅ 已完成 |
|
||||
|
||||
## 🎯 下一步立即行動
|
||||
|
||||
### 🔥 **緊急任務 (本週開始)**
|
||||
1. **備份現有代碼**
|
||||
```bash
|
||||
# 備份現有Vue版本
|
||||
cp -r apps/web apps/web-vue-backup
|
||||
```
|
||||
|
||||
2. **建立原生HTML專案結構**
|
||||
```bash
|
||||
# 創建新專案目錄
|
||||
mkdir -p apps/web-native/{pages,assets,data,docs}
|
||||
mkdir -p apps/web-native/assets/{css,js,media}
|
||||
```
|
||||
|
||||
3. **建立設計系統基礎**
|
||||
- 創建 `assets/css/main.css` (CSS變數、色彩、字體系統)
|
||||
- 創建 `assets/js/app.js` (核心JavaScript模組)
|
||||
|
||||
### ⚠️ **重要準備工作**
|
||||
- 分析現有Vue組件,列出需要轉換的功能清單
|
||||
- 建立HTML頁面模板和組件系統
|
||||
- 設計JavaScript模組化架構
|
||||
|
||||
### 📝 **一般支援工作**
|
||||
- 準備開發環境配置
|
||||
- 建立測試策略和品質檢查流程
|
||||
|
||||
## 🎪 成功關鍵因素
|
||||
|
||||
### 💡 **技術方案**
|
||||
- **漸進式遷移**: 保留Vue版本作為後備
|
||||
- **功能完整性**: 100%保持現有功能規格
|
||||
- **性能優化**: 原生HTML的性能優勢
|
||||
- **Claude Code友好**: AI開發最佳化
|
||||
|
||||
### 🚀 **執行策略**
|
||||
- **週檢查點**: 每週進度評估和調整
|
||||
- **質量保證**: 像素級設計還原檢查
|
||||
- **用戶驗證**: A/B測試確保體驗不降級
|
||||
|
||||
## ⚠️ 風險控制
|
||||
|
||||
### 🛡️ **風險緩解措施**
|
||||
- **功能回滾**: 保留完整Vue備份
|
||||
- **數據兼容**: API接口保持不變
|
||||
- **分階段部署**: 逐頁面替換降低風險
|
||||
- **性能監控**: 建立性能基準線對比
|
||||
|
||||
## 📞 後續支援
|
||||
|
||||
### 🔄 **定期檢查**
|
||||
- **每週檢查**: 進度和品質評估
|
||||
- **里程碑評估**: 每階段完成度檢查
|
||||
- **最終驗收**: 功能完整性和性能指標驗證
|
||||
|
||||
---
|
||||
|
||||
## 🚀 **準備開始執行?**
|
||||
|
||||
所有準備工作已按照CLAUDE.md SOP標準完成,包括:
|
||||
- ✅ 文件歸檔和版本管理
|
||||
- ✅ 詳細專案規劃和技術文檔
|
||||
- ✅ 任務管理系統更新
|
||||
- ✅ 正式分析報告產生
|
||||
- ✅ 風險評估和緩解策略
|
||||
|
||||
**現在可以開始執行第一階段:基礎架構建立** 🎯
|
||||
|
||||
---
|
||||
|
||||
**文檔更新**: 2025-09-10
|
||||
**相關連結**: [TASKS.md](TASKS.md) | [重構專案](projects/native-html-migration.md) | [分析報告](sop/tools/reports/analysis/2025-09-10_analysis-analysis.md)
|
||||
|
|
@ -0,0 +1,389 @@
|
|||
# 🚀 Drama Ling 企業級UI設計總體計劃
|
||||
|
||||
**建立日期**: 2025-01-15
|
||||
**計劃版本**: v4.0 - 企業級重構
|
||||
**架構基礎**: 共用模組架構 v3.0
|
||||
**設計標準**: 企業級UI/UX規範
|
||||
**執行目標**: 95+ UI畫面完整重設計
|
||||
|
||||
## 📋 計劃概述
|
||||
|
||||
### 🎯 核心目標
|
||||
基於 Drama Ling v3.0 共用模組架構,創建企業級標準的完整UI設計系統,確保:
|
||||
- **100%符合功能規格**: 嚴格按照 `/docs/02_design/function-specs/` 規格執行
|
||||
- **統一設計語言**: 完全遵循 `/docs/02_design/ui-ux/ui-ux-guidelines.md` 規範
|
||||
- **企業級品質**: 達到Fortune 500企業內部系統標準
|
||||
- **零設計債務**: 徹底重構,消除所有設計不一致問題
|
||||
|
||||
### 🏗️ 設計架構原則
|
||||
1. **規格優先**: 所有設計必須100%符合功能規格文件
|
||||
2. **模組化設計**: 基於v3.0共用模組架構的設計組件系統
|
||||
3. **一致性保證**: 跨平台設計語言統一
|
||||
4. **可擴展性**: 支援未來功能快速擴展的設計框架
|
||||
5. **無障礙標準**: 符合WCAG 2.1 AA級無障礙要求
|
||||
|
||||
## 🔍 階段一:設計規範完善與標準化 (第1-2週)
|
||||
|
||||
### 1.1 UI/UX規範補全與更新 ⭐ **CRITICAL**
|
||||
|
||||
**目標**: 建立企業級完整設計規範系統
|
||||
|
||||
**工作內容**:
|
||||
- [ ] **色彩系統完善**
|
||||
- 引用文件: `/docs/02_design/ui-ux/ui-ux-guidelines.md` (第35-103行)
|
||||
- 補全遺失的色彩定義:狀態色彩、反饋色彩、學習進度色彩
|
||||
- 建立暗色/亮色主題完整色彩對照表
|
||||
- 定義色彩使用場景和層次規範
|
||||
|
||||
- [ ] **字體系統標準化**
|
||||
- 引用文件: `/docs/02_design/ui-ux/ui-ux-guidelines.md` (第105-136行)
|
||||
- 補全遺失字體規格:多語言字體、特殊用途字體
|
||||
- 建立響應式字體大小系統 (mobile/tablet/desktop)
|
||||
- 定義字體層次和使用場景指南
|
||||
|
||||
- [ ] **間距與佈局系統**
|
||||
- 引用文件: `/docs/02_design/ui-ux/ui-ux-guidelines.md` (第139-163行)
|
||||
- 建立8px grid系統標準
|
||||
- 定義響應式佈局斷點和規則
|
||||
- 創建元件間距和頁面佈局標準模板
|
||||
|
||||
- [ ] **組件設計規範**
|
||||
- 引用文件: `/docs/02_design/ui-ux/ui-ux-guidelines.md` (第188-200行)
|
||||
- 補全缺失的組件規範:表單元件、遊戲化元件、學習專用元件
|
||||
- 建立組件狀態系統 (default/hover/active/disabled/loading)
|
||||
- 定義組件變體和使用場景
|
||||
|
||||
**輸出物**:
|
||||
- `ui-ux-guidelines.md` 完整更新版本 (企業級標準)
|
||||
- `design-system-components.md` 完整組件庫文檔
|
||||
- `responsive-design-standards.md` 響應式設計標準
|
||||
- `accessibility-guidelines.md` 無障礙設計指南
|
||||
|
||||
### 1.2 企業級設計系統建立
|
||||
|
||||
**目標**: 創建可重用的設計系統和組件庫
|
||||
|
||||
**工作內容**:
|
||||
- [ ] **原子設計系統**: Atoms → Molecules → Organisms → Templates → Pages
|
||||
- [ ] **Design Tokens**: 設計變數化管理系統
|
||||
- [ ] **組件庫標準化**: 可重用UI組件集合
|
||||
- [ ] **圖標系統**: 學習情境專用圖標設計
|
||||
- [ ] **動畫設計語言**: 統一的動畫效果和互動反饋
|
||||
|
||||
**輸出物**:
|
||||
- `design-system-tokens.css` - 完整設計變數系統
|
||||
- `component-library.html` - 互動式組件展示
|
||||
- `animation-guidelines.md` - 動畫設計標準
|
||||
- `icon-system.svg` - 完整圖標庫
|
||||
|
||||
## 📱 階段二:Mobile端企業級重設計 (第3-6週)
|
||||
|
||||
### 2.1 核心學習功能頁面群組 (第3-4週)
|
||||
|
||||
#### 2.1.1 情境對話系統 🎯 (優先級: P0)
|
||||
|
||||
**規格參考**:
|
||||
- 主規格: `/docs/02_design/function-specs/mobile/01_situational-dialogue-mobile.md`
|
||||
- 共用模組: `/docs/02_design/function-specs/common/ai-algorithm-specs.md`
|
||||
- 共用模組: `/docs/02_design/function-specs/common/speaking-evaluation-specs.md`
|
||||
- 共用模組: `/docs/02_design/function-specs/common/pragmatic-analysis-specs.md`
|
||||
|
||||
**需設計的頁面**:
|
||||
- [ ] **UI_Dialogue_Practice_Main** - 情境對話練習主界面
|
||||
- 設計要求: 語音輸入界面 (參考: ai-algorithm-specs.md 語音處理)
|
||||
- 設計要求: 即時AI反饋顯示 (參考: ai-algorithm-specs.md AI評估系統)
|
||||
- 設計要求: 劇情任務和詞彙任務雙重可視化
|
||||
- 設計要求: 300秒限時挑戰計時器
|
||||
- UI規範: 語音優先設計、即時語法反饋 (ui-ux-guidelines.md 第27-28行)
|
||||
|
||||
- [ ] **UI_Dialogue_Character_Selection** - 角色選擇頁面
|
||||
- 設計要求: 角色卡片設計,突出角色特色和學習情境
|
||||
- 設計要求: 角色能力和適合程度顯示
|
||||
|
||||
- [ ] **UI_Dialogue_Scene_Setting** - 場景設定頁面
|
||||
- 設計要求: 沉浸式場景展示
|
||||
- UI規範: 沉浸式學習環境設計 (ui-ux-guidelines.md 第9行)
|
||||
|
||||
- [ ] **UI_AI_Assistance_Panel** - AI輔助功能面板
|
||||
- 設計要求: 回覆提示道具使用界面
|
||||
- 設計要求: 語法即時檢測顯示
|
||||
- UI規範: 智慧輔助、漸進引導 (ui-ux-guidelines.md 第21-22行)
|
||||
|
||||
- [ ] **UI_Dialogue_Results** - 對話練習結果頁面
|
||||
- 設計要求: 口說評分五維雷達圖 (參考: speaking-evaluation-specs.md)
|
||||
- 設計要求: 語用分析六維評估 (參考: pragmatic-analysis-specs.md)
|
||||
- UI規範: 即時成就反饋 (ui-ux-guidelines.md 第25行)
|
||||
|
||||
#### 2.1.2 詞彙學習系統 📝 (優先級: P0)
|
||||
|
||||
**規格參考**:
|
||||
- 主規格: `/docs/02_design/function-specs/mobile/02_vocabulary-learning-mobile.md`
|
||||
- 共用模組: `/docs/02_design/function-specs/common/progressive-stage-system.md`
|
||||
|
||||
**需設計的頁面**:
|
||||
- [ ] **UI_Vocab_Learning_Enhanced** - 多媒體詞彙學習主界面
|
||||
- 設計要求: 詞彙展示 (音標、定義、例句)
|
||||
- 設計要求: 雙語音頻系統 (標準速度/慢速)
|
||||
- 設計要求: 智慧詞彙標註 (Source + Example)
|
||||
- 設計要求: 視覺輔助學習 (例句配圖)
|
||||
- UI規範: 詞彙學習流程 (ui-ux-guidelines.md 第29行)
|
||||
|
||||
- [ ] **UI_Vocab_Choice_Practice** - 詞彙選擇練習頁面
|
||||
- 設計要求: 選擇題界面,支援多選和單選模式
|
||||
- 設計要求: 即時正確性反饋
|
||||
|
||||
- [ ] **UI_Vocab_Fluency_Results** - 流暢度練習綜合結果
|
||||
- 設計要求: 學習成效可視化展示
|
||||
- 設計要求: 進度追蹤和建議系統
|
||||
|
||||
- [ ] **UI_Vocab_Review_System** - 間隔複習系統界面
|
||||
- 設計要求: 複習提醒和排程界面
|
||||
- UI規範: 間隔複習提醒 (ui-ux-guidelines.md 第31行)
|
||||
|
||||
#### 2.1.3 學習地圖系統 🗺️ (優先級: P0)
|
||||
|
||||
**規格參考**:
|
||||
- 主規格: `/docs/02_design/function-specs/mobile/03_learning-map-mobile.md`
|
||||
- 共用模組: `/docs/02_design/function-specs/common/progressive-stage-system.md`
|
||||
|
||||
**需設計的頁面**:
|
||||
- [ ] **UI_Level_Map** - 學習地圖主畫面 (線性闖關版)
|
||||
- 設計要求: 13階段×20劇本的地圖視覺化
|
||||
- 設計要求: 四關進度指示器 (詞彙學習→詞彙熟悉→口說練習→情境對話)
|
||||
- 設計要求: 關卡狀態管理 (🔒鎖定/⏳可進行/🔄進行中/✅已完成)
|
||||
- 引用規格: progressive-stage-system.md 完整關卡系統
|
||||
|
||||
- [ ] **UI_Current_Level_Info** - 當前可進行關卡資訊面板
|
||||
- 設計要求: 關卡詳細資訊展示
|
||||
- 設計要求: 開始學習入口和準備指南
|
||||
|
||||
- [ ] **UI_Level_Progress_Detail** - 關卡進度詳情頁面
|
||||
- 設計要求: 詳細進度追蹤和統計
|
||||
- 設計要求: 個人表現分析
|
||||
|
||||
- [ ] **UI_Stage_Overview** - 階段總覽和劇本選擇
|
||||
- 設計要求: 階段性學習目標展示
|
||||
- 設計要求: 劇本選擇和預覽功能
|
||||
|
||||
- [ ] **UI_Level_Locked_Modal** - 關卡鎖定提示彈窗
|
||||
- 設計要求: 解鎖條件清晰提示
|
||||
- 設計要求: 引導用戶完成前置任務
|
||||
|
||||
### 2.2 商業功能頁面群組 (第4-5週)
|
||||
|
||||
#### 2.2.1 道具商店系統 🛒 (優先級: P1)
|
||||
|
||||
**規格參考**:
|
||||
- 主規格: `/docs/02_design/function-specs/mobile/04_item-shop-mobile.md`
|
||||
- 共用模組: `/docs/02_design/function-specs/common/business-rules.md`
|
||||
|
||||
**需設計的頁面**:
|
||||
- [ ] **UI_Shop_Categories** - 道具商店分類主頁面
|
||||
- 設計要求: 鑽石購買區 (5個價格套餐)
|
||||
- 設計要求: 學習輔助道具區 (回覆提示、補命、加時)
|
||||
- 設計要求: 限時挑戰道具區 (時間暫停、時間加成)
|
||||
- 引用規格: business-rules.md 命條系統和經濟系統
|
||||
|
||||
- [ ] **UI_Diamond_Purchase** - 鑽石購買頁面
|
||||
- 設計要求: 價格套餐展示和優惠信息
|
||||
- 設計要求: 支付流程整合
|
||||
|
||||
- [ ] **UI_Item_Details** - 單一道具詳情頁面
|
||||
- 設計要求: 道具功能詳細說明
|
||||
- 設計要求: 使用場景和效果展示
|
||||
|
||||
- [ ] **UI_Shop_Item_Confirm** - 道具購買確認彈窗
|
||||
- 設計要求: 購買資訊確認和風險提示
|
||||
- UI規範: 高風險按鈕文字標注 (ui-ux-guidelines.md 第194行)
|
||||
|
||||
- [ ] **UI_Cost_Confirm_Popup** - 成本確認彈窗 (口說練習特別關卡)
|
||||
- 設計要求: 特殊關卡成本說明
|
||||
- 設計要求: 用戶決策支援資訊
|
||||
|
||||
#### 2.2.2 用戶認證系統 🔐 (優先級: P1)
|
||||
|
||||
**規格參考**:
|
||||
- 主規格: `/docs/02_design/function-specs/mobile/05_user-authentication-mobile.md`
|
||||
- 共用模組: `/docs/02_design/function-specs/common/business-rules.md`
|
||||
|
||||
**需設計的頁面**:
|
||||
- [ ] **UI_Login_Main** - 主要登入頁面
|
||||
- 設計要求: 多平台登入選項 (Google, Facebook, Apple)
|
||||
- 設計要求: 記住登入和安全驗證
|
||||
- 設計要求: 錯誤處理和安全提示
|
||||
|
||||
- [ ] **UI_SignUp_Main** - 用戶註冊頁面
|
||||
- 設計要求: 分步驟註冊流程
|
||||
- 設計要求: 即時驗證和錯誤提示
|
||||
- 設計要求: 學習目標和程度設定
|
||||
|
||||
- [ ] **UI_PasswordReset_Form** - 密碼重置表單
|
||||
- 設計要求: 多步驟驗證流程
|
||||
- 設計要求: 安全性說明和引導
|
||||
|
||||
- [ ] **UI_PasswordReset_Popup** - 密碼重置確認彈窗
|
||||
- 設計要求: 重置成功確認和後續指引
|
||||
|
||||
- [ ] **UI_Account_List** - 帳戶列表管理頁面
|
||||
- 設計要求: 多帳戶管理和切換
|
||||
- 設計要求: 帳戶安全狀態顯示
|
||||
|
||||
- [ ] **UI_Account_Option** - 帳戶選項設定頁面
|
||||
- 設計要求: 帳戶設定和隱私控制
|
||||
- 設計要求: 帳戶關聯和解綁功能
|
||||
|
||||
### 2.3 支援功能頁面群組 (第5-6週)
|
||||
|
||||
#### 2.3.1 系統介面和狀態頁面 📊 (優先級: P2)
|
||||
|
||||
**需設計的頁面**:
|
||||
- [ ] **UI_Insufficient_Resources** - 資源不足提示頁面
|
||||
- 設計要求: 清晰的資源不足說明
|
||||
- 設計要求: 獲取資源的引導路徑
|
||||
|
||||
- [ ] **UI_LifePoints_Display** - 生命點數顯示組件
|
||||
- 設計要求: 直觀的生命值視覺化
|
||||
- UI規範: 命條生命系統 (ui-ux-guidelines.md 第30行)
|
||||
|
||||
- [ ] **UI_Subscription_Success** - 訂閱成功頁面
|
||||
- 設計要求: 訂閱確認和權益說明
|
||||
- 設計要求: 後續使用指引
|
||||
|
||||
- [ ] **UI_TimeWarp_Cards** - 時光卷使用介面
|
||||
- 設計要求: 時光卷功能說明和使用確認
|
||||
- 設計要求: 使用後效果展示
|
||||
|
||||
- [ ] **UI_LevelResult_SuccessResult** - 關卡成功結果頁面
|
||||
- 設計要求: 成就慶祝動畫和統計展示
|
||||
- UI規範: 即時成就反饋 (ui-ux-guidelines.md 第25行)
|
||||
|
||||
## 💻 階段三:Web端企業級重設計 (第7-9週)
|
||||
|
||||
### 3.1 Web端專屬功能設計 (第7-8週)
|
||||
|
||||
**規格參考**: `/docs/02_design/function-specs/web/` 全部Web端規格
|
||||
|
||||
**設計重點**:
|
||||
- [ ] **桌面優化界面**: 大螢幕佈局和多視窗支援
|
||||
- [ ] **鍵盤導航**: 完整的鍵盤操作支援
|
||||
- [ ] **批量操作**: 企業級批量管理功能
|
||||
- [ ] **高級分析**: 詳細的學習分析和報告功能
|
||||
|
||||
**需設計的主要頁面**:
|
||||
- [ ] **詞彙學習Web版**: 桌面優化的詞彙學習界面
|
||||
- [ ] **情境對話Web版**: 大螢幕對話練習界面
|
||||
- [ ] **學習地圖Web版**: 多層級地圖導航界面
|
||||
- [ ] **道具商店Web版**: 企業級商店管理界面
|
||||
- [ ] **用戶認證Web版**: SSO和企業登入支援
|
||||
|
||||
### 3.2 響應式設計和跨平台整合 (第8-9週)
|
||||
|
||||
**工作內容**:
|
||||
- [ ] **響應式佈局**: Mobile → Tablet → Desktop 完整適配
|
||||
- [ ] **跨瀏覽器相容性**: Chrome, Firefox, Safari, Edge 完整支援
|
||||
- [ ] **效能優化**: 載入時間和互動回應最佳化
|
||||
- [ ] **PWA功能**: 漸進式Web應用功能整合
|
||||
|
||||
## 🔧 階段四:設計系統完善和品質保證 (第10-12週)
|
||||
|
||||
### 4.1 設計系統文檔化和工具化 (第10-11週)
|
||||
|
||||
**工作內容**:
|
||||
- [ ] **設計規範手冊**: 完整的設計規範使用指南
|
||||
- [ ] **組件使用指南**: 每個組件的使用場景和最佳實踐
|
||||
- [ ] **設計審查清單**: 品質控制清單和審查標準
|
||||
- [ ] **維護指南**: 設計系統維護和更新流程
|
||||
|
||||
### 4.2 品質保證和使用性測試 (第11-12週)
|
||||
|
||||
**工作內容**:
|
||||
- [ ] **設計一致性審查**: 跨平台設計一致性檢查
|
||||
- [ ] **無障礙性測試**: WCAG 2.1 AA級合規驗證
|
||||
- [ ] **使用性測試**: 用戶測試和回饋收集
|
||||
- [ ] **效能評估**: 設計對系統效能的影響評估
|
||||
|
||||
## 📊 成功標準和驗收條件
|
||||
|
||||
### 🎯 品質標準
|
||||
1. **功能規格符合度**: 100%符合所有功能規格要求
|
||||
2. **設計一致性**: 跨平台設計語言100%統一
|
||||
3. **無障礙標準**: WCAG 2.1 AA級100%合規
|
||||
4. **效能標準**: 頁面載入時間<3秒,互動回應時間<200ms
|
||||
5. **瀏覽器支援**: 主流瀏覽器100%相容
|
||||
|
||||
### 📋 驗收清單
|
||||
- [ ] 所有UI畫面符合對應功能規格文件要求
|
||||
- [ ] 所有設計符合UI/UX規範標準
|
||||
- [ ] 跨平台視覺一致性通過審查
|
||||
- [ ] 無障礙功能測試全部通過
|
||||
- [ ] 使用性測試滿意度≥90%
|
||||
|
||||
## 📁 交付物清單
|
||||
|
||||
### 🎨 設計文檔
|
||||
- [ ] `ui-ux-guidelines.md` - 完善的UI/UX設計規範
|
||||
- [ ] `design-system-documentation.md` - 設計系統完整文檔
|
||||
- [ ] `component-library-guide.md` - 組件庫使用指南
|
||||
- [ ] `responsive-design-standards.md` - 響應式設計標準
|
||||
|
||||
### 💻 設計資產
|
||||
- [ ] `design-system.css` - 完整CSS設計系統
|
||||
- [ ] 95+ HTML原型頁面 (Mobile + Web)
|
||||
- [ ] 完整SVG圖標庫
|
||||
- [ ] 設計系統展示網站
|
||||
|
||||
### 📋 支援文檔
|
||||
- [ ] `design-review-checklist.md` - 設計審查清單
|
||||
- [ ] `accessibility-compliance-report.md` - 無障礙合規報告
|
||||
- [ ] `usability-test-results.md` - 使用性測試報告
|
||||
- [ ] `maintenance-guidelines.md` - 維護指南
|
||||
|
||||
## 🚨 風險管控和品質保證
|
||||
|
||||
### ⚠️ 關鍵風險點
|
||||
1. **規格理解偏差**: 設計不符合功能規格要求
|
||||
- **控制措施**: 每個設計階段都進行規格文件交叉檢查
|
||||
- **責任人**: 設計師必須深度閱讀相關規格文件
|
||||
|
||||
2. **設計一致性風險**: 跨頁面設計語言不統一
|
||||
- **控制措施**: 建立設計審查機制,每週進行一致性檢查
|
||||
- **工具支援**: 建立設計系統檢查清單
|
||||
|
||||
3. **無障礙合規風險**: 無障礙功能不完整
|
||||
- **控制措施**: 每個組件設計完成都進行無障礙測試
|
||||
- **標準遵循**: 嚴格遵循WCAG 2.1 AA級標準
|
||||
|
||||
### 🔍 品質控制機制
|
||||
1. **階段性審查**: 每個階段結束進行全面審查
|
||||
2. **同行評議**: 設計師之間相互審查和回饋
|
||||
3. **用戶測試**: 關鍵頁面進行真實用戶測試
|
||||
4. **技術可行性評估**: 設計與開發團隊聯合評估
|
||||
|
||||
## 📞 執行支援和溝通機制
|
||||
|
||||
### 🤝 團隊協作
|
||||
- **設計團隊**: 負責設計執行和品質控制
|
||||
- **產品團隊**: 提供功能需求解釋和使用者回饋
|
||||
- **開發團隊**: 提供技術可行性建議和實現支援
|
||||
- **測試團隊**: 提供品質測試和驗收支援
|
||||
|
||||
### 📋 進度追蹤
|
||||
- **每週進度會議**: 檢討進度和解決阻礙
|
||||
- **里程碑審查**: 階段性成果展示和評估
|
||||
- **問題升級機制**: 重大問題快速上報和解決
|
||||
- **文檔同步更新**: 確保所有團隊資訊同步
|
||||
|
||||
---
|
||||
|
||||
**📝 重要聲明**:
|
||||
本計劃基於Drama Ling v3.0共用模組架構制定,確保所有設計完全符合功能規格要求,達到企業級應用標準。所有設計師在執行前必須深入理解相關功能規格文件和UI/UX規範,確保設計品質和一致性。
|
||||
|
||||
**🎯 最終目標**:
|
||||
創建Drama Ling史上最高品質的UI設計系統,為用戶提供世界級的沉浸式英語學習體驗。
|
||||
|
||||
---
|
||||
|
||||
**最後更新**: 2025-01-15
|
||||
**計劃版本**: v4.0 - 企業級重構
|
||||
**執行週期**: 12週
|
||||
**預期成果**: 95+ 企業級UI畫面
|
||||
|
|
@ -0,0 +1,199 @@
|
|||
# 🎯 Drama Ling 任務清單
|
||||
|
||||
## 📋 當前任務
|
||||
|
||||
### 🔥 緊急任務
|
||||
|
||||
|
||||
|
||||
### ⚠️ 重要任務
|
||||
- [x] 🎮 **練習系統核心開發** - 選擇題、圖片匹配、句子重組三種模式 (56小時) ✅
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: Page_Vocab_Choice_Practice_W等頁面,反應時間測量
|
||||
- 📋 合規基礎: function-specs定義的練習模式
|
||||
- ✅ **完成項目**:
|
||||
- 選擇題練習頁面 (VocabularyChoicePracticeView.vue)
|
||||
- 圖片匹配練習頁面 (VocabularyMatchingPracticeView.vue) - HTML5 拖放API
|
||||
- 句子重組練習頁面 (VocabularyReorganizePracticeView.vue) - 拖放重組
|
||||
- 毫秒級反應時間測量系統
|
||||
- 命條系統整合
|
||||
- 鍵盤快捷鍵支援 (Enter, Space, Escape)
|
||||
- 響應式設計和觸摸支援
|
||||
- TypeScript類型安全和Pinia狀態管理
|
||||
|
||||
- [x] 📊 **Web專用分析儀表板** - Page_Vocab_Analytics_Dashboard_W數據視覺化 (40小時) ✅
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: 統計卡片、圖表庫整合、報告匯出
|
||||
- 📋 合規基礎: Web端特色功能規格
|
||||
- ✅ **完成項目**:
|
||||
- 完整的分析儀表板頁面 (VocabularyAnalyticsDashboard.vue)
|
||||
- 統計卡片組件 (StatCard.vue) - 趨勢顯示和互動效果
|
||||
- 錯誤分析熱力圖組件 (ErrorHeatmap.vue) - 可視化錯誤模式
|
||||
- Chart.js 圖表整合 - 圓餅圖、折線圖、雷達圖
|
||||
- 多格式報告匯出功能 (PDF, Excel, CSV)
|
||||
- 時間範圍篩選和自訂日期選擇器
|
||||
- 響應式設計和列印友善格式
|
||||
- 快捷鍵支援 (T, F, Ctrl+E, Ctrl+P, F11)
|
||||
- 學習建議和薄弱點識別系統
|
||||
|
||||
- [x] 🔄 **複習系統智能化** - 間隔複習演算法,Page_Vocab_Review_Main_W (32小時) ✅
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: 學習計劃生成、薄弱點識別
|
||||
- ✅ **完成項目**:
|
||||
- 智能間隔複習演算法 (spacedRepetition.ts) - 基於Ebbinghaus遺忘曲線和SM-2演算法
|
||||
- 複習系統Pinia Store (review.ts) - 狀態管理和數據分析
|
||||
- 智能複習主頁面 (VocabularyReviewMain.vue) - Page_Vocab_Review_Main_W實現
|
||||
- 個人化學習計劃生成 - 7天智能排程系統
|
||||
- 薄弱點自動識別 - 基於錯誤模式分析
|
||||
- 自適應難度調整 - 根據表現動態調整間隔
|
||||
- 學習效率分析 - 趨勢追蹤和改善建議
|
||||
- 學習連勝和動機系統 - 遊戲化元素
|
||||
- 複習提醒和設定系統 - 個人化配置
|
||||
|
||||
- [ ] 🔧 **Web端特色功能整合** - 多標籤學習、書籤整合、PWA支援 (32小時)
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: function-specs定義的Web端獨有功能
|
||||
|
||||
### 📝 一般任務
|
||||
- [ ] 🧪 **測試框架建立和測試撰寫** - Vitest + Vue Test Utils,覆蓋率>80% (24小時)
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: 單元測試、集成測試、HTML原型視覺回歸測試
|
||||
- 📋 合規基礎: vue-development-standards.md測試規範
|
||||
|
||||
- [ ] 🔗 **後端API設計和開發** - 詞彙服務、練習記錄、進度追蹤API (48小時)
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: RESTful API、資料模型實現、音頻服務整合
|
||||
|
||||
- [ ] 📦 **PWA功能實現和部署優化** - Service Worker、離線支援、Vite打包優化 (24小時)
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: Quasar PWA plugin、離線模式、效能優化
|
||||
|
||||
- [ ] 📋 **規格合規驗收和品質保證** - 所有specification文檔對照檢查 (16小時)
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- 🎯 關鍵: HTML原型像素級檢查、function-specs功能完整性
|
||||
- 📋 驗收標準: 視覺還原度100%、功能實現率100%
|
||||
|
||||
### 💡 未來想法
|
||||
- [ ] 📱 **移動端適配** - 響應式設計優化和觸控操作支援
|
||||
- [ ] 🤖 **AI學習建議** - 個人化學習路徑推薦和薄弱點分析
|
||||
- [ ] 🌐 **多語言支援** - 界面國際化和多語言詞彙庫
|
||||
- [ ] 📈 **進階分析** - 學習模式識別和效率優化建議
|
||||
|
||||
---
|
||||
|
||||
## 📊 快速統計
|
||||
|
||||
**當前狀態**:
|
||||
- 🔥 緊急: 2個任務 (基礎架構 + 詞彙介紹頁面)
|
||||
- ⚠️ 重要: 4個任務 (練習系統 + 分析儀表板 + 複習系統)
|
||||
- 📝 一般: 4個任務 (測試 + 後端API + PWA + 品質保證)
|
||||
- 💡 想法: 4個任務 (未來擴展功能)
|
||||
|
||||
**預估工作量**: 320小時 (約6-8週,3-4人團隊)
|
||||
**規格基礎**: 嚴格基於HTML原型 + function-specs + vue-architecture
|
||||
|
||||
---
|
||||
|
||||
## 📚 已完成任務
|
||||
|
||||
### 2025-09-10 完成
|
||||
- [x] 📋 **詞彙學習開發計劃重新生成** - 嚴格基於specification文檔,避免AI偏離 ✅ (2025-09-10)
|
||||
- ✨ 完成功能: 基於4個docs文檔重新生成開發計劃
|
||||
- 📋 合規基礎: vocabulary.html + vocabulary-learning-web.md + vue-frontend-architecture.md + vue-development-standards.md
|
||||
- 🎯 關鍵改進: 像素級HTML原型對照、規格合規檢查機制、技術選型100%遵循架構文檔
|
||||
- 📄 成果: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- [x] 🔧 **修正dl工具路徑設定** - 工具腳本路徑過時,./dl issue指令失敗 🔄
|
||||
- 📄 問題: TOOLS_DIR設為 "$SCRIPT_DIR/tools" 但實際在 "sop/tools/"
|
||||
- 🎯 目標: 修正路徑設定,確保所有dl指令正常運作
|
||||
- ⚠️ 發現: issue.sh腳本仍使用舊的ISSUES.md系統,需要更新到TASKS.md
|
||||
|
||||
- [x] 🔧 **系統性SOP一致性檢查和修正** - 全面檢查所有工具與SOP的一致性,建立防護機制 ✅ (2025-09-10)
|
||||
- ✨ 完成功能:
|
||||
- 修正dl工具TOOLS_DIR路徑問題
|
||||
- 修正create_report.sh的sed語法錯誤
|
||||
- 建立正確報告工具目錄結構
|
||||
- 建立SOP一致性檢查腳本 (sop/scripts/sop_consistency_check.sh)
|
||||
- 修正報告模板中的ISSUES.md引用
|
||||
- 📊 發現問題: 15個工具腳本仍使用過時的ISSUES.md/PROJECTS.md系統
|
||||
- 🎯 建立防護: 自動化檢查機制可偵測工具與SOP不一致
|
||||
|
||||
|
||||
- [x] ✅ **清空過時任務列表** - 重置任務管理系統,準備新的任務規劃 ✅ (2025-09-10)
|
||||
|
||||
- [x] 🔧 **SOP改善 - AI開發計劃生成規範標準化** - 建立強制性docs約束機制,避免AI偏離既有規格 ✅ (2025-09-10)
|
||||
- ✨ 完成功能: 更新CLAUDE.md v4.1,新增開發計劃生成標準流程、三階段驗證機制、檢查清單
|
||||
- 📄 分析報告: [AI開發計劃SOP改善分析](sop/reports/analysis/2025-09-10_ai-development-plan-sop-improvement.md)
|
||||
- 🎯 解決問題: vocabulary-learning-web-development-plan.md 偏離docs規範,建立系統性防護機制
|
||||
|
||||
- [x] 🔧 **系統性SOP一致性檢查和修正** - 全面檢查所有工具與SOP的一致性,建立防護機制 ✅ (2025-09-10)
|
||||
- ✨ 完成功能:
|
||||
- 修正dl工具TOOLS_DIR路徑問題
|
||||
- 修正create_report.sh的sed語法錯誤
|
||||
- 建立正確報告工具目錄結構
|
||||
- 建立SOP一致性檢查腳本 (sop/scripts/sop_consistency_check.sh)
|
||||
- 修正報告模板中的ISSUES.md引用
|
||||
- **新增**: 檢查腳本自動生成詳細log到 sop/reports/logs/ (區分檢查log與分析報告)
|
||||
- 📊 發現問題: 15個工具腳本仍使用過時的ISSUES.md/PROJECTS.md系統
|
||||
- 🎯 建立防護: 自動化檢查機制可偵測工具與SOP不一致,並生成正式檢查log
|
||||
- 📄 詳細分析: [SOP工具系統全面重構分析](sop/reports/analysis/2025-09-10_sop-tools-system-overhaul.md)
|
||||
- 📄 檢查log範例: [SOP一致性檢查log](sop/reports/logs/2025-09-10_sop-consistency-check-120656.md)
|
||||
|
||||
- [x] 🏗️ **FE Vue專案基礎架構建立** - Vue 3 + Quasar詞彙學習Web版專案初始化 ✅ (2025-09-10)
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
- [x] 🎨 **FE Vue詞彙介紹頁面開發** - 基於Quasar的核心學習頁面,Web Audio API和快捷鍵支援 ✅ (2025-09-09)
|
||||
- ✨ 完成功能: 完整詞彙介紹界面、Web Audio API整合、快捷鍵系統、Composable架構
|
||||
- 📄 參考: [詞彙學習Web開發規劃](projects/vocabulary-learning-web-development-plan.md)
|
||||
|
||||
- [x] 🔑 **修復登入系統** - 解決登入流程問題,確保用戶能順利進入詞彙學習頁面 ✅ (2025-09-09)
|
||||
- ✨ 完成功能: 開發模式測試登入、路由守護、認證狀態管理、UI提示系統
|
||||
- 🧪 測試帳戶: test@dramaling.com / test123
|
||||
- 🎯 快速入口: 首頁「測試登入」按鈕或登入頁「快速填入」功能
|
||||
|
||||
|
||||
---
|
||||
|
||||
## 🛠️ 系統使用指南
|
||||
|
||||
### 查看任務
|
||||
```bash
|
||||
./dl task # 打開此任務管理文件
|
||||
./dl status # 查看任務統計
|
||||
./dl list # 快速查看待辦清單
|
||||
```
|
||||
|
||||
### 工作模式
|
||||
1. **討論階段**: 與Claude自由討論需求和想法
|
||||
2. **記錄階段**: Claude自動記錄任務到此系統,並創建對應專案詳細文檔
|
||||
3. **執行階段**: 查看此文件選擇任務批量執行
|
||||
4. **完成階段**: 標記任務完成 [x],任務自動移至已完成區域
|
||||
|
||||
### 任務格式說明
|
||||
```markdown
|
||||
- [ ] 🎯 **任務名稱** - 簡短描述 (預估時間)
|
||||
- 📄 參考: [專案詳細文檔](projects/project-name.md)
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
**建立日期**: 2025-09-09
|
||||
**最後更新**: 2025-09-10 (重新生成規格合規的詞彙學習開發任務)
|
||||
**維護者**: Claude Code & Drama Ling Team
|
||||
|
||||
---
|
||||
|
||||
## 🎯 專案任務說明
|
||||
|
||||
### 詞彙學習功能 (Web版) 開發專案
|
||||
|
||||
本專案基於完整的開發規劃,按照8週開發週期分階段執行:
|
||||
|
||||
**第一階段 (緊急)**: 專案基礎架構 + 核心學習頁面
|
||||
**第二階段 (重要)**: 練習系統 + 數據分析功能
|
||||
**第三階段 (一般)**: 整合優化 + 後端API開發
|
||||
**第四階段 (想法)**: 未來擴展功能規劃
|
||||
|
||||
**技術棧**: Vue 3 + Quasar Framework + Pinia + Web Audio API + PWA
|
||||
**團隊配置**: 前端2人 + 後端1-2人 + 可選DevOps
|
||||
**關鍵特色**: 快捷鍵操作、多標籤學習、Markdown筆記、Vue-ECharts分析
|
||||
|
||||
詳細技術規格和開發時程請參考專案規劃文檔。
|
||||
|
|
@ -0,0 +1,27 @@
|
|||
# Drama Ling Applications
|
||||
|
||||
本目錄包含 Drama Ling 專案的所有應用程式。
|
||||
|
||||
## 應用程式架構
|
||||
|
||||
```
|
||||
apps/
|
||||
├── web/ # Vue.js Web 前端應用
|
||||
├── mobile/ # Flutter 移動端應用
|
||||
└── backend/ # .NET Core 後端 API
|
||||
```
|
||||
|
||||
## 開發狀態
|
||||
|
||||
| 應用程式 | 狀態 | 技術棧 | 說明 |
|
||||
|---------|------|--------|------|
|
||||
| Web | ✅ 開發中 | Vue.js + Quasar | Web 前端界面 |
|
||||
| Mobile | ✅ 開發中 | Flutter + Riverpod | 跨平台移動應用 |
|
||||
| Backend | ✅ 開發中 | .NET Core + EF Core | REST API 服務 |
|
||||
|
||||
## 開發指南
|
||||
|
||||
各應用程式的詳細開發文檔請參考:
|
||||
- 技術文檔:`../docs/04_technical/`
|
||||
- 專案規格:`../projects/`
|
||||
- 任務管理:`../TASKS.md`
|
||||
|
|
@ -0,0 +1,58 @@
|
|||
using Microsoft.AspNetCore.Mvc;
|
||||
|
||||
namespace DramaLing.API.Controllers;
|
||||
|
||||
[ApiController]
|
||||
[Route("api/[controller]")]
|
||||
public class HealthController : ControllerBase
|
||||
{
|
||||
/// <summary>
|
||||
/// 健康檢查端點
|
||||
/// </summary>
|
||||
/// <returns>服務健康狀態</returns>
|
||||
[HttpGet]
|
||||
public IActionResult GetHealth()
|
||||
{
|
||||
var response = new
|
||||
{
|
||||
Status = "Healthy",
|
||||
Timestamp = DateTime.UtcNow,
|
||||
Version = "1.0.0",
|
||||
Service = "Drama Ling API"
|
||||
};
|
||||
|
||||
return Ok(response);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// 詳細健康檢查
|
||||
/// </summary>
|
||||
/// <returns>詳細的系統狀態</returns>
|
||||
[HttpGet("detailed")]
|
||||
public IActionResult GetDetailedHealth()
|
||||
{
|
||||
var response = new
|
||||
{
|
||||
Status = "Healthy",
|
||||
Timestamp = DateTime.UtcNow,
|
||||
Version = "1.0.0",
|
||||
Service = "Drama Ling API",
|
||||
Environment = Environment.GetEnvironmentVariable("ASPNETCORE_ENVIRONMENT") ?? "Production",
|
||||
Uptime = Environment.TickCount64,
|
||||
Database = "Connected", // TODO: 實際檢查資料庫連線
|
||||
Cache = "Connected", // TODO: 實際檢查Redis連線
|
||||
Memory = new
|
||||
{
|
||||
WorkingSet = GC.GetTotalMemory(false),
|
||||
GcCollections = new
|
||||
{
|
||||
Gen0 = GC.CollectionCount(0),
|
||||
Gen1 = GC.CollectionCount(1),
|
||||
Gen2 = GC.CollectionCount(2)
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
return Ok(response);
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,35 @@
|
|||
# Development Dockerfile for .NET API
|
||||
FROM mcr.microsoft.com/dotnet/aspnet:8.0 AS base
|
||||
WORKDIR /app
|
||||
EXPOSE 5000
|
||||
|
||||
FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
|
||||
WORKDIR /src
|
||||
|
||||
# Copy project files
|
||||
COPY ["DramaLing.API/DramaLing.API.csproj", "DramaLing.API/"]
|
||||
COPY ["DramaLing.Application/DramaLing.Application.csproj", "DramaLing.Application/"]
|
||||
COPY ["DramaLing.Core/DramaLing.Core.csproj", "DramaLing.Core/"]
|
||||
COPY ["DramaLing.Infrastructure/DramaLing.Infrastructure.csproj", "DramaLing.Infrastructure/"]
|
||||
|
||||
# Restore dependencies
|
||||
RUN dotnet restore "DramaLing.API/DramaLing.API.csproj"
|
||||
|
||||
# Copy source code
|
||||
COPY . .
|
||||
|
||||
# Build the application
|
||||
WORKDIR "/src/DramaLing.API"
|
||||
RUN dotnet build "DramaLing.API.csproj" -c Release -o /app/build
|
||||
|
||||
FROM build AS publish
|
||||
RUN dotnet publish "DramaLing.API.csproj" -c Release -o /app/publish /p:UseAppHost=false
|
||||
|
||||
FROM base AS final
|
||||
WORKDIR /app
|
||||
COPY --from=publish /app/publish .
|
||||
|
||||
# Create logs directory
|
||||
RUN mkdir -p /app/logs
|
||||
|
||||
ENTRYPOINT ["dotnet", "DramaLing.API.dll"]
|
||||
|
|
@ -0,0 +1,28 @@
|
|||
<Project Sdk="Microsoft.NET.Sdk.Web">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
||||
<DocumentationFile>bin\Debug\net8.0\DramaLing.API.xml</DocumentationFile>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="8.0.0" />
|
||||
<PackageReference Include="Swashbuckle.AspNetCore" Version="6.5.0" />
|
||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.AspNetCore.Identity.EntityFrameworkCore" Version="8.0.0" />
|
||||
<PackageReference Include="FluentValidation.AspNetCore" Version="11.3.0" />
|
||||
<PackageReference Include="Serilog.AspNetCore" Version="8.0.0" />
|
||||
<PackageReference Include="Serilog.Sinks.Console" Version="5.0.0" />
|
||||
<PackageReference Include="Serilog.Sinks.File" Version="5.0.0" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\DramaLing.Application\DramaLing.Application.csproj" />
|
||||
<ProjectReference Include="..\DramaLing.Infrastructure\DramaLing.Infrastructure.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
|
@ -0,0 +1,116 @@
|
|||
using DramaLing.Application;
|
||||
using DramaLing.Infrastructure;
|
||||
using Microsoft.OpenApi.Models;
|
||||
using Serilog;
|
||||
|
||||
var builder = WebApplication.CreateBuilder(args);
|
||||
|
||||
// Configure Serilog
|
||||
Log.Logger = new LoggerConfiguration()
|
||||
.ReadFrom.Configuration(builder.Configuration)
|
||||
.CreateLogger();
|
||||
|
||||
builder.Host.UseSerilog();
|
||||
|
||||
// Add services to the container.
|
||||
builder.Services.AddApplication();
|
||||
builder.Services.AddInfrastructure(builder.Configuration);
|
||||
|
||||
builder.Services.AddControllers();
|
||||
builder.Services.AddEndpointsApiExplorer();
|
||||
|
||||
// Configure Swagger/OpenAPI
|
||||
builder.Services.AddSwaggerGen(c =>
|
||||
{
|
||||
c.SwaggerDoc("v1", new OpenApiInfo
|
||||
{
|
||||
Title = "Drama Ling API",
|
||||
Version = "v1",
|
||||
Description = "API for Drama Ling language learning application",
|
||||
Contact = new OpenApiContact
|
||||
{
|
||||
Name = "Drama Ling Team",
|
||||
Email = "dev@dramaling.com"
|
||||
}
|
||||
});
|
||||
|
||||
// Configure JWT authentication in Swagger
|
||||
c.AddSecurityDefinition("Bearer", new OpenApiSecurityScheme
|
||||
{
|
||||
Description = "JWT Authorization header using the Bearer scheme. Example: \"Bearer {token}\"",
|
||||
Name = "Authorization",
|
||||
In = ParameterLocation.Header,
|
||||
Type = SecuritySchemeType.ApiKey,
|
||||
Scheme = "Bearer"
|
||||
});
|
||||
|
||||
c.AddSecurityRequirement(new OpenApiSecurityRequirement
|
||||
{
|
||||
{
|
||||
new OpenApiSecurityScheme
|
||||
{
|
||||
Reference = new OpenApiReference
|
||||
{
|
||||
Type = ReferenceType.SecurityScheme,
|
||||
Id = "Bearer"
|
||||
}
|
||||
},
|
||||
Array.Empty<string>()
|
||||
}
|
||||
});
|
||||
|
||||
// Include XML comments
|
||||
var xmlFile = Path.Combine(AppContext.BaseDirectory, "DramaLing.API.xml");
|
||||
if (File.Exists(xmlFile))
|
||||
{
|
||||
c.IncludeXmlComments(xmlFile);
|
||||
}
|
||||
});
|
||||
|
||||
// Configure CORS
|
||||
builder.Services.AddCors(options =>
|
||||
{
|
||||
options.AddPolicy("DramaLingPolicy", policy =>
|
||||
{
|
||||
policy.AllowAnyOrigin()
|
||||
.AllowAnyMethod()
|
||||
.AllowAnyHeader();
|
||||
});
|
||||
});
|
||||
|
||||
var app = builder.Build();
|
||||
|
||||
// Configure the HTTP request pipeline.
|
||||
if (app.Environment.IsDevelopment())
|
||||
{
|
||||
app.UseSwagger();
|
||||
app.UseSwaggerUI(c =>
|
||||
{
|
||||
c.SwaggerEndpoint("/swagger/v1/swagger.json", "Drama Ling API v1");
|
||||
c.RoutePrefix = string.Empty; // Serve Swagger UI at the app's root
|
||||
});
|
||||
}
|
||||
|
||||
app.UseHttpsRedirection();
|
||||
app.UseCors("DramaLingPolicy");
|
||||
|
||||
app.UseAuthentication();
|
||||
app.UseAuthorization();
|
||||
|
||||
app.MapControllers();
|
||||
|
||||
app.MapGet("/health", () => new { Status = "Healthy", Timestamp = DateTime.UtcNow });
|
||||
|
||||
try
|
||||
{
|
||||
Log.Information("Starting Drama Ling API");
|
||||
app.Run();
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
Log.Fatal(ex, "Application terminated unexpectedly");
|
||||
}
|
||||
finally
|
||||
{
|
||||
Log.CloseAndFlush();
|
||||
}
|
||||
|
|
@ -0,0 +1,18 @@
|
|||
{
|
||||
"Logging": {
|
||||
"LogLevel": {
|
||||
"Default": "Information",
|
||||
"Microsoft.AspNetCore": "Warning",
|
||||
"Microsoft.EntityFrameworkCore.Database.Command": "Information"
|
||||
}
|
||||
},
|
||||
"ConnectionStrings": {
|
||||
"DefaultConnection": "Host=localhost;Port=5432;Database=dramaling_dev;Username=postgres;Password=password"
|
||||
},
|
||||
"JwtSettings": {
|
||||
"Key": "development-key-256-bits-long-for-jwt-signing-purpose-only",
|
||||
"Issuer": "DramaLingAPI-Dev",
|
||||
"Audience": "DramaLingUsers-Dev",
|
||||
"DurationInMinutes": 1440
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,53 @@
|
|||
{
|
||||
"Logging": {
|
||||
"LogLevel": {
|
||||
"Default": "Information",
|
||||
"Microsoft.AspNetCore": "Warning"
|
||||
}
|
||||
},
|
||||
"Serilog": {
|
||||
"Using": ["Serilog.Sinks.Console", "Serilog.Sinks.File"],
|
||||
"MinimumLevel": {
|
||||
"Default": "Information",
|
||||
"Override": {
|
||||
"Microsoft": "Warning",
|
||||
"System": "Warning"
|
||||
}
|
||||
},
|
||||
"WriteTo": [
|
||||
{
|
||||
"Name": "Console",
|
||||
"Args": {
|
||||
"outputTemplate": "[{Timestamp:HH:mm:ss} {Level:u3}] {Message:lj} {Properties:j}{NewLine}{Exception}"
|
||||
}
|
||||
},
|
||||
{
|
||||
"Name": "File",
|
||||
"Args": {
|
||||
"path": "logs/dramaling-.txt",
|
||||
"rollingInterval": "Day",
|
||||
"outputTemplate": "[{Timestamp:yyyy-MM-dd HH:mm:ss.fff zzz} {Level:u3}] {Message:lj} {Properties:j}{NewLine}{Exception}"
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
"AllowedHosts": "*",
|
||||
"ConnectionStrings": {
|
||||
"DefaultConnection": "Host=localhost;Port=5432;Database=dramaling_dev;Username=postgres;Password=password"
|
||||
},
|
||||
"JwtSettings": {
|
||||
"Key": "your-256-bit-secret-key-here-must-be-at-least-32-characters",
|
||||
"Issuer": "DramaLingAPI",
|
||||
"Audience": "DramaLingUsers",
|
||||
"DurationInMinutes": 60
|
||||
},
|
||||
"OpenAI": {
|
||||
"ApiKey": "your-openai-api-key-here",
|
||||
"Model": "gpt-4o-mini",
|
||||
"MaxTokens": 1000,
|
||||
"Temperature": 0.7
|
||||
},
|
||||
"Redis": {
|
||||
"ConnectionString": "localhost:6379"
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,17 @@
|
|||
using Microsoft.Extensions.DependencyInjection;
|
||||
using System.Reflection;
|
||||
|
||||
namespace DramaLing.Application;
|
||||
|
||||
public static class DependencyInjection
|
||||
{
|
||||
public static IServiceCollection AddApplication(this IServiceCollection services)
|
||||
{
|
||||
services.AddMediatR(cfg =>
|
||||
cfg.RegisterServicesFromAssembly(Assembly.GetExecutingAssembly()));
|
||||
|
||||
services.AddAutoMapper(Assembly.GetExecutingAssembly());
|
||||
|
||||
return services;
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,20 @@
|
|||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="MediatR" Version="12.2.0" />
|
||||
<PackageReference Include="FluentValidation" Version="11.8.0" />
|
||||
<PackageReference Include="AutoMapper" Version="12.0.1" />
|
||||
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="8.0.0" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\DramaLing.Core\DramaLing.Core.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
|
@ -0,0 +1,9 @@
|
|||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
</PropertyGroup>
|
||||
|
||||
</Project>
|
||||
|
|
@ -0,0 +1,9 @@
|
|||
namespace DramaLing.Core.Entities;
|
||||
|
||||
public abstract class BaseEntity
|
||||
{
|
||||
public Guid Id { get; set; } = Guid.NewGuid();
|
||||
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
||||
public DateTime UpdatedAt { get; set; } = DateTime.UtcNow;
|
||||
public bool IsDeleted { get; set; } = false;
|
||||
}
|
||||
|
|
@ -0,0 +1,94 @@
|
|||
using DramaLing.Core.Enums;
|
||||
|
||||
namespace DramaLing.Core.Entities;
|
||||
|
||||
public class Achievement : BaseEntity
|
||||
{
|
||||
public string Name { get; set; } = string.Empty;
|
||||
public string Description { get; set; } = string.Empty;
|
||||
public string IconUrl { get; set; } = string.Empty;
|
||||
public AchievementType Type { get; set; }
|
||||
public int DiamondReward { get; set; } = 0;
|
||||
public int ExperienceReward { get; set; } = 0;
|
||||
public string? BadgeUrl { get; set; }
|
||||
public bool IsActive { get; set; } = true;
|
||||
|
||||
// Navigation Properties
|
||||
public virtual ICollection<UserAchievement> UserAchievements { get; set; } = new List<UserAchievement>();
|
||||
}
|
||||
|
||||
public class UserAchievement : BaseEntity
|
||||
{
|
||||
public Guid UserId { get; set; }
|
||||
public Guid AchievementId { get; set; }
|
||||
public DateTime AchievedAt { get; set; } = DateTime.UtcNow;
|
||||
public bool IsRewardClaimed { get; set; } = false;
|
||||
public DateTime? RewardClaimedAt { get; set; }
|
||||
|
||||
// Navigation Properties
|
||||
public virtual User User { get; set; } = null!;
|
||||
public virtual Achievement Achievement { get; set; } = null!;
|
||||
}
|
||||
|
||||
public class DailyMission : BaseEntity
|
||||
{
|
||||
public string Name { get; set; } = string.Empty;
|
||||
public string Description { get; set; } = string.Empty;
|
||||
public string IconUrl { get; set; } = string.Empty;
|
||||
public string Type { get; set; } = string.Empty; // vocabulary_recognition, dialogue_training, etc.
|
||||
public int TargetValue { get; set; } = 1;
|
||||
public string Unit { get; set; } = "次";
|
||||
public int ExperienceReward { get; set; } = 50;
|
||||
public int DiamondReward { get; set; } = 0;
|
||||
public bool IsActive { get; set; } = true;
|
||||
|
||||
// Navigation Properties
|
||||
public virtual ICollection<UserDailyMission> UserDailyMissions { get; set; } = new List<UserDailyMission>();
|
||||
}
|
||||
|
||||
public class UserDailyMission : BaseEntity
|
||||
{
|
||||
public Guid UserId { get; set; }
|
||||
public Guid DailyMissionId { get; set; }
|
||||
public DateTime MissionDate { get; set; }
|
||||
public int CurrentValue { get; set; } = 0;
|
||||
public int TargetValue { get; set; } = 1;
|
||||
public bool IsCompleted { get; set; } = false;
|
||||
public DateTime? CompletedAt { get; set; }
|
||||
public bool IsRewardClaimed { get; set; } = false;
|
||||
public DateTime? RewardClaimedAt { get; set; }
|
||||
|
||||
// Navigation Properties
|
||||
public virtual User User { get; set; } = null!;
|
||||
public virtual DailyMission DailyMission { get; set; } = null!;
|
||||
}
|
||||
|
||||
public class TimeWarpChallenge : BaseEntity
|
||||
{
|
||||
public Guid UserId { get; set; }
|
||||
public Guid ScenarioId { get; set; }
|
||||
public DateTime StartedAt { get; set; }
|
||||
public DateTime? CompletedAt { get; set; }
|
||||
public int TimeLimit { get; set; } = 300; // seconds
|
||||
public int TimeUsed { get; set; } = 0; // seconds
|
||||
public bool IsCompleted { get; set; } = false;
|
||||
public int Score { get; set; } = 0;
|
||||
public int ExperienceGained { get; set; } = 0;
|
||||
public int DiamondGained { get; set; } = 0;
|
||||
|
||||
// Navigation Properties
|
||||
public virtual User User { get; set; } = null!;
|
||||
public virtual Scenario Scenario { get; set; } = null!;
|
||||
}
|
||||
|
||||
public class UserLifePoint : BaseEntity
|
||||
{
|
||||
public Guid UserId { get; set; }
|
||||
public int CurrentLifePoints { get; set; } = 5;
|
||||
public int MaxLifePoints { get; set; } = 5;
|
||||
public DateTime? NextRecoveryAt { get; set; }
|
||||
public DateTime? LastUsedAt { get; set; }
|
||||
|
||||
// Navigation Properties
|
||||
public virtual User User { get; set; } = null!;
|
||||
}
|
||||
|
|
@ -0,0 +1,81 @@
|
|||
namespace DramaLing.Core.Entities;
|
||||
|
||||
public class LearningStage : BaseEntity
|
||||
{
|
||||
public string Name { get; set; } = string.Empty;
|
||||
public string Description { get; set; } = string.Empty;
|
||||
public int Order { get; set; }
|
||||
public string Level { get; set; } = string.Empty; // A1, A2, B1, etc.
|
||||
public bool IsActive { get; set; } = true;
|
||||
|
||||
// Navigation Properties
|
||||
public virtual ICollection<Scenario> Scenarios { get; set; } = new List<Scenario>();
|
||||
}
|
||||
|
||||
public class Scenario : BaseEntity
|
||||
{
|
||||
public string Name { get; set; } = string.Empty;
|
||||
public string Description { get; set; } = string.Empty;
|
||||
public string Context { get; set; } = string.Empty;
|
||||
public string Objective { get; set; } = string.Empty;
|
||||
public string Level { get; set; } = string.Empty;
|
||||
public int EstimatedMinutes { get; set; }
|
||||
public bool IsActive { get; set; } = true;
|
||||
|
||||
// Foreign Key
|
||||
public Guid LearningStageId { get; set; }
|
||||
|
||||
// Navigation Properties
|
||||
public virtual LearningStage LearningStage { get; set; } = null!;
|
||||
public virtual ICollection<Vocabulary> TargetVocabularies { get; set; } = new List<Vocabulary>();
|
||||
public virtual ICollection<UserProgress> UserProgresses { get; set; } = new List<UserProgress>();
|
||||
}
|
||||
|
||||
public class Vocabulary : BaseEntity
|
||||
{
|
||||
public string Word { get; set; } = string.Empty;
|
||||
public string Translation { get; set; } = string.Empty;
|
||||
public string Pronunciation { get; set; } = string.Empty;
|
||||
public string Definition { get; set; } = string.Empty;
|
||||
public string ExampleSentence { get; set; } = string.Empty;
|
||||
public string Level { get; set; } = string.Empty;
|
||||
public string Category { get; set; } = string.Empty;
|
||||
public int Frequency { get; set; } = 0; // Word frequency ranking
|
||||
|
||||
// Navigation Properties
|
||||
public virtual ICollection<Scenario> Scenarios { get; set; } = new List<Scenario>();
|
||||
public virtual ICollection<VocabularyProgress> VocabularyProgresses { get; set; } = new List<VocabularyProgress>();
|
||||
}
|
||||
|
||||
public class UserProgress : BaseEntity
|
||||
{
|
||||
public Guid UserId { get; set; }
|
||||
public Guid ScenarioId { get; set; }
|
||||
public int Score { get; set; }
|
||||
public int Stars { get; set; } // 0-3 stars
|
||||
public bool IsCompleted { get; set; }
|
||||
public DateTime? CompletedAt { get; set; }
|
||||
public int AttemptCount { get; set; } = 0;
|
||||
public string? FeedbackData { get; set; } // JSON data
|
||||
|
||||
// Navigation Properties
|
||||
public virtual User User { get; set; } = null!;
|
||||
public virtual Scenario Scenario { get; set; } = null!;
|
||||
}
|
||||
|
||||
public class VocabularyProgress : BaseEntity
|
||||
{
|
||||
public Guid UserId { get; set; }
|
||||
public Guid VocabularyId { get; set; }
|
||||
public int LearningStage { get; set; } = 1; // 1=Recognition, 2=Familiarity, 3=DialogueApplication
|
||||
public int MasteryLevel { get; set; } = 0; // 0-100
|
||||
public DateTime? NextReviewAt { get; set; }
|
||||
public int ReviewCount { get; set; } = 0;
|
||||
public int CorrectCount { get; set; } = 0;
|
||||
public int IncorrectCount { get; set; } = 0;
|
||||
public DateTime? LastReviewedAt { get; set; }
|
||||
|
||||
// Navigation Properties
|
||||
public virtual User User { get; set; } = null!;
|
||||
public virtual Vocabulary Vocabulary { get; set; } = null!;
|
||||
}
|
||||
|
|
@ -0,0 +1,39 @@
|
|||
using Microsoft.AspNetCore.Identity;
|
||||
|
||||
namespace DramaLing.Core.Entities;
|
||||
|
||||
public class User : IdentityUser<Guid>
|
||||
{
|
||||
public string DisplayName { get; set; } = string.Empty;
|
||||
public string? AvatarUrl { get; set; }
|
||||
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
|
||||
public DateTime UpdatedAt { get; set; } = DateTime.UtcNow;
|
||||
public DateTime? LastLoginAt { get; set; }
|
||||
public bool IsDeleted { get; set; } = false;
|
||||
|
||||
// Language Learning Properties
|
||||
public string CurrentLanguage { get; set; } = "en"; // ISO 639-1 code
|
||||
public string NativeLanguage { get; set; } = "zh"; // ISO 639-1 code
|
||||
public string CurrentLevel { get; set; } = "A1"; // CEFR level
|
||||
public int TotalExperience { get; set; } = 0;
|
||||
public int Diamonds { get; set; } = 0;
|
||||
public int LightningEnergy { get; set; } = 0;
|
||||
public int LifePoints { get; set; } = 5;
|
||||
public int MaxLifePoints { get; set; } = 5;
|
||||
public DateTime? NextLifePointRecovery { get; set; }
|
||||
|
||||
// Subscription
|
||||
public bool IsVipUser { get; set; } = false;
|
||||
public DateTime? VipExpiresAt { get; set; }
|
||||
|
||||
// Learning Statistics
|
||||
public int ConsecutiveDays { get; set; } = 0;
|
||||
public DateTime? LastLearningDate { get; set; }
|
||||
public int TotalDialoguesCompleted { get; set; } = 0;
|
||||
public int TotalVocabularyMastered { get; set; } = 0;
|
||||
|
||||
// Navigation Properties
|
||||
public virtual ICollection<UserProgress> UserProgresses { get; set; } = new List<UserProgress>();
|
||||
public virtual ICollection<UserAchievement> UserAchievements { get; set; } = new List<UserAchievement>();
|
||||
public virtual ICollection<VocabularyProgress> VocabularyProgresses { get; set; } = new List<VocabularyProgress>();
|
||||
}
|
||||
|
|
@ -0,0 +1,31 @@
|
|||
namespace DramaLing.Core.Enums;
|
||||
|
||||
public enum LearningStage
|
||||
{
|
||||
Recognition = 1, // 詞彙認識
|
||||
Familiarity = 2, // 詞彙熟悉
|
||||
DialogueApplication = 3 // 對話應用
|
||||
}
|
||||
|
||||
public enum DifficultyLevel
|
||||
{
|
||||
A1 = 1,
|
||||
A2 = 2,
|
||||
B1 = 3,
|
||||
B2 = 4,
|
||||
C1 = 5,
|
||||
C2 = 6
|
||||
}
|
||||
|
||||
public enum AchievementType
|
||||
{
|
||||
PassReward = 1, // 過關獎勵
|
||||
PerfectGrammar = 2, // 完美語法
|
||||
FluentExpression = 3, // 表達流利
|
||||
StoryMaster = 4, // 劇情大師
|
||||
VocabularyExpert = 5, // 詞彙專家
|
||||
PerfectDialogue = 6, // 完美對話
|
||||
SmartLearner = 7, // 智慧學習者
|
||||
IndependentProgress = 8, // 獨立進步
|
||||
TranslationMaster = 9 // 翻譯達人
|
||||
}
|
||||
|
|
@ -0,0 +1,99 @@
|
|||
using DramaLing.Core.Entities;
|
||||
using Microsoft.AspNetCore.Identity;
|
||||
using Microsoft.AspNetCore.Identity.EntityFrameworkCore;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
|
||||
namespace DramaLing.Infrastructure.Data;
|
||||
|
||||
public class ApplicationDbContext : IdentityDbContext<User, IdentityRole<Guid>, Guid>
|
||||
{
|
||||
public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options) : base(options)
|
||||
{
|
||||
}
|
||||
|
||||
// Learning Content
|
||||
public DbSet<LearningStage> LearningStages { get; set; }
|
||||
public DbSet<Scenario> Scenarios { get; set; }
|
||||
public DbSet<Vocabulary> Vocabularies { get; set; }
|
||||
|
||||
// User Progress
|
||||
public DbSet<UserProgress> UserProgresses { get; set; }
|
||||
public DbSet<VocabularyProgress> VocabularyProgresses { get; set; }
|
||||
|
||||
// Gamification
|
||||
public DbSet<Achievement> Achievements { get; set; }
|
||||
public DbSet<UserAchievement> UserAchievements { get; set; }
|
||||
public DbSet<DailyMission> DailyMissions { get; set; }
|
||||
public DbSet<UserDailyMission> UserDailyMissions { get; set; }
|
||||
|
||||
protected override void OnModelCreating(ModelBuilder builder)
|
||||
{
|
||||
base.OnModelCreating(builder);
|
||||
|
||||
// Configure Identity tables to use Guid
|
||||
builder.Entity<User>(entity =>
|
||||
{
|
||||
entity.ToTable("Users");
|
||||
entity.HasKey(e => e.Id);
|
||||
|
||||
// Configure indexes
|
||||
entity.HasIndex(e => e.Email).IsUnique();
|
||||
entity.HasIndex(e => e.UserName).IsUnique();
|
||||
|
||||
// Configure properties
|
||||
entity.Property(e => e.DisplayName).HasMaxLength(100).IsRequired();
|
||||
entity.Property(e => e.CurrentLanguage).HasMaxLength(5).IsRequired();
|
||||
entity.Property(e => e.NativeLanguage).HasMaxLength(5).IsRequired();
|
||||
entity.Property(e => e.CurrentLevel).HasMaxLength(5).IsRequired();
|
||||
});
|
||||
|
||||
builder.Entity<IdentityRole<Guid>>(entity =>
|
||||
{
|
||||
entity.ToTable("Roles");
|
||||
});
|
||||
|
||||
builder.Entity<IdentityUserRole<Guid>>(entity =>
|
||||
{
|
||||
entity.ToTable("UserRoles");
|
||||
});
|
||||
|
||||
builder.Entity<IdentityUserClaim<Guid>>(entity =>
|
||||
{
|
||||
entity.ToTable("UserClaims");
|
||||
});
|
||||
|
||||
builder.Entity<IdentityUserLogin<Guid>>(entity =>
|
||||
{
|
||||
entity.ToTable("UserLogins");
|
||||
});
|
||||
|
||||
builder.Entity<IdentityRoleClaim<Guid>>(entity =>
|
||||
{
|
||||
entity.ToTable("RoleClaims");
|
||||
});
|
||||
|
||||
builder.Entity<IdentityUserToken<Guid>>(entity =>
|
||||
{
|
||||
entity.ToTable("UserTokens");
|
||||
});
|
||||
|
||||
// Configure soft delete
|
||||
builder.Entity<User>()
|
||||
.HasQueryFilter(e => !e.IsDeleted);
|
||||
}
|
||||
|
||||
public override async Task<int> SaveChangesAsync(CancellationToken cancellationToken = default)
|
||||
{
|
||||
foreach (var entry in ChangeTracker.Entries<BaseEntity>())
|
||||
{
|
||||
switch (entry.State)
|
||||
{
|
||||
case EntityState.Modified:
|
||||
entry.Entity.UpdatedAt = DateTime.UtcNow;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return await base.SaveChangesAsync(cancellationToken);
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,81 @@
|
|||
using DramaLing.Core.Entities;
|
||||
using DramaLing.Infrastructure.Data;
|
||||
using Microsoft.AspNetCore.Authentication.JwtBearer;
|
||||
using Microsoft.AspNetCore.Identity;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.Extensions.Configuration;
|
||||
using Microsoft.Extensions.DependencyInjection;
|
||||
using Microsoft.IdentityModel.Tokens;
|
||||
using StackExchange.Redis;
|
||||
using System.Text;
|
||||
|
||||
namespace DramaLing.Infrastructure;
|
||||
|
||||
public static class DependencyInjection
|
||||
{
|
||||
public static IServiceCollection AddInfrastructure(
|
||||
this IServiceCollection services,
|
||||
IConfiguration configuration)
|
||||
{
|
||||
// Database
|
||||
services.AddDbContext<ApplicationDbContext>(options =>
|
||||
options.UseNpgsql(configuration.GetConnectionString("DefaultConnection")));
|
||||
|
||||
// Identity
|
||||
services.AddIdentity<User, IdentityRole<Guid>>(options =>
|
||||
{
|
||||
// Password settings
|
||||
options.Password.RequireDigit = true;
|
||||
options.Password.RequireUppercase = true;
|
||||
options.Password.RequiredLength = 8;
|
||||
options.Password.RequireNonAlphanumeric = false;
|
||||
|
||||
// Lockout settings
|
||||
options.Lockout.DefaultLockoutTimeSpan = TimeSpan.FromMinutes(5);
|
||||
options.Lockout.MaxFailedAccessAttempts = 5;
|
||||
options.Lockout.AllowedForNewUsers = true;
|
||||
|
||||
// User settings
|
||||
options.User.RequireUniqueEmail = true;
|
||||
options.SignIn.RequireConfirmedEmail = false;
|
||||
})
|
||||
.AddEntityFrameworkStores<ApplicationDbContext>()
|
||||
.AddDefaultTokenProviders();
|
||||
|
||||
// JWT Authentication
|
||||
var jwtSettings = configuration.GetSection("JwtSettings");
|
||||
var key = Encoding.ASCII.GetBytes(jwtSettings["Key"] ?? throw new InvalidOperationException("JWT Key not found"));
|
||||
|
||||
services.AddAuthentication(options =>
|
||||
{
|
||||
options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
|
||||
options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
|
||||
})
|
||||
.AddJwtBearer(options =>
|
||||
{
|
||||
options.RequireHttpsMetadata = false;
|
||||
options.SaveToken = true;
|
||||
options.TokenValidationParameters = new TokenValidationParameters
|
||||
{
|
||||
ValidateIssuerSigningKey = true,
|
||||
IssuerSigningKey = new SymmetricSecurityKey(key),
|
||||
ValidateIssuer = true,
|
||||
ValidIssuer = jwtSettings["Issuer"],
|
||||
ValidateAudience = true,
|
||||
ValidAudience = jwtSettings["Audience"],
|
||||
ValidateLifetime = true,
|
||||
ClockSkew = TimeSpan.Zero
|
||||
};
|
||||
});
|
||||
|
||||
// Redis
|
||||
var redisConnection = configuration.GetConnectionString("Redis");
|
||||
if (!string.IsNullOrEmpty(redisConnection))
|
||||
{
|
||||
services.AddSingleton<IConnectionMultiplexer>(sp =>
|
||||
ConnectionMultiplexer.Connect(redisConnection));
|
||||
}
|
||||
|
||||
return services;
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,25 @@
|
|||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Tools" Version="8.0.0" />
|
||||
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.AspNetCore.Identity.EntityFrameworkCore" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="8.0.0" />
|
||||
<PackageReference Include="StackExchange.Redis" Version="2.6.122" />
|
||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="8.0.0" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\DramaLing.Core\DramaLing.Core.csproj" />
|
||||
<ProjectReference Include="..\DramaLing.Application\DramaLing.Application.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
|
@ -0,0 +1,48 @@
|
|||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio Version 17
|
||||
VisualStudioVersion = 17.0.31903.59
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DramaLing.API", "DramaLing.API\DramaLing.API.csproj", "{8A7E8B45-1234-4567-8901-234567890123}"
|
||||
EndProject
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DramaLing.Core", "DramaLing.Core\DramaLing.Core.csproj", "{8A7E8B45-1234-4567-8901-234567890124}"
|
||||
EndProject
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DramaLing.Infrastructure", "DramaLing.Infrastructure\DramaLing.Infrastructure.csproj", "{8A7E8B45-1234-4567-8901-234567890125}"
|
||||
EndProject
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DramaLing.Application", "DramaLing.Application\DramaLing.Application.csproj", "{8A7E8B45-1234-4567-8901-234567890126}"
|
||||
EndProject
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DramaLing.Tests", "DramaLing.Tests\DramaLing.Tests.csproj", "{8A7E8B45-1234-4567-8901-234567890127}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Release|Any CPU = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{8A7E8B45-1234-4567-8901-234567890123}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890123}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890123}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890123}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890124}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890124}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890124}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890124}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890125}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890125}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890125}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890125}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890126}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890126}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890126}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890126}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890127}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890127}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890127}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{8A7E8B45-1234-4567-8901-234567890127}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
EndGlobalSection
|
||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||
SolutionGuid = {12345678-1234-5678-9012-123456789012}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
||||
|
|
@ -0,0 +1,45 @@
|
|||
# Drama Ling Backend API
|
||||
|
||||
.NET Core 後端 API 服務
|
||||
|
||||
## 技術棧
|
||||
- **.NET 8**: 跨平台框架
|
||||
- **ASP.NET Core Web API**: RESTful API
|
||||
- **Entity Framework Core**: ORM 資料庫存取
|
||||
- **PostgreSQL**: 主要資料庫
|
||||
- **Redis**: 快取和會話管理
|
||||
- **JWT**: 身份驗證
|
||||
|
||||
## 專案結構
|
||||
```
|
||||
backend/
|
||||
├── DramaLing.API/ # Web API 專案
|
||||
├── DramaLing.Application/ # 應用服務層
|
||||
├── DramaLing.Core/ # 領域模型層
|
||||
├── DramaLing.Infrastructure/ # 基礎設施層
|
||||
├── DramaLing.Tests/ # 測試專案
|
||||
└── DramaLing.sln # 解決方案檔
|
||||
```
|
||||
|
||||
## 快速開始
|
||||
|
||||
### 1. 安裝相依套件
|
||||
```bash
|
||||
dotnet restore
|
||||
```
|
||||
|
||||
### 2. 設定資料庫
|
||||
```bash
|
||||
# 建立資料庫
|
||||
dotnet ef database update --project DramaLing.Infrastructure --startup-project DramaLing.API
|
||||
```
|
||||
|
||||
### 3. 啟動開發服務器
|
||||
```bash
|
||||
dotnet run --project DramaLing.API
|
||||
# API: http://localhost:5000
|
||||
# Swagger: http://localhost:5000
|
||||
```
|
||||
|
||||
## 開發指南
|
||||
詳細開發文檔請參考:`../../docs/04_technical/`
|
||||
|
|
@ -0,0 +1,45 @@
|
|||
# Miscellaneous
|
||||
*.class
|
||||
*.log
|
||||
*.pyc
|
||||
*.swp
|
||||
.DS_Store
|
||||
.atom/
|
||||
.build/
|
||||
.buildlog/
|
||||
.history
|
||||
.svn/
|
||||
.swiftpm/
|
||||
migrate_working_dir/
|
||||
|
||||
# IntelliJ related
|
||||
*.iml
|
||||
*.ipr
|
||||
*.iws
|
||||
.idea/
|
||||
|
||||
# The .vscode folder contains launch configuration and tasks you configure in
|
||||
# VS Code which you may wish to be included in version control, so this line
|
||||
# is commented out by default.
|
||||
#.vscode/
|
||||
|
||||
# Flutter/Dart/Pub related
|
||||
**/doc/api/
|
||||
**/ios/Flutter/.last_build_id
|
||||
.dart_tool/
|
||||
.flutter-plugins-dependencies
|
||||
.pub-cache/
|
||||
.pub/
|
||||
/build/
|
||||
/coverage/
|
||||
|
||||
# Symbolication related
|
||||
app.*.symbols
|
||||
|
||||
# Obfuscation related
|
||||
app.*.map.json
|
||||
|
||||
# Android Studio will place build artifacts here
|
||||
/android/app/debug
|
||||
/android/app/profile
|
||||
/android/app/release
|
||||
|
|
@ -0,0 +1,33 @@
|
|||
# This file tracks properties of this Flutter project.
|
||||
# Used by Flutter tool to assess capabilities and perform upgrades etc.
|
||||
#
|
||||
# This file should be version controlled and should not be manually edited.
|
||||
|
||||
version:
|
||||
revision: "a402d9a4376add5bc2d6b1e33e53edaae58c07f8"
|
||||
channel: "stable"
|
||||
|
||||
project_type: app
|
||||
|
||||
# Tracks metadata for the flutter migrate command
|
||||
migration:
|
||||
platforms:
|
||||
- platform: root
|
||||
create_revision: a402d9a4376add5bc2d6b1e33e53edaae58c07f8
|
||||
base_revision: a402d9a4376add5bc2d6b1e33e53edaae58c07f8
|
||||
- platform: android
|
||||
create_revision: a402d9a4376add5bc2d6b1e33e53edaae58c07f8
|
||||
base_revision: a402d9a4376add5bc2d6b1e33e53edaae58c07f8
|
||||
- platform: ios
|
||||
create_revision: a402d9a4376add5bc2d6b1e33e53edaae58c07f8
|
||||
base_revision: a402d9a4376add5bc2d6b1e33e53edaae58c07f8
|
||||
|
||||
# User provided section
|
||||
|
||||
# List of Local paths (relative to this file) that should be
|
||||
# ignored by the migrate tool.
|
||||
#
|
||||
# Files that are not part of the templates will be ignored by default.
|
||||
unmanaged_files:
|
||||
- 'lib/main.dart'
|
||||
- 'ios/Runner.xcodeproj/project.pbxproj'
|
||||
|
|
@ -0,0 +1,43 @@
|
|||
# Drama Ling Mobile App
|
||||
|
||||
Flutter 移動端應用程式
|
||||
|
||||
## 技術棧
|
||||
- **Flutter 3.16+**: 跨平台框架
|
||||
- **Dart 3.0+**: 程式語言
|
||||
- **Riverpod**: 狀態管理
|
||||
- **Go Router**: 導航路由
|
||||
- **Dio + Retrofit**: HTTP 客戶端
|
||||
- **Hive**: 本地資料存儲
|
||||
- **Material 3**: UI 設計系統
|
||||
|
||||
## 專案結構
|
||||
```
|
||||
mobile/
|
||||
├── lib/
|
||||
│ ├── core/ # 核心功能 (常數、工具、服務)
|
||||
│ ├── features/ # 功能模組 (認證、學習、對話等)
|
||||
│ └── shared/ # 共用組件 (Widget、模型、Provider)
|
||||
└── pubspec.yaml # Flutter 專案配置
|
||||
```
|
||||
|
||||
## 快速開始
|
||||
|
||||
### 1. 安裝相依套件
|
||||
```bash
|
||||
flutter pub get
|
||||
```
|
||||
|
||||
### 2. 程式碼生成
|
||||
```bash
|
||||
dart run build_runner build
|
||||
```
|
||||
|
||||
### 3. 啟動應用
|
||||
```bash
|
||||
flutter run
|
||||
# 需要模擬器或實體裝置
|
||||
```
|
||||
|
||||
## 開發指南
|
||||
詳細開發文檔請參考:`../../docs/04_technical/`
|
||||
|
|
@ -0,0 +1,28 @@
|
|||
# This file configures the analyzer, which statically analyzes Dart code to
|
||||
# check for errors, warnings, and lints.
|
||||
#
|
||||
# The issues identified by the analyzer are surfaced in the UI of Dart-enabled
|
||||
# IDEs (https://dart.dev/tools#ides-and-editors). The analyzer can also be
|
||||
# invoked from the command line by running `flutter analyze`.
|
||||
|
||||
# The following line activates a set of recommended lints for Flutter apps,
|
||||
# packages, and plugins designed to encourage good coding practices.
|
||||
include: package:flutter_lints/flutter.yaml
|
||||
|
||||
linter:
|
||||
# The lint rules applied to this project can be customized in the
|
||||
# section below to disable rules from the `package:flutter_lints/flutter.yaml`
|
||||
# included above or to enable additional rules. A list of all available lints
|
||||
# and their documentation is published at https://dart.dev/lints.
|
||||
#
|
||||
# Instead of disabling a lint rule for the entire project in the
|
||||
# section below, it can also be suppressed for a single line of code
|
||||
# or a specific dart file by using the `// ignore: name_of_lint` and
|
||||
# `// ignore_for_file: name_of_lint` syntax on the line or in the file
|
||||
# producing the lint.
|
||||
rules:
|
||||
# avoid_print: false # Uncomment to disable the `avoid_print` rule
|
||||
# prefer_single_quotes: true # Uncomment to enable the `prefer_single_quotes` rule
|
||||
|
||||
# Additional information about this file can be found at
|
||||
# https://dart.dev/guides/language/analysis-options
|
||||
|
|
@ -0,0 +1,14 @@
|
|||
gradle-wrapper.jar
|
||||
/.gradle
|
||||
/captures/
|
||||
/gradlew
|
||||
/gradlew.bat
|
||||
/local.properties
|
||||
GeneratedPluginRegistrant.java
|
||||
.cxx/
|
||||
|
||||
# Remember to never publicly share your keystore.
|
||||
# See https://flutter.dev/to/reference-keystore
|
||||
key.properties
|
||||
**/*.keystore
|
||||
**/*.jks
|
||||
|
|
@ -0,0 +1,52 @@
|
|||
plugins {
|
||||
id("com.android.application")
|
||||
id("kotlin-android")
|
||||
// The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins.
|
||||
id("dev.flutter.flutter-gradle-plugin")
|
||||
}
|
||||
|
||||
android {
|
||||
namespace = "com.dramaling.app"
|
||||
compileSdk = flutter.compileSdkVersion
|
||||
ndkVersion = flutter.ndkVersion
|
||||
|
||||
compileOptions {
|
||||
sourceCompatibility = JavaVersion.VERSION_11
|
||||
targetCompatibility = JavaVersion.VERSION_11
|
||||
}
|
||||
|
||||
kotlinOptions {
|
||||
jvmTarget = JavaVersion.VERSION_11.toString()
|
||||
}
|
||||
|
||||
defaultConfig {
|
||||
applicationId = "com.dramaling.app"
|
||||
// You can update the following values to match your application needs.
|
||||
// For more information, see: https://flutter.dev/to/review-gradle-config.
|
||||
minSdk = flutter.minSdkVersion
|
||||
targetSdk = flutter.targetSdkVersion
|
||||
versionCode = flutter.versionCode
|
||||
versionName = flutter.versionName
|
||||
}
|
||||
|
||||
buildTypes {
|
||||
release {
|
||||
// Signing with debug keys for development
|
||||
// In production, replace with proper signing configuration
|
||||
signingConfig = signingConfigs.getByName("debug")
|
||||
|
||||
// Enable code shrinking, obfuscation, and optimization
|
||||
isMinifyEnabled = true
|
||||
isShrinkResources = true
|
||||
proguardFiles(getDefaultProguardFile("proguard-android-optimize.txt"), "proguard-rules.pro")
|
||||
}
|
||||
debug {
|
||||
applicationIdSuffix = ".debug"
|
||||
isDebuggable = true
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
flutter {
|
||||
source = "../.."
|
||||
}
|
||||
|
|
@ -0,0 +1,42 @@
|
|||
# Flutter specific rules
|
||||
-keep class io.flutter.app.** { *; }
|
||||
-keep class io.flutter.plugin.** { *; }
|
||||
-keep class io.flutter.util.** { *; }
|
||||
-keep class io.flutter.view.** { *; }
|
||||
-keep class io.flutter.** { *; }
|
||||
-keep class io.flutter.plugins.** { *; }
|
||||
-dontwarn io.flutter.embedding.**
|
||||
|
||||
# Keep Flutter engine native methods
|
||||
-keep class io.flutter.embedding.engine.FlutterJNI { *; }
|
||||
|
||||
# Audio players plugin
|
||||
-keep class com.ryanheise.just_audio.** { *; }
|
||||
-keep class xyz.luan.audioplayers.** { *; }
|
||||
|
||||
# Network (Dio/Retrofit)
|
||||
-keepattributes Signature
|
||||
-keepattributes *Annotation*
|
||||
-keep class retrofit2.** { *; }
|
||||
-keep class com.google.gson.** { *; }
|
||||
-keepclassmembers,allowobfuscation class * {
|
||||
@com.google.gson.annotations.SerializedName <fields>;
|
||||
}
|
||||
|
||||
# Riverpod
|
||||
-keep class * extends com.riverpod.** { *; }
|
||||
|
||||
# Flutter secure storage
|
||||
-keep class com.it_nomads.fluttersecurestorage.** { *; }
|
||||
|
||||
# General Android rules
|
||||
-keepclassmembers class * implements android.os.Parcelable {
|
||||
public static final android.os.Parcelable$Creator CREATOR;
|
||||
}
|
||||
|
||||
# Remove logging in release
|
||||
-assumenosideeffects class android.util.Log {
|
||||
public static *** d(...);
|
||||
public static *** v(...);
|
||||
public static *** i(...);
|
||||
}
|
||||
|
|
@ -0,0 +1,61 @@
|
|||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<!-- Network permissions -->
|
||||
<uses-permission android:name="android.permission.INTERNET" />
|
||||
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
|
||||
|
||||
<!-- Audio permissions -->
|
||||
<uses-permission android:name="android.permission.RECORD_AUDIO" />
|
||||
<uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />
|
||||
<uses-permission android:name="android.permission.WAKE_LOCK" />
|
||||
|
||||
<!-- Storage permissions -->
|
||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
|
||||
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
|
||||
|
||||
<!-- Optional: Vibrate for user feedback -->
|
||||
<uses-permission android:name="android.permission.VIBRATE" />
|
||||
|
||||
<application
|
||||
android:label="Drama Ling"
|
||||
android:name="${applicationName}"
|
||||
android:icon="@mipmap/ic_launcher">
|
||||
<activity
|
||||
android:name=".MainActivity"
|
||||
android:exported="true"
|
||||
android:launchMode="singleTop"
|
||||
android:taskAffinity=""
|
||||
android:theme="@style/LaunchTheme"
|
||||
android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode"
|
||||
android:hardwareAccelerated="true"
|
||||
android:windowSoftInputMode="adjustResize">
|
||||
<!-- Specifies an Android theme to apply to this Activity as soon as
|
||||
the Android process has started. This theme is visible to the user
|
||||
while the Flutter UI initializes. After that, this theme continues
|
||||
to determine the Window background behind the Flutter UI. -->
|
||||
<meta-data
|
||||
android:name="io.flutter.embedding.android.NormalTheme"
|
||||
android:resource="@style/NormalTheme"
|
||||
/>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MAIN"/>
|
||||
<category android:name="android.intent.category.LAUNCHER"/>
|
||||
</intent-filter>
|
||||
</activity>
|
||||
<!-- Don't delete the meta-data below.
|
||||
This is used by the Flutter tool to generate GeneratedPluginRegistrant.java -->
|
||||
<meta-data
|
||||
android:name="flutterEmbedding"
|
||||
android:value="2" />
|
||||
</application>
|
||||
<!-- Required to query activities that can process text, see:
|
||||
https://developer.android.com/training/package-visibility and
|
||||
https://developer.android.com/reference/android/content/Intent#ACTION_PROCESS_TEXT.
|
||||
|
||||
In particular, this is used by the Flutter engine in io.flutter.plugin.text.ProcessTextPlugin. -->
|
||||
<queries>
|
||||
<intent>
|
||||
<action android:name="android.intent.action.PROCESS_TEXT"/>
|
||||
<data android:mimeType="text/plain"/>
|
||||
</intent>
|
||||
</queries>
|
||||
</manifest>
|
||||
|
|
@ -0,0 +1,5 @@
|
|||
package com.dramaling.app
|
||||
|
||||
import io.flutter.embedding.android.FlutterActivity
|
||||
|
||||
class MainActivity : FlutterActivity()
|
||||
|
|
@ -0,0 +1,12 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Modify this file to customize your launch splash screen -->
|
||||
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<item android:drawable="?android:colorBackground" />
|
||||
|
||||
<!-- You can insert your own image assets here -->
|
||||
<!-- <item>
|
||||
<bitmap
|
||||
android:gravity="center"
|
||||
android:src="@mipmap/launch_image" />
|
||||
</item> -->
|
||||
</layer-list>
|
||||
|
|
@ -0,0 +1,12 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Modify this file to customize your launch splash screen -->
|
||||
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<item android:drawable="@android:color/white" />
|
||||
|
||||
<!-- You can insert your own image assets here -->
|
||||
<!-- <item>
|
||||
<bitmap
|
||||
android:gravity="center"
|
||||
android:src="@mipmap/launch_image" />
|
||||
</item> -->
|
||||
</layer-list>
|
||||
|
After Width: | Height: | Size: 544 B |
|
After Width: | Height: | Size: 442 B |
|
After Width: | Height: | Size: 721 B |
|
After Width: | Height: | Size: 1.0 KiB |
|
After Width: | Height: | Size: 1.4 KiB |
|
|
@ -0,0 +1,18 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<!-- Theme applied to the Android Window while the process is starting when the OS's Dark Mode setting is on -->
|
||||
<style name="LaunchTheme" parent="@android:style/Theme.Black.NoTitleBar">
|
||||
<!-- Show a splash screen on the activity. Automatically removed when
|
||||
the Flutter engine draws its first frame -->
|
||||
<item name="android:windowBackground">@drawable/launch_background</item>
|
||||
</style>
|
||||
<!-- Theme applied to the Android Window as soon as the process has started.
|
||||
This theme determines the color of the Android Window while your
|
||||
Flutter UI initializes, as well as behind your Flutter UI while its
|
||||
running.
|
||||
|
||||
This Theme is only used starting with V2 of Flutter's Android embedding. -->
|
||||
<style name="NormalTheme" parent="@android:style/Theme.Black.NoTitleBar">
|
||||
<item name="android:windowBackground">?android:colorBackground</item>
|
||||
</style>
|
||||
</resources>
|
||||
|
|
@ -0,0 +1,18 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<!-- Theme applied to the Android Window while the process is starting when the OS's Dark Mode setting is off -->
|
||||
<style name="LaunchTheme" parent="@android:style/Theme.Light.NoTitleBar">
|
||||
<!-- Show a splash screen on the activity. Automatically removed when
|
||||
the Flutter engine draws its first frame -->
|
||||
<item name="android:windowBackground">@drawable/launch_background</item>
|
||||
</style>
|
||||
<!-- Theme applied to the Android Window as soon as the process has started.
|
||||
This theme determines the color of the Android Window while your
|
||||
Flutter UI initializes, as well as behind your Flutter UI while its
|
||||
running.
|
||||
|
||||
This Theme is only used starting with V2 of Flutter's Android embedding. -->
|
||||
<style name="NormalTheme" parent="@android:style/Theme.Light.NoTitleBar">
|
||||
<item name="android:windowBackground">?android:colorBackground</item>
|
||||
</style>
|
||||
</resources>
|
||||
|
|
@ -0,0 +1,24 @@
|
|||
allprojects {
|
||||
repositories {
|
||||
google()
|
||||
mavenCentral()
|
||||
}
|
||||
}
|
||||
|
||||
val newBuildDir: Directory =
|
||||
rootProject.layout.buildDirectory
|
||||
.dir("../../build")
|
||||
.get()
|
||||
rootProject.layout.buildDirectory.value(newBuildDir)
|
||||
|
||||
subprojects {
|
||||
val newSubprojectBuildDir: Directory = newBuildDir.dir(project.name)
|
||||
project.layout.buildDirectory.value(newSubprojectBuildDir)
|
||||
}
|
||||
subprojects {
|
||||
project.evaluationDependsOn(":app")
|
||||
}
|
||||
|
||||
tasks.register<Delete>("clean") {
|
||||
delete(rootProject.layout.buildDirectory)
|
||||
}
|
||||
|
|
@ -0,0 +1,3 @@
|
|||
org.gradle.jvmargs=-Xmx8G -XX:MaxMetaspaceSize=4G -XX:ReservedCodeCacheSize=512m -XX:+HeapDumpOnOutOfMemoryError
|
||||
android.useAndroidX=true
|
||||
android.enableJetifier=true
|
||||
|
|
@ -0,0 +1,5 @@
|
|||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-8.12-all.zip
|
||||
|
|
@ -0,0 +1,26 @@
|
|||
pluginManagement {
|
||||
val flutterSdkPath =
|
||||
run {
|
||||
val properties = java.util.Properties()
|
||||
file("local.properties").inputStream().use { properties.load(it) }
|
||||
val flutterSdkPath = properties.getProperty("flutter.sdk")
|
||||
require(flutterSdkPath != null) { "flutter.sdk not set in local.properties" }
|
||||
flutterSdkPath
|
||||
}
|
||||
|
||||
includeBuild("$flutterSdkPath/packages/flutter_tools/gradle")
|
||||
|
||||
repositories {
|
||||
google()
|
||||
mavenCentral()
|
||||
gradlePluginPortal()
|
||||
}
|
||||
}
|
||||
|
||||
plugins {
|
||||
id("dev.flutter.flutter-plugin-loader") version "1.0.0"
|
||||
id("com.android.application") version "8.9.1" apply false
|
||||
id("org.jetbrains.kotlin.android") version "2.1.0" apply false
|
||||
}
|
||||
|
||||
include(":app")
|
||||
|
|
@ -0,0 +1,34 @@
|
|||
**/dgph
|
||||
*.mode1v3
|
||||
*.mode2v3
|
||||
*.moved-aside
|
||||
*.pbxuser
|
||||
*.perspectivev3
|
||||
**/*sync/
|
||||
.sconsign.dblite
|
||||
.tags*
|
||||
**/.vagrant/
|
||||
**/DerivedData/
|
||||
Icon?
|
||||
**/Pods/
|
||||
**/.symlinks/
|
||||
profile
|
||||
xcuserdata
|
||||
**/.generated/
|
||||
Flutter/App.framework
|
||||
Flutter/Flutter.framework
|
||||
Flutter/Flutter.podspec
|
||||
Flutter/Generated.xcconfig
|
||||
Flutter/ephemeral/
|
||||
Flutter/app.flx
|
||||
Flutter/app.zip
|
||||
Flutter/flutter_assets/
|
||||
Flutter/flutter_export_environment.sh
|
||||
ServiceDefinitions.json
|
||||
Runner/GeneratedPluginRegistrant.*
|
||||
|
||||
# Exceptions to above rules.
|
||||
!default.mode1v3
|
||||
!default.mode2v3
|
||||
!default.pbxuser
|
||||
!default.perspectivev3
|
||||
|
|
@ -0,0 +1,26 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>en</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>App</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>io.flutter.flutter.app</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>App</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>FMWK</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>1.0</string>
|
||||
<key>CFBundleSignature</key>
|
||||
<string>????</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>1.0</string>
|
||||
<key>MinimumOSVersion</key>
|
||||
<string>13.0</string>
|
||||
</dict>
|
||||
</plist>
|
||||
|
|
@ -0,0 +1,2 @@
|
|||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"
|
||||
#include "Generated.xcconfig"
|
||||
|
|
@ -0,0 +1,2 @@
|
|||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"
|
||||
#include "Generated.xcconfig"
|
||||
|
|
@ -0,0 +1,43 @@
|
|||
# Uncomment this line to define a global platform for your project
|
||||
# platform :ios, '13.0'
|
||||
|
||||
# CocoaPods analytics sends network stats synchronously affecting flutter build latency.
|
||||
ENV['COCOAPODS_DISABLE_STATS'] = 'true'
|
||||
|
||||
project 'Runner', {
|
||||
'Debug' => :debug,
|
||||
'Profile' => :release,
|
||||
'Release' => :release,
|
||||
}
|
||||
|
||||
def flutter_root
|
||||
generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'Generated.xcconfig'), __FILE__)
|
||||
unless File.exist?(generated_xcode_build_settings_path)
|
||||
raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure flutter pub get is executed first"
|
||||
end
|
||||
|
||||
File.foreach(generated_xcode_build_settings_path) do |line|
|
||||
matches = line.match(/FLUTTER_ROOT\=(.*)/)
|
||||
return matches[1].strip if matches
|
||||
end
|
||||
raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Generated.xcconfig, then run flutter pub get"
|
||||
end
|
||||
|
||||
require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root)
|
||||
|
||||
flutter_ios_podfile_setup
|
||||
|
||||
target 'Runner' do
|
||||
use_frameworks!
|
||||
|
||||
flutter_install_all_ios_pods File.dirname(File.realpath(__FILE__))
|
||||
target 'RunnerTests' do
|
||||
inherit! :search_paths
|
||||
end
|
||||
end
|
||||
|
||||
post_install do |installer|
|
||||
installer.pods_project.targets.each do |target|
|
||||
flutter_additional_ios_build_settings(target)
|
||||
end
|
||||
end
|
||||
|
|
@ -0,0 +1,81 @@
|
|||
PODS:
|
||||
- audio_session (0.0.1):
|
||||
- Flutter
|
||||
- audioplayers_darwin (0.0.1):
|
||||
- Flutter
|
||||
- Flutter (1.0.0)
|
||||
- flutter_secure_storage (6.0.0):
|
||||
- Flutter
|
||||
- just_audio (0.0.1):
|
||||
- Flutter
|
||||
- FlutterMacOS
|
||||
- path_provider_foundation (0.0.1):
|
||||
- Flutter
|
||||
- FlutterMacOS
|
||||
- permission_handler_apple (9.3.0):
|
||||
- Flutter
|
||||
- shared_preferences_foundation (0.0.1):
|
||||
- Flutter
|
||||
- FlutterMacOS
|
||||
- speech_to_text (0.0.1):
|
||||
- Flutter
|
||||
- Try
|
||||
- sqflite_darwin (0.0.4):
|
||||
- Flutter
|
||||
- FlutterMacOS
|
||||
- Try (2.1.1)
|
||||
|
||||
DEPENDENCIES:
|
||||
- audio_session (from `.symlinks/plugins/audio_session/ios`)
|
||||
- audioplayers_darwin (from `.symlinks/plugins/audioplayers_darwin/ios`)
|
||||
- Flutter (from `Flutter`)
|
||||
- flutter_secure_storage (from `.symlinks/plugins/flutter_secure_storage/ios`)
|
||||
- just_audio (from `.symlinks/plugins/just_audio/darwin`)
|
||||
- path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`)
|
||||
- permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`)
|
||||
- shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`)
|
||||
- speech_to_text (from `.symlinks/plugins/speech_to_text/ios`)
|
||||
- sqflite_darwin (from `.symlinks/plugins/sqflite_darwin/darwin`)
|
||||
|
||||
SPEC REPOS:
|
||||
trunk:
|
||||
- Try
|
||||
|
||||
EXTERNAL SOURCES:
|
||||
audio_session:
|
||||
:path: ".symlinks/plugins/audio_session/ios"
|
||||
audioplayers_darwin:
|
||||
:path: ".symlinks/plugins/audioplayers_darwin/ios"
|
||||
Flutter:
|
||||
:path: Flutter
|
||||
flutter_secure_storage:
|
||||
:path: ".symlinks/plugins/flutter_secure_storage/ios"
|
||||
just_audio:
|
||||
:path: ".symlinks/plugins/just_audio/darwin"
|
||||
path_provider_foundation:
|
||||
:path: ".symlinks/plugins/path_provider_foundation/darwin"
|
||||
permission_handler_apple:
|
||||
:path: ".symlinks/plugins/permission_handler_apple/ios"
|
||||
shared_preferences_foundation:
|
||||
:path: ".symlinks/plugins/shared_preferences_foundation/darwin"
|
||||
speech_to_text:
|
||||
:path: ".symlinks/plugins/speech_to_text/ios"
|
||||
sqflite_darwin:
|
||||
:path: ".symlinks/plugins/sqflite_darwin/darwin"
|
||||
|
||||
SPEC CHECKSUMS:
|
||||
audio_session: 19e9480dbdd4e5f6c4543826b2e8b0e4ab6145fe
|
||||
audioplayers_darwin: 877d9a4d06331c5c374595e46e16453ac7eafa40
|
||||
Flutter: cabc95a1d2626b1b06e7179b784ebcf0c0cde467
|
||||
flutter_secure_storage: d33dac7ae2ea08509be337e775f6b59f1ff45f12
|
||||
just_audio: a42c63806f16995daf5b219ae1d679deb76e6a79
|
||||
path_provider_foundation: 2b6b4c569c0fb62ec74538f866245ac84301af46
|
||||
permission_handler_apple: 9878588469a2b0d0fc1e048d9f43605f92e6cec2
|
||||
shared_preferences_foundation: fcdcbc04712aee1108ac7fda236f363274528f78
|
||||
speech_to_text: b43a7d99aef037bd758ed8e45d79bbac035d2dfe
|
||||
sqflite_darwin: 5a7236e3b501866c1c9befc6771dfd73ffb8702d
|
||||
Try: 5ef669ae832617b3cee58cb2c6f99fb767a4ff96
|
||||
|
||||
PODFILE CHECKSUM: 3c63482e143d1b91d2d2560aee9fb04ecc74ac7e
|
||||
|
||||
COCOAPODS: 1.16.2
|
||||
|
|
@ -0,0 +1,749 @@
|
|||
// !$*UTF8*$!
|
||||
{
|
||||
archiveVersion = 1;
|
||||
classes = {
|
||||
};
|
||||
objectVersion = 54;
|
||||
objects = {
|
||||
|
||||
/* Begin PBXBuildFile section */
|
||||
1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */ = {isa = PBXBuildFile; fileRef = 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */; };
|
||||
24A2F4649D20B4FBB14C191F /* Pods_Runner.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 91353D9CBF7B2EE62DCC837D /* Pods_Runner.framework */; };
|
||||
331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C807B294A618700263BE5 /* RunnerTests.swift */; };
|
||||
3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; };
|
||||
74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74858FAE1ED2DC5600515810 /* AppDelegate.swift */; };
|
||||
97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; };
|
||||
97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; };
|
||||
97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; };
|
||||
E7528E253547AB91B2B4F858 /* Pods_RunnerTests.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 0CFA8EDD757215D657BFFE46 /* Pods_RunnerTests.framework */; };
|
||||
/* End PBXBuildFile section */
|
||||
|
||||
/* Begin PBXContainerItemProxy section */
|
||||
331C8085294A63A400263BE5 /* PBXContainerItemProxy */ = {
|
||||
isa = PBXContainerItemProxy;
|
||||
containerPortal = 97C146E61CF9000F007C117D /* Project object */;
|
||||
proxyType = 1;
|
||||
remoteGlobalIDString = 97C146ED1CF9000F007C117D;
|
||||
remoteInfo = Runner;
|
||||
};
|
||||
/* End PBXContainerItemProxy section */
|
||||
|
||||
/* Begin PBXCopyFilesBuildPhase section */
|
||||
9705A1C41CF9048500538489 /* Embed Frameworks */ = {
|
||||
isa = PBXCopyFilesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
dstPath = "";
|
||||
dstSubfolderSpec = 10;
|
||||
files = (
|
||||
);
|
||||
name = "Embed Frameworks";
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXCopyFilesBuildPhase section */
|
||||
|
||||
/* Begin PBXFileReference section */
|
||||
07BFFB716E3F729DE36E9E62 /* Pods-Runner.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.debug.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"; sourceTree = "<group>"; };
|
||||
0CFA8EDD757215D657BFFE46 /* Pods_RunnerTests.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_RunnerTests.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = "<group>"; };
|
||||
1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = "<group>"; };
|
||||
2DF005C09CC99A688F7EDA9D /* Pods-RunnerTests.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.profile.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.profile.xcconfig"; sourceTree = "<group>"; };
|
||||
331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = "<group>"; };
|
||||
331C8081294A63A400263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = "<group>"; };
|
||||
4A0BF9949893D60C8D62EDB2 /* Pods-RunnerTests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.debug.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.debug.xcconfig"; sourceTree = "<group>"; };
|
||||
68C9F9FDFC9B235BF26627E6 /* Pods-Runner.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.release.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"; sourceTree = "<group>"; };
|
||||
74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = "<group>"; };
|
||||
74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; };
|
||||
7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = "<group>"; };
|
||||
82777D655BE749ED24F31C7F /* Pods-Runner.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.profile.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.profile.xcconfig"; sourceTree = "<group>"; };
|
||||
91353D9CBF7B2EE62DCC837D /* Pods_Runner.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_Runner.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = "<group>"; };
|
||||
9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = "<group>"; };
|
||||
97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
97C146FB1CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = "<group>"; };
|
||||
97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; };
|
||||
97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = "<group>"; };
|
||||
97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
|
||||
E057BCA2D564C786C955CE59 /* Pods-RunnerTests.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.release.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.release.xcconfig"; sourceTree = "<group>"; };
|
||||
/* End PBXFileReference section */
|
||||
|
||||
/* Begin PBXFrameworksBuildPhase section */
|
||||
2E17305EED7643B25507D797 /* Frameworks */ = {
|
||||
isa = PBXFrameworksBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
E7528E253547AB91B2B4F858 /* Pods_RunnerTests.framework in Frameworks */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
97C146EB1CF9000F007C117D /* Frameworks */ = {
|
||||
isa = PBXFrameworksBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
24A2F4649D20B4FBB14C191F /* Pods_Runner.framework in Frameworks */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXFrameworksBuildPhase section */
|
||||
|
||||
/* Begin PBXGroup section */
|
||||
07B3C1031D9A8587953FAF94 /* Frameworks */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
91353D9CBF7B2EE62DCC837D /* Pods_Runner.framework */,
|
||||
0CFA8EDD757215D657BFFE46 /* Pods_RunnerTests.framework */,
|
||||
);
|
||||
name = Frameworks;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
331C8082294A63A400263BE5 /* RunnerTests */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
331C807B294A618700263BE5 /* RunnerTests.swift */,
|
||||
);
|
||||
path = RunnerTests;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
9740EEB11CF90186004384FC /* Flutter */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */,
|
||||
9740EEB21CF90195004384FC /* Debug.xcconfig */,
|
||||
7AFA3C8E1D35360C0083082E /* Release.xcconfig */,
|
||||
9740EEB31CF90195004384FC /* Generated.xcconfig */,
|
||||
);
|
||||
name = Flutter;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
97C146E51CF9000F007C117D = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
9740EEB11CF90186004384FC /* Flutter */,
|
||||
97C146F01CF9000F007C117D /* Runner */,
|
||||
97C146EF1CF9000F007C117D /* Products */,
|
||||
331C8082294A63A400263BE5 /* RunnerTests */,
|
||||
C533348C0BB393CA8B27DBD4 /* Pods */,
|
||||
07B3C1031D9A8587953FAF94 /* Frameworks */,
|
||||
);
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
97C146EF1CF9000F007C117D /* Products */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
97C146EE1CF9000F007C117D /* Runner.app */,
|
||||
331C8081294A63A400263BE5 /* RunnerTests.xctest */,
|
||||
);
|
||||
name = Products;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
97C146F01CF9000F007C117D /* Runner */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
97C146FA1CF9000F007C117D /* Main.storyboard */,
|
||||
97C146FD1CF9000F007C117D /* Assets.xcassets */,
|
||||
97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */,
|
||||
97C147021CF9000F007C117D /* Info.plist */,
|
||||
1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */,
|
||||
1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */,
|
||||
74858FAE1ED2DC5600515810 /* AppDelegate.swift */,
|
||||
74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */,
|
||||
);
|
||||
path = Runner;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
C533348C0BB393CA8B27DBD4 /* Pods */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
07BFFB716E3F729DE36E9E62 /* Pods-Runner.debug.xcconfig */,
|
||||
68C9F9FDFC9B235BF26627E6 /* Pods-Runner.release.xcconfig */,
|
||||
82777D655BE749ED24F31C7F /* Pods-Runner.profile.xcconfig */,
|
||||
4A0BF9949893D60C8D62EDB2 /* Pods-RunnerTests.debug.xcconfig */,
|
||||
E057BCA2D564C786C955CE59 /* Pods-RunnerTests.release.xcconfig */,
|
||||
2DF005C09CC99A688F7EDA9D /* Pods-RunnerTests.profile.xcconfig */,
|
||||
);
|
||||
name = Pods;
|
||||
path = Pods;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
/* End PBXGroup section */
|
||||
|
||||
/* Begin PBXNativeTarget section */
|
||||
331C8080294A63A400263BE5 /* RunnerTests */ = {
|
||||
isa = PBXNativeTarget;
|
||||
buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */;
|
||||
buildPhases = (
|
||||
FEFF7BB44040F35DC10DFC87 /* [CP] Check Pods Manifest.lock */,
|
||||
331C807D294A63A400263BE5 /* Sources */,
|
||||
331C807F294A63A400263BE5 /* Resources */,
|
||||
2E17305EED7643B25507D797 /* Frameworks */,
|
||||
);
|
||||
buildRules = (
|
||||
);
|
||||
dependencies = (
|
||||
331C8086294A63A400263BE5 /* PBXTargetDependency */,
|
||||
);
|
||||
name = RunnerTests;
|
||||
productName = RunnerTests;
|
||||
productReference = 331C8081294A63A400263BE5 /* RunnerTests.xctest */;
|
||||
productType = "com.apple.product-type.bundle.unit-test";
|
||||
};
|
||||
97C146ED1CF9000F007C117D /* Runner */ = {
|
||||
isa = PBXNativeTarget;
|
||||
buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */;
|
||||
buildPhases = (
|
||||
0585B9232D8EC85B106A0C5B /* [CP] Check Pods Manifest.lock */,
|
||||
9740EEB61CF901F6004384FC /* Run Script */,
|
||||
97C146EA1CF9000F007C117D /* Sources */,
|
||||
97C146EB1CF9000F007C117D /* Frameworks */,
|
||||
97C146EC1CF9000F007C117D /* Resources */,
|
||||
9705A1C41CF9048500538489 /* Embed Frameworks */,
|
||||
3B06AD1E1E4923F5004D2608 /* Thin Binary */,
|
||||
B7DA006F490B39DC5DD7D624 /* [CP] Embed Pods Frameworks */,
|
||||
7328AAE839104E6E980FA1B3 /* [CP] Copy Pods Resources */,
|
||||
);
|
||||
buildRules = (
|
||||
);
|
||||
dependencies = (
|
||||
);
|
||||
name = Runner;
|
||||
productName = Runner;
|
||||
productReference = 97C146EE1CF9000F007C117D /* Runner.app */;
|
||||
productType = "com.apple.product-type.application";
|
||||
};
|
||||
/* End PBXNativeTarget section */
|
||||
|
||||
/* Begin PBXProject section */
|
||||
97C146E61CF9000F007C117D /* Project object */ = {
|
||||
isa = PBXProject;
|
||||
attributes = {
|
||||
BuildIndependentTargetsInParallel = YES;
|
||||
LastUpgradeCheck = 1510;
|
||||
ORGANIZATIONNAME = "";
|
||||
TargetAttributes = {
|
||||
331C8080294A63A400263BE5 = {
|
||||
CreatedOnToolsVersion = 14.0;
|
||||
TestTargetID = 97C146ED1CF9000F007C117D;
|
||||
};
|
||||
97C146ED1CF9000F007C117D = {
|
||||
CreatedOnToolsVersion = 7.3.1;
|
||||
LastSwiftMigration = 1100;
|
||||
};
|
||||
};
|
||||
};
|
||||
buildConfigurationList = 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */;
|
||||
compatibilityVersion = "Xcode 9.3";
|
||||
developmentRegion = en;
|
||||
hasScannedForEncodings = 0;
|
||||
knownRegions = (
|
||||
en,
|
||||
Base,
|
||||
);
|
||||
mainGroup = 97C146E51CF9000F007C117D;
|
||||
productRefGroup = 97C146EF1CF9000F007C117D /* Products */;
|
||||
projectDirPath = "";
|
||||
projectRoot = "";
|
||||
targets = (
|
||||
97C146ED1CF9000F007C117D /* Runner */,
|
||||
331C8080294A63A400263BE5 /* RunnerTests */,
|
||||
);
|
||||
};
|
||||
/* End PBXProject section */
|
||||
|
||||
/* Begin PBXResourcesBuildPhase section */
|
||||
331C807F294A63A400263BE5 /* Resources */ = {
|
||||
isa = PBXResourcesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
97C146EC1CF9000F007C117D /* Resources */ = {
|
||||
isa = PBXResourcesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */,
|
||||
3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */,
|
||||
97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */,
|
||||
97C146FC1CF9000F007C117D /* Main.storyboard in Resources */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXResourcesBuildPhase section */
|
||||
|
||||
/* Begin PBXShellScriptBuildPhase section */
|
||||
0585B9232D8EC85B106A0C5B /* [CP] Check Pods Manifest.lock */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
);
|
||||
inputPaths = (
|
||||
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
|
||||
"${PODS_ROOT}/Manifest.lock",
|
||||
);
|
||||
name = "[CP] Check Pods Manifest.lock";
|
||||
outputFileListPaths = (
|
||||
);
|
||||
outputPaths = (
|
||||
"$(DERIVED_FILE_DIR)/Pods-Runner-checkManifestLockResult.txt",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
3B06AD1E1E4923F5004D2608 /* Thin Binary */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
alwaysOutOfDate = 1;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputPaths = (
|
||||
"${TARGET_BUILD_DIR}/${INFOPLIST_PATH}",
|
||||
);
|
||||
name = "Thin Binary";
|
||||
outputPaths = (
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" embed_and_thin";
|
||||
};
|
||||
7328AAE839104E6E980FA1B3 /* [CP] Copy Pods Resources */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-input-files.xcfilelist",
|
||||
);
|
||||
name = "[CP] Copy Pods Resources";
|
||||
outputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-output-files.xcfilelist",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources.sh\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
9740EEB61CF901F6004384FC /* Run Script */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
alwaysOutOfDate = 1;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputPaths = (
|
||||
);
|
||||
name = "Run Script";
|
||||
outputPaths = (
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build";
|
||||
};
|
||||
B7DA006F490B39DC5DD7D624 /* [CP] Embed Pods Frameworks */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-input-files.xcfilelist",
|
||||
);
|
||||
name = "[CP] Embed Pods Frameworks";
|
||||
outputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-output-files.xcfilelist",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
FEFF7BB44040F35DC10DFC87 /* [CP] Check Pods Manifest.lock */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
);
|
||||
inputPaths = (
|
||||
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
|
||||
"${PODS_ROOT}/Manifest.lock",
|
||||
);
|
||||
name = "[CP] Check Pods Manifest.lock";
|
||||
outputFileListPaths = (
|
||||
);
|
||||
outputPaths = (
|
||||
"$(DERIVED_FILE_DIR)/Pods-RunnerTests-checkManifestLockResult.txt",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
/* End PBXShellScriptBuildPhase section */
|
||||
|
||||
/* Begin PBXSourcesBuildPhase section */
|
||||
331C807D294A63A400263BE5 /* Sources */ = {
|
||||
isa = PBXSourcesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
97C146EA1CF9000F007C117D /* Sources */ = {
|
||||
isa = PBXSourcesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */,
|
||||
1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXSourcesBuildPhase section */
|
||||
|
||||
/* Begin PBXTargetDependency section */
|
||||
331C8086294A63A400263BE5 /* PBXTargetDependency */ = {
|
||||
isa = PBXTargetDependency;
|
||||
target = 97C146ED1CF9000F007C117D /* Runner */;
|
||||
targetProxy = 331C8085294A63A400263BE5 /* PBXContainerItemProxy */;
|
||||
};
|
||||
/* End PBXTargetDependency section */
|
||||
|
||||
/* Begin PBXVariantGroup section */
|
||||
97C146FA1CF9000F007C117D /* Main.storyboard */ = {
|
||||
isa = PBXVariantGroup;
|
||||
children = (
|
||||
97C146FB1CF9000F007C117D /* Base */,
|
||||
);
|
||||
name = Main.storyboard;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */ = {
|
||||
isa = PBXVariantGroup;
|
||||
children = (
|
||||
97C147001CF9000F007C117D /* Base */,
|
||||
);
|
||||
name = LaunchScreen.storyboard;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
/* End PBXVariantGroup section */
|
||||
|
||||
/* Begin XCBuildConfiguration section */
|
||||
249021D3217E4FDB00AE95B9 /* Profile */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
buildSettings = {
|
||||
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||
ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES;
|
||||
CLANG_ANALYZER_NONNULL = YES;
|
||||
CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
|
||||
CLANG_CXX_LIBRARY = "libc++";
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CLANG_ENABLE_OBJC_ARC = YES;
|
||||
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
|
||||
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||
CLANG_WARN_COMMA = YES;
|
||||
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
|
||||
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||
CLANG_WARN_EMPTY_BODY = YES;
|
||||
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||
CLANG_WARN_INFINITE_RECURSION = YES;
|
||||
CLANG_WARN_INT_CONVERSION = YES;
|
||||
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
|
||||
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
|
||||
CLANG_WARN_STRICT_PROTOTYPES = YES;
|
||||
CLANG_WARN_SUSPICIOUS_MOVE = YES;
|
||||
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||
"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
|
||||
COPY_PHASE_STRIP = NO;
|
||||
DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
|
||||
ENABLE_NS_ASSERTIONS = NO;
|
||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||
ENABLE_USER_SCRIPT_SANDBOXING = NO;
|
||||
GCC_C_LANGUAGE_STANDARD = gnu99;
|
||||
GCC_NO_COMMON_BLOCKS = YES;
|
||||
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.0;
|
||||
MTL_ENABLE_DEBUG_INFO = NO;
|
||||
SDKROOT = iphoneos;
|
||||
SUPPORTED_PLATFORMS = iphoneos;
|
||||
TARGETED_DEVICE_FAMILY = "1,2";
|
||||
VALIDATE_PRODUCT = YES;
|
||||
};
|
||||
name = Profile;
|
||||
};
|
||||
249021D4217E4FDB00AE95B9 /* Profile */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
|
||||
buildSettings = {
|
||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||
DEVELOPMENT_TEAM = ZN2U6988BZ;
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.dramaling;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||
SWIFT_VERSION = 5.0;
|
||||
VERSIONING_SYSTEM = "apple-generic";
|
||||
};
|
||||
name = Profile;
|
||||
};
|
||||
331C8088294A63A400263BE5 /* Debug */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 4A0BF9949893D60C8D62EDB2 /* Pods-RunnerTests.debug.xcconfig */;
|
||||
buildSettings = {
|
||||
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||
CODE_SIGN_STYLE = Automatic;
|
||||
CURRENT_PROJECT_VERSION = 1;
|
||||
GENERATE_INFOPLIST_FILE = YES;
|
||||
MARKETING_VERSION = 1.0;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.dramaling.RunnerTests;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG;
|
||||
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
|
||||
SWIFT_VERSION = 5.0;
|
||||
TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Runner.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/Runner";
|
||||
};
|
||||
name = Debug;
|
||||
};
|
||||
331C8089294A63A400263BE5 /* Release */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = E057BCA2D564C786C955CE59 /* Pods-RunnerTests.release.xcconfig */;
|
||||
buildSettings = {
|
||||
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||
CODE_SIGN_STYLE = Automatic;
|
||||
CURRENT_PROJECT_VERSION = 1;
|
||||
GENERATE_INFOPLIST_FILE = YES;
|
||||
MARKETING_VERSION = 1.0;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.dramaling.RunnerTests;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_VERSION = 5.0;
|
||||
TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Runner.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/Runner";
|
||||
};
|
||||
name = Release;
|
||||
};
|
||||
331C808A294A63A400263BE5 /* Profile */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 2DF005C09CC99A688F7EDA9D /* Pods-RunnerTests.profile.xcconfig */;
|
||||
buildSettings = {
|
||||
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||
CODE_SIGN_STYLE = Automatic;
|
||||
CURRENT_PROJECT_VERSION = 1;
|
||||
GENERATE_INFOPLIST_FILE = YES;
|
||||
MARKETING_VERSION = 1.0;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.dramaling.RunnerTests;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_VERSION = 5.0;
|
||||
TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Runner.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/Runner";
|
||||
};
|
||||
name = Profile;
|
||||
};
|
||||
97C147031CF9000F007C117D /* Debug */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
buildSettings = {
|
||||
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||
ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES;
|
||||
CLANG_ANALYZER_NONNULL = YES;
|
||||
CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
|
||||
CLANG_CXX_LIBRARY = "libc++";
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CLANG_ENABLE_OBJC_ARC = YES;
|
||||
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
|
||||
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||
CLANG_WARN_COMMA = YES;
|
||||
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
|
||||
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||
CLANG_WARN_EMPTY_BODY = YES;
|
||||
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||
CLANG_WARN_INFINITE_RECURSION = YES;
|
||||
CLANG_WARN_INT_CONVERSION = YES;
|
||||
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
|
||||
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
|
||||
CLANG_WARN_STRICT_PROTOTYPES = YES;
|
||||
CLANG_WARN_SUSPICIOUS_MOVE = YES;
|
||||
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||
"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
|
||||
COPY_PHASE_STRIP = NO;
|
||||
DEBUG_INFORMATION_FORMAT = dwarf;
|
||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||
ENABLE_TESTABILITY = YES;
|
||||
ENABLE_USER_SCRIPT_SANDBOXING = NO;
|
||||
GCC_C_LANGUAGE_STANDARD = gnu99;
|
||||
GCC_DYNAMIC_NO_PIC = NO;
|
||||
GCC_NO_COMMON_BLOCKS = YES;
|
||||
GCC_OPTIMIZATION_LEVEL = 0;
|
||||
GCC_PREPROCESSOR_DEFINITIONS = (
|
||||
"DEBUG=1",
|
||||
"$(inherited)",
|
||||
);
|
||||
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.0;
|
||||
MTL_ENABLE_DEBUG_INFO = YES;
|
||||
ONLY_ACTIVE_ARCH = YES;
|
||||
SDKROOT = iphoneos;
|
||||
TARGETED_DEVICE_FAMILY = "1,2";
|
||||
};
|
||||
name = Debug;
|
||||
};
|
||||
97C147041CF9000F007C117D /* Release */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
buildSettings = {
|
||||
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||
ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES;
|
||||
CLANG_ANALYZER_NONNULL = YES;
|
||||
CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
|
||||
CLANG_CXX_LIBRARY = "libc++";
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CLANG_ENABLE_OBJC_ARC = YES;
|
||||
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
|
||||
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||
CLANG_WARN_COMMA = YES;
|
||||
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
|
||||
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||
CLANG_WARN_EMPTY_BODY = YES;
|
||||
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||
CLANG_WARN_INFINITE_RECURSION = YES;
|
||||
CLANG_WARN_INT_CONVERSION = YES;
|
||||
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
|
||||
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
|
||||
CLANG_WARN_STRICT_PROTOTYPES = YES;
|
||||
CLANG_WARN_SUSPICIOUS_MOVE = YES;
|
||||
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||
"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
|
||||
COPY_PHASE_STRIP = NO;
|
||||
DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
|
||||
ENABLE_NS_ASSERTIONS = NO;
|
||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||
ENABLE_USER_SCRIPT_SANDBOXING = NO;
|
||||
GCC_C_LANGUAGE_STANDARD = gnu99;
|
||||
GCC_NO_COMMON_BLOCKS = YES;
|
||||
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 17.0;
|
||||
MTL_ENABLE_DEBUG_INFO = NO;
|
||||
SDKROOT = iphoneos;
|
||||
SUPPORTED_PLATFORMS = iphoneos;
|
||||
SWIFT_COMPILATION_MODE = wholemodule;
|
||||
SWIFT_OPTIMIZATION_LEVEL = "-O";
|
||||
TARGETED_DEVICE_FAMILY = "1,2";
|
||||
VALIDATE_PRODUCT = YES;
|
||||
};
|
||||
name = Release;
|
||||
};
|
||||
97C147061CF9000F007C117D /* Debug */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */;
|
||||
buildSettings = {
|
||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||
DEVELOPMENT_TEAM = ZN2U6988BZ;
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.dramaling;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
|
||||
SWIFT_VERSION = 5.0;
|
||||
VERSIONING_SYSTEM = "apple-generic";
|
||||
};
|
||||
name = Debug;
|
||||
};
|
||||
97C147071CF9000F007C117D /* Release */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
|
||||
buildSettings = {
|
||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||
DEVELOPMENT_TEAM = ZN2U6988BZ;
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.dramaling;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||
SWIFT_VERSION = 5.0;
|
||||
VERSIONING_SYSTEM = "apple-generic";
|
||||
};
|
||||
name = Release;
|
||||
};
|
||||
/* End XCBuildConfiguration section */
|
||||
|
||||
/* Begin XCConfigurationList section */
|
||||
331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */ = {
|
||||
isa = XCConfigurationList;
|
||||
buildConfigurations = (
|
||||
331C8088294A63A400263BE5 /* Debug */,
|
||||
331C8089294A63A400263BE5 /* Release */,
|
||||
331C808A294A63A400263BE5 /* Profile */,
|
||||
);
|
||||
defaultConfigurationIsVisible = 0;
|
||||
defaultConfigurationName = Release;
|
||||
};
|
||||
97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */ = {
|
||||
isa = XCConfigurationList;
|
||||
buildConfigurations = (
|
||||
97C147031CF9000F007C117D /* Debug */,
|
||||
97C147041CF9000F007C117D /* Release */,
|
||||
249021D3217E4FDB00AE95B9 /* Profile */,
|
||||
);
|
||||
defaultConfigurationIsVisible = 0;
|
||||
defaultConfigurationName = Release;
|
||||
};
|
||||
97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */ = {
|
||||
isa = XCConfigurationList;
|
||||
buildConfigurations = (
|
||||
97C147061CF9000F007C117D /* Debug */,
|
||||
97C147071CF9000F007C117D /* Release */,
|
||||
249021D4217E4FDB00AE95B9 /* Profile */,
|
||||
);
|
||||
defaultConfigurationIsVisible = 0;
|
||||
defaultConfigurationName = Release;
|
||||
};
|
||||
/* End XCConfigurationList section */
|
||||
};
|
||||
rootObject = 97C146E61CF9000F007C117D /* Project object */;
|
||||
}
|
||||
7
apps/mobile/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata
generated
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<Workspace
|
||||
version = "1.0">
|
||||
<FileRef
|
||||
location = "self:">
|
||||
</FileRef>
|
||||
</Workspace>
|
||||
|
|
@ -0,0 +1,8 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>IDEDidComputeMac32BitWarning</key>
|
||||
<true/>
|
||||
</dict>
|
||||
</plist>
|
||||
|
|
@ -0,0 +1,8 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>PreviewsEnabled</key>
|
||||
<false/>
|
||||
</dict>
|
||||
</plist>
|
||||
|
|
@ -0,0 +1,101 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<Scheme
|
||||
LastUpgradeVersion = "1510"
|
||||
version = "1.3">
|
||||
<BuildAction
|
||||
parallelizeBuildables = "YES"
|
||||
buildImplicitDependencies = "YES">
|
||||
<BuildActionEntries>
|
||||
<BuildActionEntry
|
||||
buildForTesting = "YES"
|
||||
buildForRunning = "YES"
|
||||
buildForProfiling = "YES"
|
||||
buildForArchiving = "YES"
|
||||
buildForAnalyzing = "YES">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildActionEntry>
|
||||
</BuildActionEntries>
|
||||
</BuildAction>
|
||||
<TestAction
|
||||
buildConfiguration = "Debug"
|
||||
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||
customLLDBInitFile = "$(SRCROOT)/Flutter/ephemeral/flutter_lldbinit"
|
||||
shouldUseLaunchSchemeArgsEnv = "YES">
|
||||
<MacroExpansion>
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</MacroExpansion>
|
||||
<Testables>
|
||||
<TestableReference
|
||||
skipped = "NO"
|
||||
parallelizable = "YES">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "331C8080294A63A400263BE5"
|
||||
BuildableName = "RunnerTests.xctest"
|
||||
BlueprintName = "RunnerTests"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</TestableReference>
|
||||
</Testables>
|
||||
</TestAction>
|
||||
<LaunchAction
|
||||
buildConfiguration = "Debug"
|
||||
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||
customLLDBInitFile = "$(SRCROOT)/Flutter/ephemeral/flutter_lldbinit"
|
||||
launchStyle = "0"
|
||||
useCustomWorkingDirectory = "NO"
|
||||
ignoresPersistentStateOnLaunch = "NO"
|
||||
debugDocumentVersioning = "YES"
|
||||
debugServiceExtension = "internal"
|
||||
enableGPUValidationMode = "1"
|
||||
allowLocationSimulation = "YES">
|
||||
<BuildableProductRunnable
|
||||
runnableDebuggingMode = "0">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildableProductRunnable>
|
||||
</LaunchAction>
|
||||
<ProfileAction
|
||||
buildConfiguration = "Profile"
|
||||
shouldUseLaunchSchemeArgsEnv = "YES"
|
||||
savedToolIdentifier = ""
|
||||
useCustomWorkingDirectory = "NO"
|
||||
debugDocumentVersioning = "YES">
|
||||
<BuildableProductRunnable
|
||||
runnableDebuggingMode = "0">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildableProductRunnable>
|
||||
</ProfileAction>
|
||||
<AnalyzeAction
|
||||
buildConfiguration = "Debug">
|
||||
</AnalyzeAction>
|
||||
<ArchiveAction
|
||||
buildConfiguration = "Release"
|
||||
revealArchiveInOrganizer = "YES">
|
||||
</ArchiveAction>
|
||||
</Scheme>
|
||||
|
|
@ -0,0 +1,10 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<Workspace
|
||||
version = "1.0">
|
||||
<FileRef
|
||||
location = "group:Runner.xcodeproj">
|
||||
</FileRef>
|
||||
<FileRef
|
||||
location = "group:Pods/Pods.xcodeproj">
|
||||
</FileRef>
|
||||
</Workspace>
|
||||
|
|
@ -0,0 +1,8 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>IDEDidComputeMac32BitWarning</key>
|
||||
<true/>
|
||||
</dict>
|
||||
</plist>
|
||||
|
|
@ -0,0 +1,8 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>PreviewsEnabled</key>
|
||||
<false/>
|
||||
</dict>
|
||||
</plist>
|
||||
|
|
@ -0,0 +1,13 @@
|
|||
import Flutter
|
||||
import UIKit
|
||||
|
||||
@main
|
||||
@objc class AppDelegate: FlutterAppDelegate {
|
||||
override func application(
|
||||
_ application: UIApplication,
|
||||
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
|
||||
) -> Bool {
|
||||
GeneratedPluginRegistrant.register(with: self)
|
||||
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,122 @@
|
|||
{
|
||||
"images" : [
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-20x20@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-20x20@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-29x29@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-29x29@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-29x29@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-40x40@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-40x40@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "60x60",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-60x60@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "60x60",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-60x60@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-20x20@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-20x20@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-29x29@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-29x29@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-40x40@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-40x40@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "76x76",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-76x76@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "76x76",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-76x76@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "83.5x83.5",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-83.5x83.5@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "1024x1024",
|
||||
"idiom" : "ios-marketing",
|
||||
"filename" : "Icon-App-1024x1024@1x.png",
|
||||
"scale" : "1x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"version" : 1,
|
||||
"author" : "xcode"
|
||||
}
|
||||
}
|
||||
|
After Width: | Height: | Size: 11 KiB |
|
After Width: | Height: | Size: 295 B |
|
After Width: | Height: | Size: 406 B |
|
After Width: | Height: | Size: 450 B |
|
After Width: | Height: | Size: 282 B |
|
After Width: | Height: | Size: 462 B |
|
After Width: | Height: | Size: 704 B |
|
After Width: | Height: | Size: 406 B |
|
After Width: | Height: | Size: 586 B |
|
After Width: | Height: | Size: 862 B |
|
After Width: | Height: | Size: 862 B |
|
After Width: | Height: | Size: 1.6 KiB |
|
After Width: | Height: | Size: 762 B |
|
After Width: | Height: | Size: 1.2 KiB |
|
After Width: | Height: | Size: 1.4 KiB |
|
|
@ -0,0 +1,23 @@
|
|||
{
|
||||
"images" : [
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"filename" : "LaunchImage.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"filename" : "LaunchImage@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"idiom" : "universal",
|
||||
"filename" : "LaunchImage@3x.png",
|
||||
"scale" : "3x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"version" : 1,
|
||||
"author" : "xcode"
|
||||
}
|
||||
}
|
||||
BIN
apps/mobile/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png
vendored
Normal file
|
After Width: | Height: | Size: 68 B |
BIN
apps/mobile/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png
vendored
Normal file
|
After Width: | Height: | Size: 68 B |
BIN
apps/mobile/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png
vendored
Normal file
|
After Width: | Height: | Size: 68 B |
|
|
@ -0,0 +1,5 @@
|
|||
# Launch Screen Assets
|
||||
|
||||
You can customize the launch screen with your own desired assets by replacing the image files in this directory.
|
||||
|
||||
You can also do it by opening your Flutter project's Xcode project with `open ios/Runner.xcworkspace`, selecting `Runner/Assets.xcassets` in the Project Navigator and dropping in the desired images.
|
||||
|
|
@ -0,0 +1,37 @@
|
|||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="12121" systemVersion="16G29" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" colorMatched="YES" initialViewController="01J-lp-oVM">
|
||||
<dependencies>
|
||||
<deployment identifier="iOS"/>
|
||||
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="12089"/>
|
||||
</dependencies>
|
||||
<scenes>
|
||||
<!--View Controller-->
|
||||
<scene sceneID="EHf-IW-A2E">
|
||||
<objects>
|
||||
<viewController id="01J-lp-oVM" sceneMemberID="viewController">
|
||||
<layoutGuides>
|
||||
<viewControllerLayoutGuide type="top" id="Ydg-fD-yQy"/>
|
||||
<viewControllerLayoutGuide type="bottom" id="xbc-2k-c8Z"/>
|
||||
</layoutGuides>
|
||||
<view key="view" contentMode="scaleToFill" id="Ze5-6b-2t3">
|
||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||
<subviews>
|
||||
<imageView opaque="NO" clipsSubviews="YES" multipleTouchEnabled="YES" contentMode="center" image="LaunchImage" translatesAutoresizingMaskIntoConstraints="NO" id="YRO-k0-Ey4">
|
||||
</imageView>
|
||||
</subviews>
|
||||
<color key="backgroundColor" red="1" green="1" blue="1" alpha="1" colorSpace="custom" customColorSpace="sRGB"/>
|
||||
<constraints>
|
||||
<constraint firstItem="YRO-k0-Ey4" firstAttribute="centerX" secondItem="Ze5-6b-2t3" secondAttribute="centerX" id="1a2-6s-vTC"/>
|
||||
<constraint firstItem="YRO-k0-Ey4" firstAttribute="centerY" secondItem="Ze5-6b-2t3" secondAttribute="centerY" id="4X2-HB-R7a"/>
|
||||
</constraints>
|
||||
</view>
|
||||
</viewController>
|
||||
<placeholder placeholderIdentifier="IBFirstResponder" id="iYj-Kq-Ea1" userLabel="First Responder" sceneMemberID="firstResponder"/>
|
||||
</objects>
|
||||
<point key="canvasLocation" x="53" y="375"/>
|
||||
</scene>
|
||||
</scenes>
|
||||
<resources>
|
||||
<image name="LaunchImage" width="168" height="185"/>
|
||||
</resources>
|
||||
</document>
|
||||
|
|
@ -0,0 +1,26 @@
|
|||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="10117" systemVersion="15F34" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" initialViewController="BYZ-38-t0r">
|
||||
<dependencies>
|
||||
<deployment identifier="iOS"/>
|
||||
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="10085"/>
|
||||
</dependencies>
|
||||
<scenes>
|
||||
<!--Flutter View Controller-->
|
||||
<scene sceneID="tne-QT-ifu">
|
||||
<objects>
|
||||
<viewController id="BYZ-38-t0r" customClass="FlutterViewController" sceneMemberID="viewController">
|
||||
<layoutGuides>
|
||||
<viewControllerLayoutGuide type="top" id="y3c-jy-aDJ"/>
|
||||
<viewControllerLayoutGuide type="bottom" id="wfy-db-euE"/>
|
||||
</layoutGuides>
|
||||
<view key="view" contentMode="scaleToFill" id="8bC-Xf-vdC">
|
||||
<rect key="frame" x="0.0" y="0.0" width="600" height="600"/>
|
||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||
<color key="backgroundColor" white="1" alpha="1" colorSpace="custom" customColorSpace="calibratedWhite"/>
|
||||
</view>
|
||||
</viewController>
|
||||
<placeholder placeholderIdentifier="IBFirstResponder" id="dkx-z0-nzr" sceneMemberID="firstResponder"/>
|
||||
</objects>
|
||||
</scene>
|
||||
</scenes>
|
||||
</document>
|
||||
|
|
@ -0,0 +1,49 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||
<key>CFBundleDisplayName</key>
|
||||
<string>Dramaling</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>$(EXECUTABLE_NAME)</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>dramaling</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>APPL</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>$(FLUTTER_BUILD_NAME)</string>
|
||||
<key>CFBundleSignature</key>
|
||||
<string>????</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>$(FLUTTER_BUILD_NUMBER)</string>
|
||||
<key>LSRequiresIPhoneOS</key>
|
||||
<true/>
|
||||
<key>UILaunchStoryboardName</key>
|
||||
<string>LaunchScreen</string>
|
||||
<key>UIMainStoryboardFile</key>
|
||||
<string>Main</string>
|
||||
<key>UISupportedInterfaceOrientations</key>
|
||||
<array>
|
||||
<string>UIInterfaceOrientationPortrait</string>
|
||||
<string>UIInterfaceOrientationLandscapeLeft</string>
|
||||
<string>UIInterfaceOrientationLandscapeRight</string>
|
||||
</array>
|
||||
<key>UISupportedInterfaceOrientations~ipad</key>
|
||||
<array>
|
||||
<string>UIInterfaceOrientationPortrait</string>
|
||||
<string>UIInterfaceOrientationPortraitUpsideDown</string>
|
||||
<string>UIInterfaceOrientationLandscapeLeft</string>
|
||||
<string>UIInterfaceOrientationLandscapeRight</string>
|
||||
</array>
|
||||
<key>CADisableMinimumFrameDurationOnPhone</key>
|
||||
<true/>
|
||||
<key>UIApplicationSupportsIndirectInputEvents</key>
|
||||
<true/>
|
||||
</dict>
|
||||
</plist>
|
||||
|
|
@ -0,0 +1 @@
|
|||
#import "GeneratedPluginRegistrant.h"
|
||||
|
|
@ -0,0 +1,12 @@
|
|||
import Flutter
|
||||
import UIKit
|
||||
import XCTest
|
||||
|
||||
class RunnerTests: XCTestCase {
|
||||
|
||||
func testExample() {
|
||||
// If you add code to the Runner application, consider adding tests here.
|
||||
// See https://developer.apple.com/documentation/xctest for more information about using XCTest.
|
||||
}
|
||||
|
||||
}
|
||||
|
|
@ -0,0 +1,83 @@
|
|||
class AppConstants {
|
||||
// App Info
|
||||
static const String appName = 'Drama Ling';
|
||||
static const String appVersion = '1.0.0';
|
||||
|
||||
// API Configuration
|
||||
static const String baseUrl = 'https://api.dramaling.com';
|
||||
static const String apiVersion = 'v1';
|
||||
static const String apiBaseUrl = '$baseUrl/api/$apiVersion';
|
||||
|
||||
// Local API for Development
|
||||
static const String localBaseUrl = 'http://localhost:5000';
|
||||
static const String localApiBaseUrl = '$localBaseUrl/api/$apiVersion';
|
||||
|
||||
// Storage Keys
|
||||
static const String accessTokenKey = 'access_token';
|
||||
static const String refreshTokenKey = 'refresh_token';
|
||||
static const String userDataKey = 'user_data';
|
||||
static const String languageKey = 'language';
|
||||
static const String themeKey = 'theme';
|
||||
|
||||
// Learning Constants
|
||||
static const int maxLifePoints = 5;
|
||||
static const int lifePointRecoveryHours = 5;
|
||||
static const int dailyExperienceBonus = 50;
|
||||
|
||||
// Dialogue Constants
|
||||
static const int maxDialogueTurns = 12;
|
||||
static const int dialogueTimeoutSeconds = 600; // 10 minutes
|
||||
static const double passingScore = 60.0;
|
||||
static const double excellentScore = 90.0;
|
||||
|
||||
// Time Challenge Constants
|
||||
static const int timeChallengeSeconds = 300; // 5 minutes
|
||||
static const int timeWarningSeconds = 60;
|
||||
static const int timeCriticalSeconds = 30;
|
||||
|
||||
// Animation Durations
|
||||
static const Duration shortAnimation = Duration(milliseconds: 200);
|
||||
static const Duration normalAnimation = Duration(milliseconds: 300);
|
||||
static const Duration longAnimation = Duration(milliseconds: 500);
|
||||
|
||||
// Network
|
||||
static const Duration connectionTimeout = Duration(seconds: 30);
|
||||
static const Duration receiveTimeout = Duration(seconds: 30);
|
||||
|
||||
// Pagination
|
||||
static const int defaultPageSize = 20;
|
||||
static const int maxPageSize = 100;
|
||||
}
|
||||
|
||||
class AppStrings {
|
||||
// Common
|
||||
static const String ok = '確定';
|
||||
static const String cancel = '取消';
|
||||
static const String retry = '重試';
|
||||
static const String loading = '載入中...';
|
||||
static const String error = '錯誤';
|
||||
static const String success = '成功';
|
||||
|
||||
// Authentication
|
||||
static const String login = '登入';
|
||||
static const String register = '註冊';
|
||||
static const String logout = '登出';
|
||||
static const String email = '電子郵件';
|
||||
static const String password = '密碼';
|
||||
static const String confirmPassword = '確認密碼';
|
||||
static const String forgotPassword = '忘記密碼?';
|
||||
|
||||
// Learning
|
||||
static const String startLearning = '開始學習';
|
||||
static const String continueDialogue = '繼續對話';
|
||||
static const String vocabularyPractice = '詞彙練習';
|
||||
static const String dialoguePractice = '對話練習';
|
||||
static const String timeChallenge = '限時挑戰';
|
||||
|
||||
// Errors
|
||||
static const String networkError = '網路連線錯誤';
|
||||
static const String serverError = '伺服器錯誤';
|
||||
static const String unknownError = '未知錯誤';
|
||||
static const String invalidCredentials = '帳號或密碼錯誤';
|
||||
static const String sessionExpired = '登入已過期,請重新登入';
|
||||
}
|
||||
|
|
@ -0,0 +1,125 @@
|
|||
import 'package:hive_flutter/hive_flutter.dart';
|
||||
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
||||
|
||||
class StorageService {
|
||||
static late Box _box;
|
||||
static const FlutterSecureStorage _secureStorage = FlutterSecureStorage(
|
||||
aOptions: AndroidOptions(
|
||||
encryptedSharedPreferences: true,
|
||||
),
|
||||
iOptions: IOSOptions(
|
||||
accessibility: KeychainAccessibility.first_unlock_this_device,
|
||||
),
|
||||
);
|
||||
|
||||
/// 初始化儲存服務
|
||||
static Future<void> init() async {
|
||||
await Hive.initFlutter();
|
||||
_box = await Hive.openBox('dramaling_storage');
|
||||
}
|
||||
|
||||
// 一般資料存取 (使用Hive)
|
||||
|
||||
/// 儲存資料
|
||||
static Future<void> setData<T>(String key, T value) async {
|
||||
await _box.put(key, value);
|
||||
}
|
||||
|
||||
/// 獲取資料
|
||||
static T? getData<T>(String key) {
|
||||
return _box.get(key) as T?;
|
||||
}
|
||||
|
||||
/// 移除資料
|
||||
static Future<void> removeData(String key) async {
|
||||
await _box.delete(key);
|
||||
}
|
||||
|
||||
/// 清除所有資料
|
||||
static Future<void> clearAll() async {
|
||||
await _box.clear();
|
||||
}
|
||||
|
||||
/// 檢查資料是否存在
|
||||
static bool hasData(String key) {
|
||||
return _box.containsKey(key);
|
||||
}
|
||||
|
||||
// 安全資料存取 (使用FlutterSecureStorage)
|
||||
|
||||
/// 安全儲存資料 (用於敏感資料如token)
|
||||
static Future<void> setSecureData(String key, String value) async {
|
||||
await _secureStorage.write(key: key, value: value);
|
||||
}
|
||||
|
||||
/// 獲取安全資料
|
||||
static Future<String?> getSecureData(String key) async {
|
||||
return await _secureStorage.read(key: key);
|
||||
}
|
||||
|
||||
/// 移除安全資料
|
||||
static Future<void> removeSecureData(String key) async {
|
||||
await _secureStorage.delete(key: key);
|
||||
}
|
||||
|
||||
/// 清除所有安全資料
|
||||
static Future<void> clearSecureData() async {
|
||||
await _secureStorage.deleteAll();
|
||||
}
|
||||
|
||||
/// 檢查安全資料是否存在
|
||||
static Future<bool> hasSecureData(String key) async {
|
||||
final data = await _secureStorage.read(key: key);
|
||||
return data != null;
|
||||
}
|
||||
|
||||
// 便捷方法
|
||||
|
||||
/// 儲存用戶Token
|
||||
static Future<void> saveTokens({
|
||||
required String accessToken,
|
||||
required String refreshToken,
|
||||
}) async {
|
||||
await setSecureData('access_token', accessToken);
|
||||
await setSecureData('refresh_token', refreshToken);
|
||||
}
|
||||
|
||||
/// 獲取存取Token
|
||||
static Future<String?> getAccessToken() async {
|
||||
return await getSecureData('access_token');
|
||||
}
|
||||
|
||||
/// 獲取刷新Token
|
||||
static Future<String?> getRefreshToken() async {
|
||||
return await getSecureData('refresh_token');
|
||||
}
|
||||
|
||||
/// 清除所有Token
|
||||
static Future<void> clearTokens() async {
|
||||
await removeSecureData('access_token');
|
||||
await removeSecureData('refresh_token');
|
||||
}
|
||||
|
||||
/// 儲存用戶偏好設定
|
||||
static Future<void> saveUserPreferences({
|
||||
String? language,
|
||||
String? theme,
|
||||
bool? soundEnabled,
|
||||
bool? vibrationEnabled,
|
||||
}) async {
|
||||
if (language != null) await setData('language', language);
|
||||
if (theme != null) await setData('theme', theme);
|
||||
if (soundEnabled != null) await setData('sound_enabled', soundEnabled);
|
||||
if (vibrationEnabled != null) await setData('vibration_enabled', vibrationEnabled);
|
||||
}
|
||||
|
||||
/// 獲取用戶偏好設定
|
||||
static Map<String, dynamic> getUserPreferences() {
|
||||
return {
|
||||
'language': getData<String>('language') ?? 'zh',
|
||||
'theme': getData<String>('theme') ?? 'system',
|
||||
'sound_enabled': getData<bool>('sound_enabled') ?? true,
|
||||
'vibration_enabled': getData<bool>('vibration_enabled') ?? true,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,355 @@
|
|||
import 'dart:async';
|
||||
|
||||
import 'package:flutter/foundation.dart';
|
||||
import 'package:permission_handler/permission_handler.dart';
|
||||
import 'package:speech_to_text/speech_recognition_error.dart';
|
||||
import 'package:speech_to_text/speech_recognition_result.dart';
|
||||
import 'package:speech_to_text/speech_to_text.dart';
|
||||
|
||||
/// AI語音識別服務
|
||||
///
|
||||
/// 提供完整的語音識別功能,支援:
|
||||
/// - 實時語音轉文字
|
||||
/// - 多語言支援(中文、英文)
|
||||
/// - 音頻權限管理
|
||||
/// - 錯誤處理與重試機制
|
||||
/// - 音量監測
|
||||
class VoiceRecognitionService {
|
||||
static final VoiceRecognitionService _instance = VoiceRecognitionService._internal();
|
||||
factory VoiceRecognitionService() => _instance;
|
||||
VoiceRecognitionService._internal();
|
||||
|
||||
final SpeechToText _speechToText = SpeechToText();
|
||||
|
||||
// 語音識別狀態
|
||||
bool _isInitialized = false;
|
||||
bool _isListening = false;
|
||||
bool _isAvailable = false;
|
||||
|
||||
// 識別結果回調
|
||||
final StreamController<VoiceRecognitionResult> _resultController =
|
||||
StreamController<VoiceRecognitionResult>.broadcast();
|
||||
|
||||
// 音量回調
|
||||
final StreamController<double> _soundLevelController =
|
||||
StreamController<double>.broadcast();
|
||||
|
||||
// 狀態回調
|
||||
final StreamController<VoiceRecognitionState> _stateController =
|
||||
StreamController<VoiceRecognitionState>.broadcast();
|
||||
|
||||
// 支援的語言
|
||||
static const Map<String, String> supportedLanguages = {
|
||||
'zh-TW': '繁體中文',
|
||||
'zh-CN': '簡體中文',
|
||||
'en-US': 'English (US)',
|
||||
'en-GB': 'English (UK)',
|
||||
};
|
||||
|
||||
// Getters
|
||||
bool get isInitialized => _isInitialized;
|
||||
bool get isListening => _isListening;
|
||||
bool get isAvailable => _isAvailable;
|
||||
|
||||
// Streams
|
||||
Stream<VoiceRecognitionResult> get resultStream => _resultController.stream;
|
||||
Stream<double> get soundLevelStream => _soundLevelController.stream;
|
||||
Stream<VoiceRecognitionState> get stateStream => _stateController.stream;
|
||||
|
||||
/// 初始化語音識別服務
|
||||
Future<bool> initialize() async {
|
||||
try {
|
||||
// 檢查並請求麥克風權限
|
||||
final permissionStatus = await _requestMicrophonePermission();
|
||||
if (!permissionStatus) {
|
||||
debugPrint('VoiceRecognitionService: 麥克風權限被拒絕');
|
||||
return false;
|
||||
}
|
||||
|
||||
// 初始化語音識別引擎
|
||||
_isAvailable = await _speechToText.initialize(
|
||||
onError: _onError,
|
||||
onStatus: _onStatus,
|
||||
);
|
||||
|
||||
if (_isAvailable) {
|
||||
_isInitialized = true;
|
||||
_stateController.add(VoiceRecognitionState.initialized);
|
||||
debugPrint('VoiceRecognitionService: 初始化成功');
|
||||
return true;
|
||||
} else {
|
||||
debugPrint('VoiceRecognitionService: 語音識別不可用');
|
||||
return false;
|
||||
}
|
||||
} catch (e) {
|
||||
debugPrint('VoiceRecognitionService: 初始化失敗 - $e');
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/// 開始語音識別
|
||||
Future<bool> startListening({
|
||||
String languageId = 'zh-TW',
|
||||
Duration timeout = const Duration(seconds: 30),
|
||||
bool partialResults = true,
|
||||
}) async {
|
||||
if (!_isInitialized || !_isAvailable) {
|
||||
debugPrint('VoiceRecognitionService: 服務未初始化或不可用');
|
||||
return false;
|
||||
}
|
||||
|
||||
if (_isListening) {
|
||||
debugPrint('VoiceRecognitionService: 已在監聽中');
|
||||
return true;
|
||||
}
|
||||
|
||||
try {
|
||||
await _speechToText.listen(
|
||||
onResult: _onResult,
|
||||
listenFor: timeout,
|
||||
pauseFor: const Duration(seconds: 3),
|
||||
partialResults: partialResults,
|
||||
localeId: languageId,
|
||||
onSoundLevelChange: _onSoundLevelChange,
|
||||
listenMode: ListenMode.confirmation,
|
||||
);
|
||||
|
||||
_isListening = true;
|
||||
_stateController.add(VoiceRecognitionState.listening);
|
||||
debugPrint('VoiceRecognitionService: 開始監聽');
|
||||
return true;
|
||||
} catch (e) {
|
||||
debugPrint('VoiceRecognitionService: 開始監聽失敗 - $e');
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/// 停止語音識別
|
||||
Future<void> stopListening() async {
|
||||
if (!_isListening) return;
|
||||
|
||||
try {
|
||||
await _speechToText.stop();
|
||||
_isListening = false;
|
||||
_stateController.add(VoiceRecognitionState.stopped);
|
||||
debugPrint('VoiceRecognitionService: 停止監聽');
|
||||
} catch (e) {
|
||||
debugPrint('VoiceRecognitionService: 停止監聽失敗 - $e');
|
||||
}
|
||||
}
|
||||
|
||||
/// 取消語音識別
|
||||
Future<void> cancel() async {
|
||||
if (!_isListening) return;
|
||||
|
||||
try {
|
||||
await _speechToText.cancel();
|
||||
_isListening = false;
|
||||
_stateController.add(VoiceRecognitionState.cancelled);
|
||||
debugPrint('VoiceRecognitionService: 取消監聽');
|
||||
} catch (e) {
|
||||
debugPrint('VoiceRecognitionService: 取消監聽失敗 - $e');
|
||||
}
|
||||
}
|
||||
|
||||
/// 獲取支援的語言
|
||||
Future<List<LocaleName>> getAvailableLanguages() async {
|
||||
if (!_isInitialized) return [];
|
||||
return await _speechToText.locales();
|
||||
}
|
||||
|
||||
/// 檢查特定語言是否支援
|
||||
Future<bool> isLanguageSupported(String languageId) async {
|
||||
final locales = await getAvailableLanguages();
|
||||
return locales.any((locale) => locale.localeId == languageId);
|
||||
}
|
||||
|
||||
/// 請求麥克風權限
|
||||
Future<bool> _requestMicrophonePermission() async {
|
||||
final status = await Permission.microphone.status;
|
||||
|
||||
if (status.isGranted) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (status.isDenied) {
|
||||
final result = await Permission.microphone.request();
|
||||
return result.isGranted;
|
||||
}
|
||||
|
||||
if (status.isPermanentlyDenied) {
|
||||
await openAppSettings();
|
||||
return false;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/// 處理識別結果
|
||||
void _onResult(SpeechRecognitionResult result) {
|
||||
final voiceResult = VoiceRecognitionResult(
|
||||
recognizedWords: result.recognizedWords,
|
||||
confidence: result.confidence,
|
||||
isFinal: result.finalResult,
|
||||
alternatives: result.alternates.map((alt) =>
|
||||
VoiceAlternative(
|
||||
text: alt.recognizedWords,
|
||||
confidence: alt.confidence,
|
||||
)
|
||||
).toList(),
|
||||
);
|
||||
|
||||
_resultController.add(voiceResult);
|
||||
|
||||
if (result.finalResult) {
|
||||
debugPrint('VoiceRecognitionService: 最終結果 - ${result.recognizedWords}');
|
||||
} else {
|
||||
debugPrint('VoiceRecognitionService: 部分結果 - ${result.recognizedWords}');
|
||||
}
|
||||
}
|
||||
|
||||
/// 處理錯誤
|
||||
void _onError(SpeechRecognitionError error) {
|
||||
debugPrint('VoiceRecognitionService: 錯誤 - ${error.errorMsg}');
|
||||
|
||||
final errorType = _mapErrorType(error.errorMsg);
|
||||
_stateController.add(VoiceRecognitionState.error(errorType, error.errorMsg));
|
||||
|
||||
_isListening = false;
|
||||
}
|
||||
|
||||
/// 處理狀態變化
|
||||
void _onStatus(String status) {
|
||||
debugPrint('VoiceRecognitionService: 狀態變化 - $status');
|
||||
|
||||
switch (status) {
|
||||
case 'listening':
|
||||
_isListening = true;
|
||||
_stateController.add(VoiceRecognitionState.listening);
|
||||
break;
|
||||
case 'notListening':
|
||||
_isListening = false;
|
||||
_stateController.add(VoiceRecognitionState.stopped);
|
||||
break;
|
||||
case 'done':
|
||||
_isListening = false;
|
||||
_stateController.add(VoiceRecognitionState.completed);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/// 處理音量變化
|
||||
void _onSoundLevelChange(double level) {
|
||||
_soundLevelController.add(level);
|
||||
}
|
||||
|
||||
/// 映射錯誤類型
|
||||
VoiceRecognitionErrorType _mapErrorType(String errorMsg) {
|
||||
if (errorMsg.contains('network')) {
|
||||
return VoiceRecognitionErrorType.network;
|
||||
} else if (errorMsg.contains('audio')) {
|
||||
return VoiceRecognitionErrorType.audio;
|
||||
} else if (errorMsg.contains('permission')) {
|
||||
return VoiceRecognitionErrorType.permission;
|
||||
} else if (errorMsg.contains('timeout')) {
|
||||
return VoiceRecognitionErrorType.timeout;
|
||||
} else {
|
||||
return VoiceRecognitionErrorType.unknown;
|
||||
}
|
||||
}
|
||||
|
||||
/// 清理資源
|
||||
void dispose() {
|
||||
_resultController.close();
|
||||
_soundLevelController.close();
|
||||
_stateController.close();
|
||||
}
|
||||
}
|
||||
|
||||
/// 語音識別結果
|
||||
class VoiceRecognitionResult {
|
||||
final String recognizedWords;
|
||||
final double confidence;
|
||||
final bool isFinal;
|
||||
final List<VoiceAlternative> alternatives;
|
||||
|
||||
VoiceRecognitionResult({
|
||||
required this.recognizedWords,
|
||||
required this.confidence,
|
||||
required this.isFinal,
|
||||
this.alternatives = const [],
|
||||
});
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return 'VoiceRecognitionResult(words: $recognizedWords, confidence: $confidence, isFinal: $isFinal)';
|
||||
}
|
||||
}
|
||||
|
||||
/// 語音識別替代結果
|
||||
class VoiceAlternative {
|
||||
final String text;
|
||||
final double confidence;
|
||||
|
||||
VoiceAlternative({
|
||||
required this.text,
|
||||
required this.confidence,
|
||||
});
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return 'VoiceAlternative(text: $text, confidence: $confidence)';
|
||||
}
|
||||
}
|
||||
|
||||
/// 語音識別狀態
|
||||
class VoiceRecognitionState {
|
||||
final VoiceRecognitionStatus status;
|
||||
final VoiceRecognitionErrorType? errorType;
|
||||
final String? errorMessage;
|
||||
|
||||
VoiceRecognitionState._(this.status, [this.errorType, this.errorMessage]);
|
||||
|
||||
static VoiceRecognitionState get uninitialized =>
|
||||
VoiceRecognitionState._(VoiceRecognitionStatus.uninitialized);
|
||||
|
||||
static VoiceRecognitionState get initialized =>
|
||||
VoiceRecognitionState._(VoiceRecognitionStatus.initialized);
|
||||
|
||||
static VoiceRecognitionState get listening =>
|
||||
VoiceRecognitionState._(VoiceRecognitionStatus.listening);
|
||||
|
||||
static VoiceRecognitionState get stopped =>
|
||||
VoiceRecognitionState._(VoiceRecognitionStatus.stopped);
|
||||
|
||||
static VoiceRecognitionState get completed =>
|
||||
VoiceRecognitionState._(VoiceRecognitionStatus.completed);
|
||||
|
||||
static VoiceRecognitionState get cancelled =>
|
||||
VoiceRecognitionState._(VoiceRecognitionStatus.cancelled);
|
||||
|
||||
static VoiceRecognitionState error(VoiceRecognitionErrorType errorType, String message) =>
|
||||
VoiceRecognitionState._(VoiceRecognitionStatus.error, errorType, message);
|
||||
|
||||
bool get hasError => status == VoiceRecognitionStatus.error;
|
||||
}
|
||||
|
||||
/// 語音識別狀態枚舉
|
||||
enum VoiceRecognitionStatus {
|
||||
uninitialized,
|
||||
initialized,
|
||||
listening,
|
||||
stopped,
|
||||
completed,
|
||||
cancelled,
|
||||
error,
|
||||
}
|
||||
|
||||
/// 語音識別錯誤類型
|
||||
enum VoiceRecognitionErrorType {
|
||||
network,
|
||||
audio,
|
||||
permission,
|
||||
timeout,
|
||||
unknown,
|
||||
}
|
||||
|
|
@ -0,0 +1,116 @@
|
|||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||
|
||||
import '../../features/auth/screens/login_screen.dart';
|
||||
import '../../features/auth/screens/register_screen.dart';
|
||||
import '../../features/learning/screens/home_screen.dart';
|
||||
import '../../features/dialogue/screens/dialogue_main_screen.dart';
|
||||
import '../../shared/providers/auth_provider.dart';
|
||||
|
||||
final routerProvider = Provider<GoRouter>((ref) {
|
||||
final authState = ref.watch(authProvider);
|
||||
|
||||
return GoRouter(
|
||||
initialLocation: authState.isAuthenticated ? '/home' : '/login',
|
||||
redirect: (context, state) {
|
||||
final isAuthenticated = authState.isAuthenticated;
|
||||
final isAuthRoute = state.uri.path.startsWith('/auth');
|
||||
|
||||
if (!isAuthenticated && !isAuthRoute) {
|
||||
return '/login';
|
||||
}
|
||||
|
||||
if (isAuthenticated && isAuthRoute) {
|
||||
return '/home';
|
||||
}
|
||||
|
||||
return null;
|
||||
},
|
||||
routes: [
|
||||
// Authentication Routes
|
||||
GoRoute(
|
||||
path: '/login',
|
||||
builder: (context, state) => const LoginScreen(),
|
||||
),
|
||||
GoRoute(
|
||||
path: '/register',
|
||||
builder: (context, state) => const RegisterScreen(),
|
||||
),
|
||||
|
||||
// Main App Routes
|
||||
GoRoute(
|
||||
path: '/home',
|
||||
builder: (context, state) => const HomeScreen(),
|
||||
),
|
||||
|
||||
// Learning Routes
|
||||
GoRoute(
|
||||
path: '/vocabulary',
|
||||
builder: (context, state) => const Scaffold(
|
||||
body: Center(child: Text('詞彙練習頁面')),
|
||||
),
|
||||
),
|
||||
GoRoute(
|
||||
path: '/dialogue',
|
||||
builder: (context, state) {
|
||||
final scenarioId = state.uri.queryParameters['scenarioId'] ?? 'restaurant_001';
|
||||
final levelId = state.uri.queryParameters['levelId'] ?? 'level_001';
|
||||
final isTimeChallenge = state.uri.queryParameters['timeChallenge'] == 'true';
|
||||
|
||||
return DialogueMainScreen(
|
||||
scenarioId: scenarioId,
|
||||
levelId: levelId,
|
||||
isTimeChallenge: isTimeChallenge,
|
||||
);
|
||||
},
|
||||
),
|
||||
GoRoute(
|
||||
path: '/challenge',
|
||||
builder: (context, state) => const Scaffold(
|
||||
body: Center(child: Text('限時挑戰頁面')),
|
||||
),
|
||||
),
|
||||
|
||||
// Profile Routes
|
||||
GoRoute(
|
||||
path: '/profile',
|
||||
builder: (context, state) => const Scaffold(
|
||||
body: Center(child: Text('個人檔案頁面')),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
||||
// Error handling
|
||||
errorBuilder: (context, state) => Scaffold(
|
||||
body: Center(
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
const Icon(
|
||||
Icons.error_outline,
|
||||
size: 64,
|
||||
color: Colors.red,
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
Text(
|
||||
'頁面未找到',
|
||||
style: Theme.of(context).textTheme.headlineSmall,
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
Text(
|
||||
state.error?.toString() ?? '未知錯誤',
|
||||
style: Theme.of(context).textTheme.bodyMedium,
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
const SizedBox(height: 24),
|
||||
ElevatedButton(
|
||||
onPressed: () => context.go('/home'),
|
||||
child: const Text('返回首頁'),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
});
|
||||
|
|
@ -0,0 +1,184 @@
|
|||
import 'package:flutter/material.dart';
|
||||
import 'package:google_fonts/google_fonts.dart';
|
||||
|
||||
class AppColors {
|
||||
// Primary Colors
|
||||
static const Color primary = Color(0xFF6366F1); // Indigo
|
||||
static const Color primaryLight = Color(0xFF8B5CF6); // Violet
|
||||
static const Color primaryDark = Color(0xFF4F46E5); // Dark Indigo
|
||||
|
||||
// Secondary Colors
|
||||
static const Color secondary = Color(0xFFF59E0B); // Amber
|
||||
static const Color secondaryLight = Color(0xFFFBBF24);
|
||||
static const Color secondaryDark = Color(0xFFD97706);
|
||||
|
||||
// Learning Status Colors
|
||||
static const Color success = Color(0xFF10B981); // Emerald
|
||||
static const Color warning = Color(0xFFF59E0B); // Amber
|
||||
static const Color error = Color(0xFFEF4444); // Red
|
||||
static const Color info = Color(0xFF3B82F6); // Blue
|
||||
|
||||
// Neutral Colors
|
||||
static const Color white = Color(0xFFFFFFFF);
|
||||
static const Color black = Color(0xFF000000);
|
||||
static const Color grey50 = Color(0xFFF9FAFB);
|
||||
static const Color grey100 = Color(0xFFF3F4F6);
|
||||
static const Color grey200 = Color(0xFFE5E7EB);
|
||||
static const Color grey300 = Color(0xFFD1D5DB);
|
||||
static const Color grey400 = Color(0xFF9CA3AF);
|
||||
static const Color grey500 = Color(0xFF6B7280);
|
||||
static const Color grey600 = Color(0xFF4B5563);
|
||||
static const Color grey700 = Color(0xFF374151);
|
||||
static const Color grey800 = Color(0xFF1F2937);
|
||||
static const Color grey900 = Color(0xFF111827);
|
||||
|
||||
// Surface Colors
|
||||
static const Color surface = Color(0xFFFFFFFF);
|
||||
static const Color surfaceDark = Color(0xFF1F2937);
|
||||
static const Color background = Color(0xFFF9FAFB);
|
||||
static const Color backgroundDark = Color(0xFF111827);
|
||||
}
|
||||
|
||||
class AppTheme {
|
||||
static ThemeData get lightTheme {
|
||||
return ThemeData(
|
||||
useMaterial3: true,
|
||||
brightness: Brightness.light,
|
||||
|
||||
// Color Scheme
|
||||
colorScheme: const ColorScheme.light(
|
||||
primary: AppColors.primary,
|
||||
secondary: AppColors.secondary,
|
||||
surface: AppColors.surface,
|
||||
background: AppColors.background,
|
||||
error: AppColors.error,
|
||||
),
|
||||
|
||||
// Typography
|
||||
textTheme: GoogleFonts.notoSansTextTheme().copyWith(
|
||||
headlineLarge: GoogleFonts.notoSans(
|
||||
fontSize: 32,
|
||||
fontWeight: FontWeight.bold,
|
||||
color: AppColors.grey900,
|
||||
),
|
||||
headlineMedium: GoogleFonts.notoSans(
|
||||
fontSize: 28,
|
||||
fontWeight: FontWeight.bold,
|
||||
color: AppColors.grey900,
|
||||
),
|
||||
titleLarge: GoogleFonts.notoSans(
|
||||
fontSize: 22,
|
||||
fontWeight: FontWeight.w600,
|
||||
color: AppColors.grey900,
|
||||
),
|
||||
titleMedium: GoogleFonts.notoSans(
|
||||
fontSize: 16,
|
||||
fontWeight: FontWeight.w600,
|
||||
color: AppColors.grey700,
|
||||
),
|
||||
bodyLarge: GoogleFonts.notoSans(
|
||||
fontSize: 16,
|
||||
fontWeight: FontWeight.normal,
|
||||
color: AppColors.grey800,
|
||||
),
|
||||
bodyMedium: GoogleFonts.notoSans(
|
||||
fontSize: 14,
|
||||
fontWeight: FontWeight.normal,
|
||||
color: AppColors.grey700,
|
||||
),
|
||||
),
|
||||
|
||||
// Component Themes
|
||||
appBarTheme: AppBarTheme(
|
||||
backgroundColor: AppColors.white,
|
||||
foregroundColor: AppColors.grey900,
|
||||
elevation: 0,
|
||||
centerTitle: true,
|
||||
titleTextStyle: GoogleFonts.notoSans(
|
||||
fontSize: 18,
|
||||
fontWeight: FontWeight.w600,
|
||||
color: AppColors.grey900,
|
||||
),
|
||||
),
|
||||
|
||||
elevatedButtonTheme: ElevatedButtonThemeData(
|
||||
style: ElevatedButton.styleFrom(
|
||||
backgroundColor: AppColors.primary,
|
||||
foregroundColor: AppColors.white,
|
||||
elevation: 2,
|
||||
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
),
|
||||
),
|
||||
),
|
||||
|
||||
inputDecorationTheme: InputDecorationTheme(
|
||||
border: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
borderSide: const BorderSide(color: AppColors.grey300),
|
||||
),
|
||||
focusedBorder: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
borderSide: const BorderSide(color: AppColors.primary, width: 2),
|
||||
),
|
||||
contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
|
||||
),
|
||||
|
||||
cardTheme: const CardThemeData(
|
||||
elevation: 2,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.all(Radius.circular(16)),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
static ThemeData get darkTheme {
|
||||
return ThemeData(
|
||||
useMaterial3: true,
|
||||
brightness: Brightness.dark,
|
||||
|
||||
// Color Scheme
|
||||
colorScheme: const ColorScheme.dark(
|
||||
primary: AppColors.primaryLight,
|
||||
secondary: AppColors.secondaryLight,
|
||||
surface: AppColors.surfaceDark,
|
||||
background: AppColors.backgroundDark,
|
||||
error: AppColors.error,
|
||||
),
|
||||
|
||||
// Typography
|
||||
textTheme: GoogleFonts.notoSansTextTheme(ThemeData.dark().textTheme).copyWith(
|
||||
headlineLarge: GoogleFonts.notoSans(
|
||||
fontSize: 32,
|
||||
fontWeight: FontWeight.bold,
|
||||
color: AppColors.white,
|
||||
),
|
||||
headlineMedium: GoogleFonts.notoSans(
|
||||
fontSize: 28,
|
||||
fontWeight: FontWeight.bold,
|
||||
color: AppColors.white,
|
||||
),
|
||||
titleLarge: GoogleFonts.notoSans(
|
||||
fontSize: 22,
|
||||
fontWeight: FontWeight.w600,
|
||||
color: AppColors.white,
|
||||
),
|
||||
bodyLarge: GoogleFonts.notoSans(
|
||||
fontSize: 16,
|
||||
fontWeight: FontWeight.normal,
|
||||
color: AppColors.grey200,
|
||||
),
|
||||
),
|
||||
|
||||
// Component Themes
|
||||
appBarTheme: AppBarTheme(
|
||||
backgroundColor: AppColors.surfaceDark,
|
||||
foregroundColor: AppColors.white,
|
||||
elevation: 0,
|
||||
centerTitle: true,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,226 @@
|
|||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
class LoginScreen extends StatefulWidget {
|
||||
const LoginScreen({super.key});
|
||||
|
||||
@override
|
||||
State<LoginScreen> createState() => _LoginScreenState();
|
||||
}
|
||||
|
||||
class _LoginScreenState extends State<LoginScreen> {
|
||||
final _formKey = GlobalKey<FormState>();
|
||||
final _emailController = TextEditingController();
|
||||
final _passwordController = TextEditingController();
|
||||
bool _isLoading = false;
|
||||
bool _obscurePassword = true;
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_emailController.dispose();
|
||||
_passwordController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
backgroundColor: Theme.of(context).colorScheme.background,
|
||||
body: SafeArea(
|
||||
child: SingleChildScrollView(
|
||||
padding: const EdgeInsets.all(24.0),
|
||||
child: Form(
|
||||
key: _formKey,
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
const SizedBox(height: 60),
|
||||
|
||||
// Logo and Title
|
||||
Column(
|
||||
children: [
|
||||
Container(
|
||||
width: 120,
|
||||
height: 120,
|
||||
decoration: BoxDecoration(
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
shape: BoxShape.circle,
|
||||
),
|
||||
child: const Icon(
|
||||
Icons.theater_comedy,
|
||||
color: Colors.white,
|
||||
size: 60,
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 24),
|
||||
Text(
|
||||
'Drama Ling',
|
||||
style: Theme.of(context).textTheme.headlineLarge?.copyWith(
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
fontWeight: FontWeight.bold,
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
Text(
|
||||
'歡迎回來!開始您的語言學習之旅',
|
||||
style: Theme.of(context).textTheme.bodyLarge?.copyWith(
|
||||
color: Theme.of(context).colorScheme.onBackground.withOpacity(0.7),
|
||||
),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
],
|
||||
),
|
||||
|
||||
const SizedBox(height: 48),
|
||||
|
||||
// Email Field
|
||||
TextFormField(
|
||||
controller: _emailController,
|
||||
keyboardType: TextInputType.emailAddress,
|
||||
decoration: const InputDecoration(
|
||||
labelText: '電子郵件',
|
||||
hintText: '請輸入您的電子郵件',
|
||||
prefixIcon: Icon(Icons.email_outlined),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return '請輸入電子郵件';
|
||||
}
|
||||
if (!value.contains('@')) {
|
||||
return '請輸入有效的電子郵件格式';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
|
||||
const SizedBox(height: 16),
|
||||
|
||||
// Password Field
|
||||
TextFormField(
|
||||
controller: _passwordController,
|
||||
obscureText: _obscurePassword,
|
||||
decoration: InputDecoration(
|
||||
labelText: '密碼',
|
||||
hintText: '請輸入您的密碼',
|
||||
prefixIcon: const Icon(Icons.lock_outlined),
|
||||
suffixIcon: IconButton(
|
||||
icon: Icon(
|
||||
_obscurePassword
|
||||
? Icons.visibility_outlined
|
||||
: Icons.visibility_off_outlined,
|
||||
),
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
_obscurePassword = !_obscurePassword;
|
||||
});
|
||||
},
|
||||
),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return '請輸入密碼';
|
||||
}
|
||||
if (value.length < 6) {
|
||||
return '密碼長度至少需要6個字符';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
|
||||
const SizedBox(height: 24),
|
||||
|
||||
// Login Button
|
||||
SizedBox(
|
||||
height: 48,
|
||||
child: ElevatedButton(
|
||||
onPressed: _isLoading ? null : _handleLogin,
|
||||
child: _isLoading
|
||||
? const SizedBox(
|
||||
width: 20,
|
||||
height: 20,
|
||||
child: CircularProgressIndicator(strokeWidth: 2),
|
||||
)
|
||||
: const Text('登入'),
|
||||
),
|
||||
),
|
||||
|
||||
const SizedBox(height: 16),
|
||||
|
||||
// Forgot Password
|
||||
TextButton(
|
||||
onPressed: () {
|
||||
// TODO: 實現忘記密碼功能
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
const SnackBar(content: Text('忘記密碼功能開發中')),
|
||||
);
|
||||
},
|
||||
child: const Text('忘記密碼?'),
|
||||
),
|
||||
|
||||
const SizedBox(height: 32),
|
||||
|
||||
// Divider
|
||||
Row(
|
||||
children: [
|
||||
const Expanded(child: Divider()),
|
||||
Padding(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16),
|
||||
child: Text(
|
||||
'或',
|
||||
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
||||
color: Theme.of(context).colorScheme.onBackground.withOpacity(0.5),
|
||||
),
|
||||
),
|
||||
),
|
||||
const Expanded(child: Divider()),
|
||||
],
|
||||
),
|
||||
|
||||
const SizedBox(height: 32),
|
||||
|
||||
// Register Button
|
||||
OutlinedButton(
|
||||
onPressed: () => context.go('/register'),
|
||||
child: const Text('建立新帳號'),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Future<void> _handleLogin() async {
|
||||
if (!_formKey.currentState!.validate()) return;
|
||||
|
||||
setState(() {
|
||||
_isLoading = true;
|
||||
});
|
||||
|
||||
try {
|
||||
// TODO: 實現實際的登入邏輯
|
||||
await Future.delayed(const Duration(seconds: 2)); // 模擬API調用
|
||||
|
||||
if (mounted) {
|
||||
// 登入成功,導航到首頁
|
||||
context.go('/home');
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
const SnackBar(content: Text('登入成功!')),
|
||||
);
|
||||
}
|
||||
} catch (e) {
|
||||
if (mounted) {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(content: Text('登入失敗:$e')),
|
||||
);
|
||||
}
|
||||
} finally {
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
_isLoading = false;
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,311 @@
|
|||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
class RegisterScreen extends StatefulWidget {
|
||||
const RegisterScreen({super.key});
|
||||
|
||||
@override
|
||||
State<RegisterScreen> createState() => _RegisterScreenState();
|
||||
}
|
||||
|
||||
class _RegisterScreenState extends State<RegisterScreen> {
|
||||
final _formKey = GlobalKey<FormState>();
|
||||
final _emailController = TextEditingController();
|
||||
final _passwordController = TextEditingController();
|
||||
final _confirmPasswordController = TextEditingController();
|
||||
final _displayNameController = TextEditingController();
|
||||
bool _isLoading = false;
|
||||
bool _obscurePassword = true;
|
||||
bool _obscureConfirmPassword = true;
|
||||
bool _acceptTerms = false;
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_emailController.dispose();
|
||||
_passwordController.dispose();
|
||||
_confirmPasswordController.dispose();
|
||||
_displayNameController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
backgroundColor: Theme.of(context).colorScheme.background,
|
||||
appBar: AppBar(
|
||||
backgroundColor: Colors.transparent,
|
||||
elevation: 0,
|
||||
leading: IconButton(
|
||||
icon: const Icon(Icons.arrow_back),
|
||||
onPressed: () => context.go('/login'),
|
||||
),
|
||||
),
|
||||
body: SafeArea(
|
||||
child: SingleChildScrollView(
|
||||
padding: const EdgeInsets.all(24.0),
|
||||
child: Form(
|
||||
key: _formKey,
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: [
|
||||
// Title
|
||||
Text(
|
||||
'建立新帳號',
|
||||
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
|
||||
fontWeight: FontWeight.bold,
|
||||
),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
Text(
|
||||
'開始您的語言學習之旅',
|
||||
style: Theme.of(context).textTheme.bodyLarge?.copyWith(
|
||||
color: Theme.of(context).colorScheme.onBackground.withOpacity(0.7),
|
||||
),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
|
||||
const SizedBox(height: 32),
|
||||
|
||||
// Display Name Field
|
||||
TextFormField(
|
||||
controller: _displayNameController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: '顯示名稱',
|
||||
hintText: '請輸入您的顯示名稱',
|
||||
prefixIcon: Icon(Icons.person_outlined),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return '請輸入顯示名稱';
|
||||
}
|
||||
if (value.length < 2) {
|
||||
return '顯示名稱至少需要2個字符';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
|
||||
const SizedBox(height: 16),
|
||||
|
||||
// Email Field
|
||||
TextFormField(
|
||||
controller: _emailController,
|
||||
keyboardType: TextInputType.emailAddress,
|
||||
decoration: const InputDecoration(
|
||||
labelText: '電子郵件',
|
||||
hintText: '請輸入您的電子郵件',
|
||||
prefixIcon: Icon(Icons.email_outlined),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return '請輸入電子郵件';
|
||||
}
|
||||
if (!RegExp(r'^[\w-\.]+@([\w-]+\.)+[\w-]{2,4}$').hasMatch(value)) {
|
||||
return '請輸入有效的電子郵件格式';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
|
||||
const SizedBox(height: 16),
|
||||
|
||||
// Password Field
|
||||
TextFormField(
|
||||
controller: _passwordController,
|
||||
obscureText: _obscurePassword,
|
||||
decoration: InputDecoration(
|
||||
labelText: '密碼',
|
||||
hintText: '請輸入密碼(至少8個字符)',
|
||||
prefixIcon: const Icon(Icons.lock_outlined),
|
||||
suffixIcon: IconButton(
|
||||
icon: Icon(
|
||||
_obscurePassword
|
||||
? Icons.visibility_outlined
|
||||
: Icons.visibility_off_outlined,
|
||||
),
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
_obscurePassword = !_obscurePassword;
|
||||
});
|
||||
},
|
||||
),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return '請輸入密碼';
|
||||
}
|
||||
if (value.length < 8) {
|
||||
return '密碼長度至少需要8個字符';
|
||||
}
|
||||
if (!RegExp(r'^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)').hasMatch(value)) {
|
||||
return '密碼需包含大寫字母、小寫字母和數字';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
|
||||
const SizedBox(height: 16),
|
||||
|
||||
// Confirm Password Field
|
||||
TextFormField(
|
||||
controller: _confirmPasswordController,
|
||||
obscureText: _obscureConfirmPassword,
|
||||
decoration: InputDecoration(
|
||||
labelText: '確認密碼',
|
||||
hintText: '請再次輸入密碼',
|
||||
prefixIcon: const Icon(Icons.lock_outlined),
|
||||
suffixIcon: IconButton(
|
||||
icon: Icon(
|
||||
_obscureConfirmPassword
|
||||
? Icons.visibility_outlined
|
||||
: Icons.visibility_off_outlined,
|
||||
),
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
_obscureConfirmPassword = !_obscureConfirmPassword;
|
||||
});
|
||||
},
|
||||
),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return '請確認密碼';
|
||||
}
|
||||
if (value != _passwordController.text) {
|
||||
return '密碼不一致';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
|
||||
const SizedBox(height: 24),
|
||||
|
||||
// Terms and Conditions
|
||||
Row(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Checkbox(
|
||||
value: _acceptTerms,
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_acceptTerms = value ?? false;
|
||||
});
|
||||
},
|
||||
),
|
||||
Expanded(
|
||||
child: GestureDetector(
|
||||
onTap: () {
|
||||
setState(() {
|
||||
_acceptTerms = !_acceptTerms;
|
||||
});
|
||||
},
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(top: 12),
|
||||
child: RichText(
|
||||
text: TextSpan(
|
||||
style: Theme.of(context).textTheme.bodyMedium,
|
||||
children: [
|
||||
const TextSpan(text: '我同意'),
|
||||
TextSpan(
|
||||
text: '服務條款',
|
||||
style: TextStyle(
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
decoration: TextDecoration.underline,
|
||||
),
|
||||
),
|
||||
const TextSpan(text: '和'),
|
||||
TextSpan(
|
||||
text: '隱私政策',
|
||||
style: TextStyle(
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
decoration: TextDecoration.underline,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
|
||||
const SizedBox(height: 24),
|
||||
|
||||
// Register Button
|
||||
SizedBox(
|
||||
height: 48,
|
||||
child: ElevatedButton(
|
||||
onPressed: (_isLoading || !_acceptTerms) ? null : _handleRegister,
|
||||
child: _isLoading
|
||||
? const SizedBox(
|
||||
width: 20,
|
||||
height: 20,
|
||||
child: CircularProgressIndicator(strokeWidth: 2),
|
||||
)
|
||||
: const Text('註冊'),
|
||||
),
|
||||
),
|
||||
|
||||
const SizedBox(height: 16),
|
||||
|
||||
// Back to Login
|
||||
Row(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
const Text('已經有帳號了?'),
|
||||
TextButton(
|
||||
onPressed: () => context.go('/login'),
|
||||
child: const Text('立即登入'),
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Future<void> _handleRegister() async {
|
||||
if (!_formKey.currentState!.validate()) return;
|
||||
|
||||
if (!_acceptTerms) {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
const SnackBar(content: Text('請同意服務條款和隱私政策')),
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
setState(() {
|
||||
_isLoading = true;
|
||||
});
|
||||
|
||||
try {
|
||||
// TODO: 實現實際的註冊邏輯
|
||||
await Future.delayed(const Duration(seconds: 2)); // 模擬API調用
|
||||
|
||||
if (mounted) {
|
||||
// 註冊成功,導航到首頁
|
||||
context.go('/home');
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
const SnackBar(content: Text('註冊成功!歡迎使用Drama Ling')),
|
||||
);
|
||||
}
|
||||
} catch (e) {
|
||||
if (mounted) {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(content: Text('註冊失敗:$e')),
|
||||
);
|
||||
}
|
||||
} finally {
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
_isLoading = false;
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,547 @@
|
|||
/// 對話場景模型
|
||||
class DialogueScene {
|
||||
final String id;
|
||||
final String name;
|
||||
final String description;
|
||||
final String backgroundImageUrl;
|
||||
final String characterId;
|
||||
final String difficultyLevel;
|
||||
final List<String> tags;
|
||||
final Map<String, dynamic> metadata;
|
||||
|
||||
DialogueScene({
|
||||
required this.id,
|
||||
required this.name,
|
||||
required this.description,
|
||||
required this.backgroundImageUrl,
|
||||
required this.characterId,
|
||||
required this.difficultyLevel,
|
||||
this.tags = const [],
|
||||
this.metadata = const {},
|
||||
});
|
||||
|
||||
factory DialogueScene.fromJson(Map<String, dynamic> json) {
|
||||
return DialogueScene(
|
||||
id: json['id'] as String,
|
||||
name: json['name'] as String,
|
||||
description: json['description'] as String,
|
||||
backgroundImageUrl: json['backgroundImageUrl'] as String,
|
||||
characterId: json['characterId'] as String,
|
||||
difficultyLevel: json['difficultyLevel'] as String,
|
||||
tags: List<String>.from(json['tags'] ?? []),
|
||||
metadata: json['metadata'] as Map<String, dynamic>? ?? {},
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'name': name,
|
||||
'description': description,
|
||||
'backgroundImageUrl': backgroundImageUrl,
|
||||
'characterId': characterId,
|
||||
'difficultyLevel': difficultyLevel,
|
||||
'tags': tags,
|
||||
'metadata': metadata,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
/// 對話角色模型
|
||||
class DialogueCharacter {
|
||||
final String id;
|
||||
final String name;
|
||||
final String description;
|
||||
final String avatarUrl;
|
||||
final String personality;
|
||||
final String role;
|
||||
final String background;
|
||||
final List<String> specialities;
|
||||
final Map<String, String> localizedNames;
|
||||
|
||||
DialogueCharacter({
|
||||
required this.id,
|
||||
required this.name,
|
||||
required this.description,
|
||||
required this.avatarUrl,
|
||||
required this.personality,
|
||||
required this.role,
|
||||
required this.background,
|
||||
this.specialities = const [],
|
||||
this.localizedNames = const {},
|
||||
});
|
||||
|
||||
factory DialogueCharacter.fromJson(Map<String, dynamic> json) {
|
||||
return DialogueCharacter(
|
||||
id: json['id'] as String,
|
||||
name: json['name'] as String,
|
||||
description: json['description'] as String,
|
||||
avatarUrl: json['avatarUrl'] as String,
|
||||
personality: json['personality'] as String,
|
||||
role: json['role'] as String,
|
||||
background: json['background'] as String,
|
||||
specialities: List<String>.from(json['specialities'] ?? []),
|
||||
localizedNames: Map<String, String>.from(json['localizedNames'] ?? {}),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'name': name,
|
||||
'description': description,
|
||||
'avatarUrl': avatarUrl,
|
||||
'personality': personality,
|
||||
'role': role,
|
||||
'background': background,
|
||||
'specialities': specialities,
|
||||
'localizedNames': localizedNames,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
/// 對話消息模型
|
||||
class DialogueMessage {
|
||||
final String id;
|
||||
final String content;
|
||||
final bool isUser;
|
||||
final DateTime timestamp;
|
||||
final DialogueMessageType type;
|
||||
final Map<String, dynamic>? metadata;
|
||||
final String? audioUrl;
|
||||
final double? confidence;
|
||||
|
||||
DialogueMessage({
|
||||
required this.id,
|
||||
required this.content,
|
||||
required this.isUser,
|
||||
required this.timestamp,
|
||||
this.type = DialogueMessageType.text,
|
||||
this.metadata,
|
||||
this.audioUrl,
|
||||
this.confidence,
|
||||
});
|
||||
|
||||
factory DialogueMessage.fromJson(Map<String, dynamic> json) {
|
||||
return DialogueMessage(
|
||||
id: json['id'] as String,
|
||||
content: json['content'] as String,
|
||||
isUser: json['isUser'] as bool,
|
||||
timestamp: DateTime.parse(json['timestamp'] as String),
|
||||
type: DialogueMessageType.values.firstWhere(
|
||||
(e) => e.toString() == 'DialogueMessageType.${json['type']}',
|
||||
orElse: () => DialogueMessageType.text,
|
||||
),
|
||||
metadata: json['metadata'] as Map<String, dynamic>?,
|
||||
audioUrl: json['audioUrl'] as String?,
|
||||
confidence: json['confidence'] as double?,
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'content': content,
|
||||
'isUser': isUser,
|
||||
'timestamp': timestamp.toIso8601String(),
|
||||
'type': type.toString().split('.').last,
|
||||
'metadata': metadata,
|
||||
'audioUrl': audioUrl,
|
||||
'confidence': confidence,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
/// 對話消息類型
|
||||
enum DialogueMessageType {
|
||||
text,
|
||||
audio,
|
||||
system,
|
||||
hint,
|
||||
}
|
||||
|
||||
/// 對話任務模型
|
||||
class DialogueTask {
|
||||
final String id;
|
||||
final String title;
|
||||
final String description;
|
||||
final DialogueTaskType type;
|
||||
final Map<String, dynamic> requirements;
|
||||
final double progress;
|
||||
final bool isCompleted;
|
||||
final int maxAttempts;
|
||||
final int currentAttempts;
|
||||
final String? completionMessage;
|
||||
|
||||
DialogueTask({
|
||||
required this.id,
|
||||
required this.title,
|
||||
required this.description,
|
||||
required this.type,
|
||||
required this.requirements,
|
||||
this.progress = 0.0,
|
||||
this.isCompleted = false,
|
||||
this.maxAttempts = 3,
|
||||
this.currentAttempts = 0,
|
||||
this.completionMessage,
|
||||
});
|
||||
|
||||
factory DialogueTask.fromJson(Map<String, dynamic> json) {
|
||||
return DialogueTask(
|
||||
id: json['id'] as String,
|
||||
title: json['title'] as String,
|
||||
description: json['description'] as String,
|
||||
type: DialogueTaskType.values.firstWhere(
|
||||
(e) => e.toString() == 'DialogueTaskType.${json['type']}',
|
||||
orElse: () => DialogueTaskType.conversation,
|
||||
),
|
||||
requirements: json['requirements'] as Map<String, dynamic>,
|
||||
progress: json['progress'] as double? ?? 0.0,
|
||||
isCompleted: json['isCompleted'] as bool? ?? false,
|
||||
maxAttempts: json['maxAttempts'] as int? ?? 3,
|
||||
currentAttempts: json['currentAttempts'] as int? ?? 0,
|
||||
completionMessage: json['completionMessage'] as String?,
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'title': title,
|
||||
'description': description,
|
||||
'type': type.toString().split('.').last,
|
||||
'requirements': requirements,
|
||||
'progress': progress,
|
||||
'isCompleted': isCompleted,
|
||||
'maxAttempts': maxAttempts,
|
||||
'currentAttempts': currentAttempts,
|
||||
'completionMessage': completionMessage,
|
||||
};
|
||||
}
|
||||
|
||||
DialogueTask copyWith({
|
||||
String? id,
|
||||
String? title,
|
||||
String? description,
|
||||
DialogueTaskType? type,
|
||||
Map<String, dynamic>? requirements,
|
||||
double? progress,
|
||||
bool? isCompleted,
|
||||
int? maxAttempts,
|
||||
int? currentAttempts,
|
||||
String? completionMessage,
|
||||
}) {
|
||||
return DialogueTask(
|
||||
id: id ?? this.id,
|
||||
title: title ?? this.title,
|
||||
description: description ?? this.description,
|
||||
type: type ?? this.type,
|
||||
requirements: requirements ?? this.requirements,
|
||||
progress: progress ?? this.progress,
|
||||
isCompleted: isCompleted ?? this.isCompleted,
|
||||
maxAttempts: maxAttempts ?? this.maxAttempts,
|
||||
currentAttempts: currentAttempts ?? this.currentAttempts,
|
||||
completionMessage: completionMessage ?? this.completionMessage,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/// 對話任務類型
|
||||
enum DialogueTaskType {
|
||||
conversation, // 完成對話
|
||||
vocabulary, // 使用指定詞彙
|
||||
grammar, // 語法練習
|
||||
pronunciation, // 發音練習
|
||||
comprehension, // 理解測試
|
||||
}
|
||||
|
||||
/// 對話分析結果模型
|
||||
class DialogueAnalysis {
|
||||
final String id;
|
||||
final String userReply;
|
||||
final DateTime timestamp;
|
||||
|
||||
// 三維度評分
|
||||
final double grammarScore;
|
||||
final double semanticsScore;
|
||||
final double fluencyScore;
|
||||
|
||||
// 詳細分析
|
||||
final List<GrammarIssue> grammarIssues;
|
||||
final List<String> usedVocabulary;
|
||||
final List<String> missedVocabulary;
|
||||
final List<String> suggestions;
|
||||
|
||||
// 任務相關
|
||||
final double? taskProgress;
|
||||
final bool isDialogueComplete;
|
||||
|
||||
// 其他
|
||||
final Map<String, dynamic> metadata;
|
||||
|
||||
DialogueAnalysis({
|
||||
required this.id,
|
||||
required this.userReply,
|
||||
required this.timestamp,
|
||||
required this.grammarScore,
|
||||
required this.semanticsScore,
|
||||
required this.fluencyScore,
|
||||
this.grammarIssues = const [],
|
||||
this.usedVocabulary = const [],
|
||||
this.missedVocabulary = const [],
|
||||
this.suggestions = const [],
|
||||
this.taskProgress,
|
||||
this.isDialogueComplete = false,
|
||||
this.metadata = const {},
|
||||
});
|
||||
|
||||
factory DialogueAnalysis.fromJson(Map<String, dynamic> json) {
|
||||
return DialogueAnalysis(
|
||||
id: json['id'] as String,
|
||||
userReply: json['userReply'] as String,
|
||||
timestamp: DateTime.parse(json['timestamp'] as String),
|
||||
grammarScore: json['grammarScore'] as double,
|
||||
semanticsScore: json['semanticsScore'] as double,
|
||||
fluencyScore: json['fluencyScore'] as double,
|
||||
grammarIssues: (json['grammarIssues'] as List?)
|
||||
?.map((e) => GrammarIssue.fromJson(e as Map<String, dynamic>))
|
||||
.toList() ?? [],
|
||||
usedVocabulary: List<String>.from(json['usedVocabulary'] ?? []),
|
||||
missedVocabulary: List<String>.from(json['missedVocabulary'] ?? []),
|
||||
suggestions: List<String>.from(json['suggestions'] ?? []),
|
||||
taskProgress: json['taskProgress'] as double?,
|
||||
isDialogueComplete: json['isDialogueComplete'] as bool? ?? false,
|
||||
metadata: json['metadata'] as Map<String, dynamic>? ?? {},
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'userReply': userReply,
|
||||
'timestamp': timestamp.toIso8601String(),
|
||||
'grammarScore': grammarScore,
|
||||
'semanticsScore': semanticsScore,
|
||||
'fluencyScore': fluencyScore,
|
||||
'grammarIssues': grammarIssues.map((e) => e.toJson()).toList(),
|
||||
'usedVocabulary': usedVocabulary,
|
||||
'missedVocabulary': missedVocabulary,
|
||||
'suggestions': suggestions,
|
||||
'taskProgress': taskProgress,
|
||||
'isDialogueComplete': isDialogueComplete,
|
||||
'metadata': metadata,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
/// 語法問題模型
|
||||
class GrammarIssue {
|
||||
final String type;
|
||||
final String description;
|
||||
final String originalText;
|
||||
final String suggestedText;
|
||||
final int position;
|
||||
final int length;
|
||||
final GrammarIssueSeverity severity;
|
||||
|
||||
GrammarIssue({
|
||||
required this.type,
|
||||
required this.description,
|
||||
required this.originalText,
|
||||
required this.suggestedText,
|
||||
required this.position,
|
||||
required this.length,
|
||||
required this.severity,
|
||||
});
|
||||
|
||||
factory GrammarIssue.fromJson(Map<String, dynamic> json) {
|
||||
return GrammarIssue(
|
||||
type: json['type'] as String,
|
||||
description: json['description'] as String,
|
||||
originalText: json['originalText'] as String,
|
||||
suggestedText: json['suggestedText'] as String,
|
||||
position: json['position'] as int,
|
||||
length: json['length'] as int,
|
||||
severity: GrammarIssueSeverity.values.firstWhere(
|
||||
(e) => e.toString() == 'GrammarIssueSeverity.${json['severity']}',
|
||||
orElse: () => GrammarIssueSeverity.minor,
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'type': type,
|
||||
'description': description,
|
||||
'originalText': originalText,
|
||||
'suggestedText': suggestedText,
|
||||
'position': position,
|
||||
'length': length,
|
||||
'severity': severity.toString().split('.').last,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
/// 語法問題嚴重程度
|
||||
enum GrammarIssueSeverity {
|
||||
minor,
|
||||
moderate,
|
||||
major,
|
||||
critical,
|
||||
}
|
||||
|
||||
/// 對話最終得分模型
|
||||
class DialogueScore {
|
||||
final double grammarScore;
|
||||
final double semanticsScore;
|
||||
final double fluencyScore;
|
||||
final double taskBonus;
|
||||
final double vocabularyBonus;
|
||||
final double timeBonus;
|
||||
final double totalScore;
|
||||
final int starRating;
|
||||
final DateTime timestamp;
|
||||
final Map<String, dynamic> breakdown;
|
||||
|
||||
DialogueScore({
|
||||
required this.grammarScore,
|
||||
required this.semanticsScore,
|
||||
required this.fluencyScore,
|
||||
required this.taskBonus,
|
||||
required this.vocabularyBonus,
|
||||
required this.timeBonus,
|
||||
required this.totalScore,
|
||||
required this.starRating,
|
||||
DateTime? timestamp,
|
||||
this.breakdown = const {},
|
||||
}) : timestamp = timestamp ?? DateTime.now();
|
||||
|
||||
factory DialogueScore.fromJson(Map<String, dynamic> json) {
|
||||
return DialogueScore(
|
||||
grammarScore: json['grammarScore'] as double,
|
||||
semanticsScore: json['semanticsScore'] as double,
|
||||
fluencyScore: json['fluencyScore'] as double,
|
||||
taskBonus: json['taskBonus'] as double,
|
||||
vocabularyBonus: json['vocabularyBonus'] as double,
|
||||
timeBonus: json['timeBonus'] as double,
|
||||
totalScore: json['totalScore'] as double,
|
||||
starRating: json['starRating'] as int,
|
||||
timestamp: DateTime.parse(json['timestamp'] as String),
|
||||
breakdown: json['breakdown'] as Map<String, dynamic>? ?? {},
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'grammarScore': grammarScore,
|
||||
'semanticsScore': semanticsScore,
|
||||
'fluencyScore': fluencyScore,
|
||||
'taskBonus': taskBonus,
|
||||
'vocabularyBonus': vocabularyBonus,
|
||||
'timeBonus': timeBonus,
|
||||
'totalScore': totalScore,
|
||||
'starRating': starRating,
|
||||
'timestamp': timestamp.toIso8601String(),
|
||||
'breakdown': breakdown,
|
||||
};
|
||||
}
|
||||
|
||||
String get grade {
|
||||
if (totalScore >= 90) return 'A+';
|
||||
if (totalScore >= 80) return 'A';
|
||||
if (totalScore >= 70) return 'B';
|
||||
if (totalScore >= 60) return 'C';
|
||||
if (totalScore >= 50) return 'D';
|
||||
return 'F';
|
||||
}
|
||||
|
||||
String get comment {
|
||||
switch (starRating) {
|
||||
case 3:
|
||||
return '優秀!你的表現非常出色!';
|
||||
case 2:
|
||||
return '很好!繼續努力就能更進一步!';
|
||||
case 1:
|
||||
return '不錯!還有改進的空間。';
|
||||
default:
|
||||
return '需要更多練習,加油!';
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// 詞彙項目模型
|
||||
class VocabularyItem {
|
||||
final String id;
|
||||
final String word;
|
||||
final String definition;
|
||||
final String pronunciation;
|
||||
final List<String> examples;
|
||||
final String category;
|
||||
final int difficulty;
|
||||
final bool isRequired;
|
||||
final bool isUsed;
|
||||
|
||||
VocabularyItem({
|
||||
required this.id,
|
||||
required this.word,
|
||||
required this.definition,
|
||||
required this.pronunciation,
|
||||
this.examples = const [],
|
||||
this.category = '',
|
||||
this.difficulty = 1,
|
||||
this.isRequired = false,
|
||||
this.isUsed = false,
|
||||
});
|
||||
|
||||
factory VocabularyItem.fromJson(Map<String, dynamic> json) {
|
||||
return VocabularyItem(
|
||||
id: json['id'] as String,
|
||||
word: json['word'] as String,
|
||||
definition: json['definition'] as String,
|
||||
pronunciation: json['pronunciation'] as String,
|
||||
examples: List<String>.from(json['examples'] ?? []),
|
||||
category: json['category'] as String? ?? '',
|
||||
difficulty: json['difficulty'] as int? ?? 1,
|
||||
isRequired: json['isRequired'] as bool? ?? false,
|
||||
isUsed: json['isUsed'] as bool? ?? false,
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'word': word,
|
||||
'definition': definition,
|
||||
'pronunciation': pronunciation,
|
||||
'examples': examples,
|
||||
'category': category,
|
||||
'difficulty': difficulty,
|
||||
'isRequired': isRequired,
|
||||
'isUsed': isUsed,
|
||||
};
|
||||
}
|
||||
|
||||
VocabularyItem copyWith({
|
||||
String? id,
|
||||
String? word,
|
||||
String? definition,
|
||||
String? pronunciation,
|
||||
List<String>? examples,
|
||||
String? category,
|
||||
int? difficulty,
|
||||
bool? isRequired,
|
||||
bool? isUsed,
|
||||
}) {
|
||||
return VocabularyItem(
|
||||
id: id ?? this.id,
|
||||
word: word ?? this.word,
|
||||
definition: definition ?? this.definition,
|
||||
pronunciation: pronunciation ?? this.pronunciation,
|
||||
examples: examples ?? this.examples,
|
||||
category: category ?? this.category,
|
||||
difficulty: difficulty ?? this.difficulty,
|
||||
isRequired: isRequired ?? this.isRequired,
|
||||
isUsed: isUsed ?? this.isUsed,
|
||||
);
|
||||
}
|
||||
}
|
||||