init
This commit is contained in:
605
app/admin/page.tsx
Normal file
605
app/admin/page.tsx
Normal file
File diff suppressed because one or more lines are too long
29
app/api/database/test/route.ts
Normal file
29
app/api/database/test/route.ts
Normal file
@@ -0,0 +1,29 @@
|
||||
import { NextResponse } from 'next/server'
|
||||
import { testConnection } from '@/lib/database'
|
||||
|
||||
// GET - 測試資料庫連接
|
||||
export async function GET() {
|
||||
try {
|
||||
const isConnected = await testConnection()
|
||||
|
||||
if (isConnected) {
|
||||
return NextResponse.json({
|
||||
status: 'success',
|
||||
message: '資料庫連接正常',
|
||||
timestamp: new Date().toISOString()
|
||||
})
|
||||
} else {
|
||||
return NextResponse.json({
|
||||
status: 'error',
|
||||
message: '資料庫連接失敗'
|
||||
}, { status: 500 })
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('資料庫測試失敗:', error)
|
||||
return NextResponse.json({
|
||||
status: 'error',
|
||||
message: '資料庫測試失敗',
|
||||
error: error instanceof Error ? error.message : '未知錯誤'
|
||||
}, { status: 500 })
|
||||
}
|
||||
}
|
19
app/api/kpi/[id]/progress/route.ts
Normal file
19
app/api/kpi/[id]/progress/route.ts
Normal file
@@ -0,0 +1,19 @@
|
||||
import { NextRequest, NextResponse } from 'next/server'
|
||||
import { kpiService } from '@/lib/database'
|
||||
|
||||
// POST - 更新 KPI 進度
|
||||
export async function POST(
|
||||
request: NextRequest,
|
||||
{ params }: { params: { id: string } }
|
||||
) {
|
||||
try {
|
||||
const body = await request.json()
|
||||
const { progress, comment, blockers, userId } = body
|
||||
|
||||
await kpiService.updateKPIProgress(params.id, progress, comment, blockers, userId)
|
||||
return NextResponse.json({ message: '進度更新成功' })
|
||||
} catch (error) {
|
||||
console.error('更新進度失敗:', error)
|
||||
return NextResponse.json({ error: '更新進度失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
31
app/api/kpi/[id]/route.ts
Normal file
31
app/api/kpi/[id]/route.ts
Normal file
@@ -0,0 +1,31 @@
|
||||
import { NextRequest, NextResponse } from 'next/server'
|
||||
import { kpiService } from '@/lib/database'
|
||||
|
||||
// PUT - 更新 KPI
|
||||
export async function PUT(
|
||||
request: NextRequest,
|
||||
{ params }: { params: { id: string } }
|
||||
) {
|
||||
try {
|
||||
const body = await request.json()
|
||||
const kpi = await kpiService.updateKPI(params.id, body)
|
||||
return NextResponse.json(kpi)
|
||||
} catch (error) {
|
||||
console.error('更新 KPI 失敗:', error)
|
||||
return NextResponse.json({ error: '更新 KPI 失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
||||
|
||||
// DELETE - 刪除 KPI
|
||||
export async function DELETE(
|
||||
request: NextRequest,
|
||||
{ params }: { params: { id: string } }
|
||||
) {
|
||||
try {
|
||||
await kpiService.deleteKPI(params.id)
|
||||
return NextResponse.json({ message: 'KPI 刪除成功' })
|
||||
} catch (error) {
|
||||
console.error('刪除 KPI 失敗:', error)
|
||||
return NextResponse.json({ error: '刪除 KPI 失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
32
app/api/kpi/route.ts
Normal file
32
app/api/kpi/route.ts
Normal file
@@ -0,0 +1,32 @@
|
||||
import { NextRequest, NextResponse } from 'next/server'
|
||||
import { kpiService } from '@/lib/database'
|
||||
|
||||
// GET - 獲取 KPI 列表
|
||||
export async function GET(request: NextRequest) {
|
||||
try {
|
||||
const { searchParams } = new URL(request.url)
|
||||
const userId = searchParams.get('userId')
|
||||
|
||||
if (!userId) {
|
||||
return NextResponse.json({ error: '缺少用戶 ID' }, { status: 400 })
|
||||
}
|
||||
|
||||
const kpis = await kpiService.getKPIsByUser(userId)
|
||||
return NextResponse.json(kpis)
|
||||
} catch (error) {
|
||||
console.error('獲取 KPI 失敗:', error)
|
||||
return NextResponse.json({ error: '獲取 KPI 失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
||||
|
||||
// POST - 創建新 KPI
|
||||
export async function POST(request: NextRequest) {
|
||||
try {
|
||||
const body = await request.json()
|
||||
const kpi = await kpiService.createKPI(body)
|
||||
return NextResponse.json(kpi, { status: 201 })
|
||||
} catch (error) {
|
||||
console.error('創建 KPI 失敗:', error)
|
||||
return NextResponse.json({ error: '創建 KPI 失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
34
app/api/reviews/route.ts
Normal file
34
app/api/reviews/route.ts
Normal file
@@ -0,0 +1,34 @@
|
||||
import { NextRequest, NextResponse } from 'next/server'
|
||||
import { reviewService } from '@/lib/database'
|
||||
|
||||
// GET - 獲取評審列表
|
||||
export async function GET(request: NextRequest) {
|
||||
try {
|
||||
const { searchParams } = new URL(request.url)
|
||||
const userId = searchParams.get('userId')
|
||||
const type = searchParams.get('type') // 'upcoming' | 'all'
|
||||
|
||||
if (type === 'upcoming' && userId) {
|
||||
const reviews = await reviewService.getUpcomingReviews(userId)
|
||||
return NextResponse.json(reviews)
|
||||
} else {
|
||||
const reviews = await reviewService.getAllReviews()
|
||||
return NextResponse.json(reviews)
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('獲取評審列表失敗:', error)
|
||||
return NextResponse.json({ error: '獲取評審列表失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
||||
|
||||
// POST - 創建新評審
|
||||
export async function POST(request: NextRequest) {
|
||||
try {
|
||||
const body = await request.json()
|
||||
const review = await reviewService.createReview(body)
|
||||
return NextResponse.json(review, { status: 201 })
|
||||
} catch (error) {
|
||||
console.error('創建評審失敗:', error)
|
||||
return NextResponse.json({ error: '創建評審失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
25
app/api/users/route.ts
Normal file
25
app/api/users/route.ts
Normal file
@@ -0,0 +1,25 @@
|
||||
import { NextRequest, NextResponse } from 'next/server'
|
||||
import { userService } from '@/lib/database'
|
||||
|
||||
// GET - 獲取所有用戶
|
||||
export async function GET() {
|
||||
try {
|
||||
const users = await userService.getAllUsers()
|
||||
return NextResponse.json(users)
|
||||
} catch (error) {
|
||||
console.error('獲取用戶列表失敗:', error)
|
||||
return NextResponse.json({ error: '獲取用戶列表失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
||||
|
||||
// POST - 創建新用戶
|
||||
export async function POST(request: NextRequest) {
|
||||
try {
|
||||
const body = await request.json()
|
||||
const user = await userService.createUser(body)
|
||||
return NextResponse.json(user, { status: 201 })
|
||||
} catch (error) {
|
||||
console.error('創建用戶失敗:', error)
|
||||
return NextResponse.json({ error: '創建用戶失敗' }, { status: 500 })
|
||||
}
|
||||
}
|
90
app/globals.css
Normal file
90
app/globals.css
Normal file
@@ -0,0 +1,90 @@
|
||||
@tailwind base;
|
||||
@tailwind components;
|
||||
@tailwind utilities;
|
||||
|
||||
@layer utilities {
|
||||
.text-balance {
|
||||
text-wrap: balance;
|
||||
}
|
||||
}
|
||||
|
||||
@layer base {
|
||||
:root {
|
||||
--background: 0 0% 100%;
|
||||
--foreground: 0 0% 3.9%;
|
||||
--card: 0 0% 100%;
|
||||
--card-foreground: 0 0% 3.9%;
|
||||
--popover: 0 0% 100%;
|
||||
--popover-foreground: 0 0% 3.9%;
|
||||
--primary: 0 0% 9%;
|
||||
--primary-foreground: 0 0% 98%;
|
||||
--secondary: 0 0% 96.1%;
|
||||
--secondary-foreground: 0 0% 9%;
|
||||
--muted: 0 0% 96.1%;
|
||||
--muted-foreground: 0 0% 45.1%;
|
||||
--accent: 0 0% 96.1%;
|
||||
--accent-foreground: 0 0% 9%;
|
||||
--destructive: 0 84.2% 60.2%;
|
||||
--destructive-foreground: 0 0% 98%;
|
||||
--border: 0 0% 89.8%;
|
||||
--input: 0 0% 89.8%;
|
||||
--ring: 0 0% 3.9%;
|
||||
--chart-1: 12 76% 61%;
|
||||
--chart-2: 173 58% 39%;
|
||||
--chart-3: 197 37% 24%;
|
||||
--chart-4: 43 74% 66%;
|
||||
--chart-5: 27 87% 67%;
|
||||
--radius: 0.5rem;
|
||||
--sidebar-background: 0 0% 98%;
|
||||
--sidebar-foreground: 240 5.3% 26.1%;
|
||||
--sidebar-primary: 240 5.9% 10%;
|
||||
--sidebar-primary-foreground: 0 0% 98%;
|
||||
--sidebar-accent: 240 4.8% 95.9%;
|
||||
--sidebar-accent-foreground: 240 5.9% 10%;
|
||||
--sidebar-border: 220 13% 91%;
|
||||
--sidebar-ring: 217.2 91.2% 59.8%;
|
||||
}
|
||||
.dark {
|
||||
--background: 0 0% 3.9%;
|
||||
--foreground: 0 0% 98%;
|
||||
--card: 0 0% 3.9%;
|
||||
--card-foreground: 0 0% 98%;
|
||||
--popover: 0 0% 3.9%;
|
||||
--popover-foreground: 0 0% 98%;
|
||||
--primary: 0 0% 98%;
|
||||
--primary-foreground: 0 0% 9%;
|
||||
--secondary: 0 0% 14.9%;
|
||||
--secondary-foreground: 0 0% 98%;
|
||||
--muted: 0 0% 14.9%;
|
||||
--muted-foreground: 0 0% 63.9%;
|
||||
--accent: 0 0% 14.9%;
|
||||
--accent-foreground: 0 0% 98%;
|
||||
--destructive: 0 62.8% 30.6%;
|
||||
--destructive-foreground: 0 0% 98%;
|
||||
--border: 0 0% 14.9%;
|
||||
--input: 0 0% 14.9%;
|
||||
--ring: 0 0% 83.1%;
|
||||
--chart-1: 220 70% 50%;
|
||||
--chart-2: 160 60% 45%;
|
||||
--chart-3: 30 80% 55%;
|
||||
--chart-4: 280 65% 60%;
|
||||
--chart-5: 340 75% 55%;
|
||||
--sidebar-background: 240 5.9% 10%;
|
||||
--sidebar-foreground: 240 4.8% 95.9%;
|
||||
--sidebar-primary: 224.3 76.3% 48%;
|
||||
--sidebar-primary-foreground: 0 0% 100%;
|
||||
--sidebar-accent: 240 3.7% 15.9%;
|
||||
--sidebar-accent-foreground: 240 4.8% 95.9%;
|
||||
--sidebar-border: 240 3.7% 15.9%;
|
||||
--sidebar-ring: 217.2 91.2% 59.8%;
|
||||
}
|
||||
}
|
||||
|
||||
@layer base {
|
||||
* {
|
||||
@apply border-border;
|
||||
}
|
||||
body {
|
||||
@apply bg-background text-foreground;
|
||||
}
|
||||
}
|
739
app/kpi/page.tsx
Normal file
739
app/kpi/page.tsx
Normal file
File diff suppressed because one or more lines are too long
52
app/layout.tsx
Normal file
52
app/layout.tsx
Normal file
File diff suppressed because one or more lines are too long
446
app/page.tsx
Normal file
446
app/page.tsx
Normal file
File diff suppressed because one or more lines are too long
385
app/progress/page.tsx
Normal file
385
app/progress/page.tsx
Normal file
File diff suppressed because one or more lines are too long
455
app/reports/page.tsx
Normal file
455
app/reports/page.tsx
Normal file
File diff suppressed because one or more lines are too long
621
app/reviews/page.tsx
Normal file
621
app/reviews/page.tsx
Normal file
File diff suppressed because one or more lines are too long
Reference in New Issue
Block a user