admin crud ops implemented partialy
This commit is contained in:
echo 2025-11-09 18:11:40 +01:00
parent c357de515f
commit 9d2bfda0ca
21 changed files with 6108 additions and 3172 deletions

Binary file not shown.

View File

@ -1,5 +1,6 @@
/// <reference types="next" /> /// <reference types="next" />
/// <reference types="next/image-types/global" /> /// <reference types="next/image-types/global" />
import "./.next/dev/types/routes.d.ts";
// NOTE: This file should not be edited // NOTE: This file should not be edited
// see https://nextjs.org/docs/app/building-your-application/configuring/typescript for more information. // see https://nextjs.org/docs/app/api-reference/config/typescript for more information.

File diff suppressed because it is too large Load Diff

View File

@ -12,38 +12,39 @@
}, },
"dependencies": { "dependencies": {
"@fitai/shared": "file:../../packages/shared", "@fitai/shared": "file:../../packages/shared",
"@hookform/resolvers": "^3.3.0", "@hookform/resolvers": "^5.2.2",
"@tanstack/react-query": "^5.0.0", "@tailwindcss/postcss": "^4.1.17",
"@types/bcryptjs": "^2.4.6", "@tanstack/react-query": "^5.90.7",
"@types/sqlite3": "^3.1.11", "@types/bcryptjs": "^3.0.0",
"ag-charts-community": "^9.0.0", "@types/sqlite3": "^5.1.0",
"ag-charts-react": "^9.0.0", "ag-charts-community": "^12.3.1",
"ag-grid-community": "^32.0.0", "ag-charts-react": "^12.3.1",
"ag-grid-react": "^32.0.0", "ag-grid-community": "^34.3.1",
"autoprefixer": "^10.4.0", "ag-grid-react": "^34.3.1",
"axios": "^1.6.0", "autoprefixer": "^10.4.21",
"bcryptjs": "^2.4.3", "axios": "^1.13.2",
"lucide-react": "^0.294.0", "bcryptjs": "^3.0.3",
"next": "^14.0.0", "lucide-react": "^0.553.0",
"postcss": "^8.4.0", "next": "^16.0.1",
"react": "^18.0.0", "postcss": "^8.5.6",
"react-dom": "^18.0.0", "react": "^19.2.0",
"react-hook-form": "^7.47.0", "react-dom": "^19.2.0",
"recharts": "^2.8.0", "react-hook-form": "^7.66.0",
"recharts": "^3.3.0",
"sqlite": "^5.1.1", "sqlite": "^5.1.1",
"sqlite3": "^5.1.7", "sqlite3": "^5.1.7",
"tailwindcss": "^3.3.0", "tailwindcss": "^4.1.17",
"zod": "^3.22.0" "zod": "^4.1.12"
}, },
"devDependencies": { "devDependencies": {
"@testing-library/jest-dom": "^6.1.0", "@testing-library/jest-dom": "^6.9.1",
"@testing-library/react": "^13.4.0", "@testing-library/react": "^16.3.0",
"@types/node": "^20.0.0", "@types/node": "^24.10.0",
"@types/react": "18.3.26", "@types/react": "19.2.2",
"@types/react-dom": "^18.0.0", "@types/react-dom": "^19.2.2",
"eslint": "^8.45.0", "eslint": "^9.39.1",
"eslint-config-next": "^14.0.0", "eslint-config-next": "^16.0.1",
"jest": "^29.7.0", "jest": "^30.2.0",
"typescript": "^5.0.0" "typescript": "^5.9.3"
} }
} }

View File

@ -1,6 +1,6 @@
module.exports = { module.exports = {
plugins: { plugins: {
tailwindcss: {}, "@tailwindcss/postcss": {},
autoprefixer: {}, autoprefixer: {},
}, },
} };

View File

@ -1,32 +1,162 @@
import { NextRequest, NextResponse } from 'next/server' import { NextRequest, NextResponse } from "next/server";
import { getDatabase } from '../../../lib/database/index' import { getDatabase } from "../../../lib/database/index";
import bcrypt from "bcryptjs";
export async function GET(request: NextRequest) { export async function GET(request: NextRequest) {
try { try {
const db = await getDatabase() const db = await getDatabase();
const { searchParams } = new URL(request.url) const { searchParams } = new URL(request.url);
const role = searchParams.get('role') const role = searchParams.get("role");
let users = await db.getAllUsers() let users = await db.getAllUsers();
if (role) { if (role) {
users = users.filter(user => user.role === role) users = users.filter((user) => user.role === role);
} }
const usersWithClients = await Promise.all( const usersWithClients = await Promise.all(
users.map(async (user) => { users.map(async (user) => {
const { password: _, ...userWithoutPassword } = user const { password: _, ...userWithoutPassword } = user;
const client = await db.getClientByUserId(user.id) const client = await db.getClientByUserId(user.id);
return { ...userWithoutPassword, client } return { ...userWithoutPassword, client };
}) }),
) );
return NextResponse.json({ users: usersWithClients }) return NextResponse.json({ users: usersWithClients });
} catch (error) { } catch (error) {
console.error('Get users error:', error) console.error("Get users error:", error);
return NextResponse.json( return NextResponse.json(
{ error: 'Internal server error' }, { error: "Internal server error" },
{ status: 500 } { status: 500 },
) );
}
}
export async function POST(request: NextRequest) {
try {
const db = await getDatabase();
const body = await request.json();
const { email, password, firstName, lastName, role, phone } = body;
if (!email || !password || !firstName || !lastName || !role) {
return NextResponse.json(
{ error: "Missing required fields" },
{ status: 400 },
);
}
// Check if user already exists
const existingUser = await db.getUserByEmail(email);
if (existingUser) {
return NextResponse.json(
{ error: "User with this email already exists" },
{ status: 409 },
);
}
// Hash password
const hashedPassword = await bcrypt.hash(password, 12);
// Create user
const userId = await db.createUser({
email,
password: hashedPassword,
firstName,
lastName,
role,
phone,
});
return NextResponse.json({ userId }, { status: 201 });
} catch (error) {
console.error("Create user error:", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function PUT(request: NextRequest) {
try {
const db = await getDatabase();
const body = await request.json();
const { id, email, firstName, lastName, role, phone } = body;
if (!id) {
return NextResponse.json(
{ error: "User ID is required" },
{ status: 400 },
);
}
// Get existing user
const existingUser = await db.getUserById(id);
if (!existingUser) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
// Check if email is being changed and if it's already taken
if (email && email !== existingUser.email) {
const userWithEmail = await db.getUserByEmail(email);
if (userWithEmail) {
return NextResponse.json(
{ error: "Email already in use" },
{ status: 409 },
);
}
}
// Update user
await db.updateUser(id, {
email: email || existingUser.email,
firstName: firstName || existingUser.firstName,
lastName: lastName || existingUser.lastName,
role: role || existingUser.role,
phone: phone !== undefined ? phone : existingUser.phone,
});
return NextResponse.json({ success: true });
} catch (error) {
console.error("Update user error:", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
}
}
export async function DELETE(request: NextRequest) {
try {
const db = await getDatabase();
const { searchParams } = new URL(request.url);
const id = searchParams.get("id");
const body = await request.json().catch(() => ({}));
const { ids } = body;
if (ids && Array.isArray(ids)) {
// Bulk delete
await Promise.all(ids.map((userId: string) => db.deleteUser(userId)));
return NextResponse.json({ success: true, deleted: ids.length });
} else if (id) {
// Single delete
const user = await db.getUserById(id);
if (!user) {
return NextResponse.json({ error: "User not found" }, { status: 404 });
}
await db.deleteUser(id);
return NextResponse.json({ success: true });
} else {
return NextResponse.json(
{ error: "User ID or IDs array required" },
{ status: 400 },
);
}
} catch (error) {
console.error("Delete user error:", error);
return NextResponse.json(
{ error: "Internal server error" },
{ status: 500 },
);
} }
} }

View File

@ -1,8 +1,8 @@
'use client' "use client";
import Link from 'next/link' import Link from "next/link";
import { UserManagement } from '@/components/users/UserManagement' import { UserManagement } from "@/components/users/UserManagement";
import { AnalyticsDashboard } from '@/components/analytics/AnalyticsDashboard' import { AnalyticsDashboard } from "@/components/analytics/AnalyticsDashboard";
export default function Home() { export default function Home() {
return ( return (
@ -31,7 +31,9 @@ export default function Home() {
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6 mb-8"> <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6 mb-8">
<div className="bg-white rounded-lg shadow p-6"> <div className="bg-white rounded-lg shadow p-6">
<h2 className="text-xl font-semibold mb-4">Client Management</h2> <h2 className="text-xl font-semibold mb-4">Client Management</h2>
<p className="text-gray-600">Manage fitness clients and their profiles</p> <p className="text-gray-600">
Manage fitness clients and their profiles
</p>
</div> </div>
<div className="bg-white rounded-lg shadow p-6"> <div className="bg-white rounded-lg shadow p-6">
<h2 className="text-xl font-semibold mb-4">Payment Tracking</h2> <h2 className="text-xl font-semibold mb-4">Payment Tracking</h2>
@ -45,20 +47,22 @@ export default function Home() {
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6"> <div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<div className="bg-white rounded-lg shadow p-6"> <div className="bg-white rounded-lg shadow p-6">
<h2 className="text-2xl font-semibold mb-6">Recent User Activity</h2> <h2 className="text-2xl font-semibold mb-6">
<div className="h-96"> Recent User Activity
</h2>
<div>
<UserManagement /> <UserManagement />
</div> </div>
</div> </div>
<div className="bg-white rounded-lg shadow p-6"> <div className="bg-white rounded-lg shadow p-6">
<h2 className="text-2xl font-semibold mb-6">Quick Analytics</h2> <h2 className="text-2xl font-semibold mb-6">Quick Analytics</h2>
<div className="h-96"> <div>
<AnalyticsDashboard /> <AnalyticsDashboard />
</div> </div>
</div> </div>
</div> </div>
</div> </div>
</main> </main>
) );
} }

View File

@ -1,78 +1,90 @@
'use client' "use client";
import React, { useMemo } from 'react' import React, { useMemo } from "react";
import { AgChartsReact } from 'ag-charts-react' import { AgCharts } from "ag-charts-react";
import { AgChartOptions } from 'ag-charts-community' import { AgChartOptions } from "ag-charts-community";
interface PieData { interface PieData {
label: string label: string;
value: number value: number;
color?: string color?: string;
} }
interface MembershipDistributionChartProps { interface MembershipDistributionChartProps {
data: PieData[] data: PieData[];
title?: string title?: string;
} }
export function MembershipDistributionChart({ data, title = 'Membership Distribution' }: MembershipDistributionChartProps) { export function MembershipDistributionChart({
const chartOptions: AgChartOptions = useMemo(() => ({ data,
title = "Membership Distribution",
}: MembershipDistributionChartProps) {
const chartOptions: AgChartOptions = useMemo(
() => ({
title: { title: {
text: title, text: title,
fontSize: 18, fontSize: 18,
fontWeight: 'bold', fontWeight: "bold",
}, },
data, data,
series: [ series: [
{ {
type: 'pie', type: "pie",
calloutLabelKey: 'label', calloutLabelKey: "label",
angleKey: 'value', angleKey: "value",
sectorLabelKey: 'label', sectorLabelKey: "label",
fills: data.map(item => item.color || '#3b82f6'), fills: data.map((item) => item.color || "#3b82f6"),
strokes: ['#ffffff'], strokes: ["#ffffff"],
strokeWidth: 2, strokeWidth: 2,
calloutLabel: { calloutLabel: {
enabled: true, enabled: true,
fontSize: 12, fontSize: 12,
fontWeight: 'bold', fontWeight: "bold",
}, },
sectorLabel: { sectorLabel: {
enabled: true, enabled: true,
fontSize: 14, fontSize: 14,
fontWeight: 'bold', fontWeight: "bold",
color: '#ffffff', color: "#ffffff",
formatter: (params: any) => { formatter: (params: any) => {
const percentage = ((params.datum.value / data.reduce((sum, item) => sum + item.value, 0)) * 100).toFixed(1) const percentage = (
return `${params.datum.label}: ${percentage}%` (params.datum.value /
data.reduce((sum, item) => sum + item.value, 0)) *
100
).toFixed(1);
return `${params.datum.label}: ${percentage}%`;
}, },
}, },
highlightStyle: { highlightStyle: {
item: { item: {
fillOpacity: 0.8, fillOpacity: 0.8,
stroke: '#000000', stroke: "#000000",
strokeWidth: 2, strokeWidth: 2,
}, },
}, },
tooltip: { tooltip: {
enabled: true, enabled: true,
renderer: (params: any) => { renderer: (params: any) => {
const percentage = ((params.datum.value / data.reduce((sum, item) => sum + item.value, 0)) * 100).toFixed(1) const percentage = (
(params.datum.value /
data.reduce((sum, item) => sum + item.value, 0)) *
100
).toFixed(1);
return `<div class="bg-white p-2 rounded shadow-lg border"> return `<div class="bg-white p-2 rounded shadow-lg border">
<div class="font-bold">${params.datum.label}</div> <div class="font-bold">${params.datum.label}</div>
<div class="text-sm">Count: ${params.datum.value}</div> <div class="text-sm">Count: ${params.datum.value}</div>
<div class="text-sm">Percentage: ${percentage}%</div> <div class="text-sm">Percentage: ${percentage}%</div>
</div>` </div>`;
}, },
}, },
}, },
], ],
legend: { legend: {
enabled: true, enabled: true,
position: 'right', position: "right",
fontSize: 12, fontSize: 12,
marker: { marker: {
shape: 'square', shape: "square",
size: 12, size: 12,
}, },
}, },
@ -82,7 +94,9 @@ export function MembershipDistributionChart({ data, title = 'Membership Distribu
bottom: 20, bottom: 20,
left: 20, left: 20,
}, },
}), [data, title]) }),
[data, title],
);
return <AgChartsReact options={chartOptions} /> return <AgCharts options={chartOptions} />;
} }

View File

@ -1,50 +1,54 @@
'use client' "use client";
import React, { useMemo } from 'react' import React, { useMemo } from "react";
import { AgChartsReact } from 'ag-charts-react' import { AgCharts } from "ag-charts-react";
import { AgChartOptions } from 'ag-charts-community' import { AgChartOptions } from "ag-charts-community";
interface BarData { interface BarData {
category: string category: string;
value: number value: number;
color?: string color?: string;
} }
interface RevenueChartProps { interface RevenueChartProps {
data: BarData[] data: BarData[];
title?: string title?: string;
} }
export function RevenueChart({ data, title = 'Monthly Revenue' }: RevenueChartProps) { export function RevenueChart({
const chartOptions: AgChartOptions = useMemo(() => ({ data,
title = "Monthly Revenue",
}: RevenueChartProps) {
const chartOptions: AgChartOptions = useMemo(
() => ({
title: { title: {
text: title, text: title,
fontSize: 18, fontSize: 18,
fontWeight: 'bold', fontWeight: "bold",
}, },
data, data,
series: [ series: [
{ {
type: 'bar', type: "bar",
xKey: 'category', xKey: "category",
yKey: 'value', yKey: "value",
fills: data.map(item => item.color || '#10b981'), fills: data.map((item) => item.color || "#10b981"),
strokes: ['#ffffff'], strokes: ["#ffffff"],
strokeWidth: 2, strokeWidth: 2,
cornerRadius: 4, cornerRadius: 4,
highlightStyle: { highlightStyle: {
item: { item: {
fill: '#059669', fill: "#059669",
stroke: '#ffffff', stroke: "#ffffff",
strokeWidth: 2, strokeWidth: 2,
}, },
}, },
label: { label: {
enabled: true, enabled: true,
position: 'top', position: "top",
fontSize: 12, fontSize: 12,
fontWeight: 'bold', fontWeight: "bold",
color: '#374151', color: "#374151",
formatter: (params: any) => `$${params.value.toLocaleString()}`, formatter: (params: any) => `$${params.value.toLocaleString()}`,
}, },
tooltip: { tooltip: {
@ -53,17 +57,17 @@ export function RevenueChart({ data, title = 'Monthly Revenue' }: RevenueChartPr
return `<div class="bg-white p-2 rounded shadow-lg border"> return `<div class="bg-white p-2 rounded shadow-lg border">
<div class="font-bold">${params.datum.category}</div> <div class="font-bold">${params.datum.category}</div>
<div class="text-sm">Revenue: $${params.datum.value.toLocaleString()}</div> <div class="text-sm">Revenue: $${params.datum.value.toLocaleString()}</div>
</div>` </div>`;
}, },
}, },
}, },
], ],
axes: [ axes: [
{ {
type: 'category', type: "category",
position: 'bottom', position: "bottom",
title: { title: {
text: 'Month', text: "Month",
fontSize: 14, fontSize: 14,
}, },
label: { label: {
@ -72,10 +76,10 @@ export function RevenueChart({ data, title = 'Monthly Revenue' }: RevenueChartPr
}, },
}, },
{ {
type: 'number', type: "number",
position: 'left', position: "left",
title: { title: {
text: 'Revenue ($)', text: "Revenue ($)",
fontSize: 14, fontSize: 14,
}, },
label: { label: {
@ -93,7 +97,9 @@ export function RevenueChart({ data, title = 'Monthly Revenue' }: RevenueChartPr
bottom: 60, bottom: 60,
left: 80, left: 80,
}, },
}), [data, title]) }),
[data, title],
);
return <AgChartsReact options={chartOptions} /> return <AgCharts options={chartOptions} />;
} }

View File

@ -1,45 +1,49 @@
'use client' "use client";
import React, { useMemo } from 'react' import React, { useMemo } from "react";
import { AgChartsReact } from 'ag-charts-react' import { AgCharts } from "ag-charts-react";
import { AgChartOptions } from 'ag-charts-community' import { AgChartOptions } from "ag-charts-community";
interface ChartData { interface ChartData {
label: string label: string;
value: number value: number;
color?: string color?: string;
} }
interface UserGrowthChartProps { interface UserGrowthChartProps {
data: ChartData[] data: ChartData[];
title?: string title?: string;
} }
export function UserGrowthChart({ data, title = 'User Growth' }: UserGrowthChartProps) { export function UserGrowthChart({
const chartOptions: AgChartOptions = useMemo(() => ({ data,
title = "User Growth",
}: UserGrowthChartProps) {
const chartOptions: AgChartOptions = useMemo(
() => ({
title: { title: {
text: title, text: title,
fontSize: 18, fontSize: 18,
fontWeight: 'bold', fontWeight: "bold",
}, },
data, data,
series: [ series: [
{ {
type: 'line', type: "line",
xKey: 'label', xKey: "label",
yKey: 'value', yKey: "value",
stroke: '#3b82f6', stroke: "#3b82f6",
strokeWidth: 3, strokeWidth: 3,
marker: { marker: {
size: 6, size: 6,
fill: '#3b82f6', fill: "#3b82f6",
stroke: '#ffffff', stroke: "#ffffff",
strokeWidth: 2, strokeWidth: 2,
}, },
highlightStyle: { highlightStyle: {
item: { item: {
fill: '#1d4ed8', fill: "#1d4ed8",
stroke: '#ffffff', stroke: "#ffffff",
strokeWidth: 2, strokeWidth: 2,
}, },
}, },
@ -47,18 +51,18 @@ export function UserGrowthChart({ data, title = 'User Growth' }: UserGrowthChart
], ],
axes: [ axes: [
{ {
type: 'category', type: "category",
position: 'bottom', position: "bottom",
title: { title: {
text: 'Time Period', text: "Time Period",
fontSize: 14, fontSize: 14,
}, },
}, },
{ {
type: 'number', type: "number",
position: 'left', position: "left",
title: { title: {
text: 'Number of Users', text: "Number of Users",
fontSize: 14, fontSize: 14,
}, },
}, },
@ -72,7 +76,9 @@ export function UserGrowthChart({ data, title = 'User Growth' }: UserGrowthChart
bottom: 20, bottom: 20,
left: 20, left: 20,
}, },
}), [data, title]) }),
[data, title],
);
return <AgChartsReact options={chartOptions} /> return <AgCharts options={chartOptions} />;
} }

View File

@ -1,167 +1,248 @@
'use client' "use client";
import React, { useState, useMemo } from 'react' import React, { useState, useMemo } from "react";
import { AgGridReact } from 'ag-grid-react' import { AgGridReact } from "ag-grid-react";
import { ColDef } from 'ag-grid-community' import { ColDef, ModuleRegistry, AllCommunityModule } from "ag-grid-community";
import 'ag-grid-community/styles/ag-grid.css' import "ag-grid-community/styles/ag-grid.css";
import 'ag-grid-community/styles/ag-theme-alpine.css' import "ag-grid-community/styles/ag-theme-alpine.css";
import { formatDate } from '@/lib/utils' import { formatDate } from "@/lib/utils";
ModuleRegistry.registerModules([AllCommunityModule]);
interface User { interface User {
id: string id: string;
email: string email: string;
firstName: string firstName: string;
lastName: string lastName: string;
role: string role: string;
phone?: string phone?: string;
createdAt: Date createdAt: Date;
client?: { client?: {
id: string id: string;
membershipType: string membershipType: string;
membershipStatus: string membershipStatus: string;
joinDate: Date joinDate: Date;
lastVisit?: Date lastVisit?: Date;
} };
} }
interface UserGridProps { interface UserGridProps {
users: User[] users: User[];
onUserSelect?: (user: User) => void onUserSelect?: (user: User) => void;
loading?: boolean onEditUser?: (user: User) => void;
onDeleteUser?: (user: User) => void;
onBulkDelete?: (users: User[]) => void;
loading?: boolean;
} }
export function UserGrid({ users, onUserSelect, loading = false }: UserGridProps) { export function UserGrid({
const [selectedUser, setSelectedUser] = useState<User | null>(null) users,
onUserSelect,
onEditUser,
onDeleteUser,
onBulkDelete,
loading = false,
}: UserGridProps) {
const [selectedUsers, setSelectedUsers] = useState<User[]>([]);
const [searchQuery, setSearchQuery] = useState<string>("");
const columnDefs: ColDef<User>[] = useMemo(() => [ const columnDefs: ColDef<User>[] = useMemo(
() => [
{ {
headerName: 'Name', headerName: "Name",
valueGetter: (params) => `${params.data?.firstName} ${params.data?.lastName}`, valueGetter: (params) =>
filter: 'agTextColumnFilter', `${params.data?.firstName} ${params.data?.lastName}`,
filter: "agTextColumnFilter",
sortable: true, sortable: true,
minWidth: 150, minWidth: 150,
}, },
{ {
headerName: 'Email', headerName: "Email",
field: 'email', field: "email",
filter: 'agTextColumnFilter', filter: "agTextColumnFilter",
sortable: true, sortable: true,
minWidth: 200, minWidth: 200,
}, },
{ {
headerName: 'Role', headerName: "Role",
field: 'role', field: "role",
filter: 'agSetColumnFilter', filter: "agTextColumnFilter",
sortable: true, sortable: true,
cellRenderer: (params: any) => { cellRenderer: (params: any) => {
const roleColors = { const roleColors = {
admin: 'bg-purple-100 text-purple-800', admin: "bg-purple-100 text-purple-800",
trainer: 'bg-blue-100 text-blue-800', trainer: "bg-blue-100 text-blue-800",
client: 'bg-green-100 text-green-800', client: "bg-green-100 text-green-800",
} };
const colorClass = roleColors[params.value as keyof typeof roleColors] || 'bg-gray-100 text-gray-800' const colorClass =
return `<span class="px-2 py-1 rounded text-xs font-medium ${colorClass}">${params.value}</span>` roleColors[params.value as keyof typeof roleColors] ||
"bg-gray-100 text-gray-800";
return `<span class="px-2 py-1 rounded text-xs font-medium ${colorClass}">${params.value}</span>`;
}, },
minWidth: 120, minWidth: 120,
}, },
{ {
headerName: 'Phone', headerName: "Phone",
field: 'phone', field: "phone",
filter: 'agTextColumnFilter', filter: "agTextColumnFilter",
sortable: true, sortable: true,
minWidth: 130, minWidth: 130,
}, },
{ {
headerName: 'Membership', headerName: "Membership",
valueGetter: (params) => params.data?.client?.membershipType || 'N/A', valueGetter: (params) => params.data?.client?.membershipType || "N/A",
filter: 'agSetColumnFilter', filter: "agTextColumnFilter",
sortable: true, sortable: true,
cellRenderer: (params: any) => { cellRenderer: (params: any) => {
if (!params.value || params.value === 'N/A') return 'N/A' if (!params.value || params.value === "N/A") return "N/A";
const membershipColors = { const membershipColors = {
vip: 'bg-yellow-100 text-yellow-800', vip: "bg-yellow-100 text-yellow-800",
premium: 'bg-blue-100 text-blue-800', premium: "bg-blue-100 text-blue-800",
basic: 'bg-gray-100 text-gray-800', basic: "bg-gray-100 text-gray-800",
} };
const colorClass = membershipColors[params.value as keyof typeof membershipColors] || 'bg-gray-100 text-gray-800' const colorClass =
return `<span class="px-2 py-1 rounded text-xs font-medium ${colorClass}">${params.value}</span>` membershipColors[params.value as keyof typeof membershipColors] ||
"bg-gray-100 text-gray-800";
return `<span class="px-2 py-1 rounded text-xs font-medium ${colorClass}">${params.value}</span>`;
}, },
minWidth: 120, minWidth: 120,
}, },
{ {
headerName: 'Status', headerName: "Status",
valueGetter: (params) => params.data?.client?.membershipStatus || 'N/A', valueGetter: (params) => params.data?.client?.membershipStatus || "N/A",
filter: 'agSetColumnFilter', filter: "agTextColumnFilter",
sortable: true, sortable: true,
cellRenderer: (params: any) => { cellRenderer: (params: any) => {
if (!params.value || params.value === 'N/A') return 'N/A' if (!params.value || params.value === "N/A") return "N/A";
const statusColors = { const statusColors = {
active: 'bg-green-100 text-green-800', active: "bg-green-100 text-green-800",
inactive: 'bg-red-100 text-red-800', inactive: "bg-red-100 text-red-800",
suspended: 'bg-yellow-100 text-yellow-800', suspended: "bg-yellow-100 text-yellow-800",
} };
const colorClass = statusColors[params.value as keyof typeof statusColors] || 'bg-gray-100 text-gray-800' const colorClass =
return `<span class="px-2 py-1 rounded text-xs font-medium ${colorClass}">${params.value}</span>` statusColors[params.value as keyof typeof statusColors] ||
"bg-gray-100 text-gray-800";
return `<span class="px-2 py-1 rounded text-xs font-medium ${colorClass}">${params.value}</span>`;
}, },
minWidth: 120, minWidth: 120,
}, },
{ {
headerName: 'Join Date', headerName: "Join Date",
valueGetter: (params) => params.data?.client?.joinDate || params.data?.createdAt, valueGetter: (params) =>
filter: 'agDateColumnFilter', params.data?.client?.joinDate || params.data?.createdAt,
filter: "agDateColumnFilter",
sortable: true, sortable: true,
valueFormatter: (params: any) => formatDate(new Date(params.value)), valueFormatter: (params: any) => formatDate(new Date(params.value)),
minWidth: 120, minWidth: 120,
}, },
{ {
headerName: 'Last Visit', headerName: "Last Visit",
valueGetter: (params) => params.data?.client?.lastVisit, valueGetter: (params) => params.data?.client?.lastVisit,
filter: 'agDateColumnFilter', filter: "agDateColumnFilter",
sortable: true, sortable: true,
valueFormatter: (params: any) => params.value ? formatDate(new Date(params.value)) : 'Never', valueFormatter: (params: any) =>
params.value ? formatDate(new Date(params.value)) : "Never",
minWidth: 120, minWidth: 120,
}, },
], []) ],
[],
);
const defaultColDef: ColDef = useMemo(() => ({ const defaultColDef: ColDef = useMemo(
() => ({
flex: 1, flex: 1,
resizable: true, resizable: true,
floatingFilter: true, floatingFilter: true,
suppressMenu: true, suppressMenu: true,
}), []) }),
[],
);
const onSelectionChanged = () => { const gridRef = React.useRef<AgGridReact<User>>(null);
const selectedNodes = gridRef.current?.api.getSelectedNodes()
if (selectedNodes?.length > 0) {
const user = selectedNodes[0].data
setSelectedUser(user)
onUserSelect?.(user)
}
}
const gridRef = React.useRef<AgGridReact<User>>(null)
const gridOptions = { const gridOptions = {
theme: "legacy",
columnDefs, columnDefs,
defaultColDef, defaultColDef,
rowData: users, rowData: users,
rowSelection: 'single', rowSelection: "multiple",
onSelectionChanged, onSelectionChanged: () => {
enableRangeSelection: true, const selectedNodes = gridRef.current?.api.getSelectedNodes();
enableCellTextSelection: true, const selectedData = selectedNodes?.map((node) => node.data) || [];
setSelectedUsers(selectedData);
if (selectedData.length === 1 && onUserSelect) {
onUserSelect(selectedData[0]);
}
},
suppressRowClickSelection: false, suppressRowClickSelection: false,
animateRows: true, animateRows: true,
loading: loading, loading: loading,
pagination: true, pagination: true,
paginationPageSize: 20, paginationPageSize: 20,
paginationPageSizeSelector: [10, 20, 50, 100], paginationPageSizeSelector: [10, 20, 50, 100],
quickFilterText: searchQuery,
};
const handleEdit = () => {
if (selectedUsers.length === 1 && onEditUser) {
onEditUser(selectedUsers[0]);
} }
};
const handleDelete = () => {
if (selectedUsers.length === 1 && onDeleteUser) {
onDeleteUser(selectedUsers[0]);
}
};
const handleBulkDelete = () => {
if (selectedUsers.length > 0 && onBulkDelete) {
onBulkDelete(selectedUsers);
}
};
return ( return (
<div className="ag-theme-alpine" style={{ height: '600px', width: '100%' }}> <div>
<div className="flex justify-between items-center mb-4">
<input
type="text"
placeholder="Search users..."
className="border border-gray-300 rounded px-4 py-2"
value={searchQuery}
onChange={(e) => setSearchQuery(e.target.value)}
/>
<div className="flex gap-2">
<button
className="bg-green-500 text-white px-4 py-2 rounded disabled:opacity-50"
onClick={handleEdit}
disabled={selectedUsers.length !== 1}
>
Edit User
</button>
<button
className="bg-red-500 text-white px-4 py-2 rounded disabled:opacity-50"
onClick={handleDelete}
disabled={selectedUsers.length !== 1}
>
Delete User
</button>
<button
className="bg-yellow-500 text-white px-4 py-2 rounded disabled:opacity-50"
onClick={handleBulkDelete}
disabled={selectedUsers.length === 0}
>
Bulk Delete
</button>
</div>
</div>
<div
className="ag-theme-alpine"
style={{ height: "600px", width: "100%" }}
>
<AgGridReact<User> {...gridOptions} ref={gridRef} /> <AgGridReact<User> {...gridOptions} ref={gridRef} />
</div> </div>
) </div>
);
} }

View File

@ -1,85 +1,182 @@
'use client' "use client";
import { useState, useEffect } from 'react' import { useState, useEffect } from "react";
import { UserGrid } from '@/components/users/UserGrid' import { UserGrid } from "@/components/users/UserGrid";
import { Button } from '@/components/ui/Button' import { Button } from "@/components/ui/Button";
import { Card, CardHeader, CardContent } from '@/components/ui/Card' import { Card, CardHeader, CardContent } from "@/components/ui/Card";
interface User { interface User {
id: string id: string;
email: string email: string;
firstName: string firstName: string;
lastName: string lastName: string;
role: string role: string;
phone?: string phone?: string;
createdAt: Date createdAt: Date;
client?: { client?: {
id: string id: string;
membershipType: string membershipType: string;
membershipStatus: string membershipStatus: string;
joinDate: Date joinDate: Date;
lastVisit?: Date lastVisit?: Date;
} };
} }
export function UserManagement() { export function UserManagement() {
const [users, setUsers] = useState<User[]>([]) const [users, setUsers] = useState<User[]>([]);
const [loading, setLoading] = useState(true) const [loading, setLoading] = useState(true);
const [filter, setFilter] = useState<string>('all') const [filter, setFilter] = useState<string>("all");
const [selectedUser, setSelectedUser] = useState<User | null>(null) const [selectedUser, setSelectedUser] = useState<User | null>(null);
const [isEditing, setIsEditing] = useState(false);
const [isDeleting, setIsDeleting] = useState(false);
const [editForm, setEditForm] = useState<{
firstName: string;
lastName: string;
email: string;
role: string;
phone: string;
} | null>(null);
useEffect(() => { useEffect(() => {
fetchUsers() fetchUsers();
}, [filter]) }, [filter]);
const fetchUsers = async () => { const fetchUsers = async () => {
setLoading(true) setLoading(true);
try { try {
const url = filter === 'all' const url = filter === "all" ? "/api/users" : `/api/users?role=${filter}`;
? '/api/users'
: `/api/users?role=${filter}`
const response = await fetch(url) const response = await fetch(url);
const data = await response.json() const data = await response.json();
setUsers(data.users || []) setUsers(data.users || []);
} catch (error) { } catch (error) {
console.error('Failed to fetch users:', error) console.error("Failed to fetch users:", error);
} finally { } finally {
setLoading(false) setLoading(false);
}
} }
};
const handleUserSelect = (user: User) => { const handleUserSelect = (user: User | null) => {
setSelectedUser(user) setSelectedUser(user);
};
const handleEditUser = (user: User) => {
setSelectedUser(user);
setEditForm({
firstName: user.firstName,
lastName: user.lastName,
email: user.email,
role: user.role,
phone: user.phone || "",
});
setIsEditing(true);
};
const handleDeleteUser = (user: User) => {
setSelectedUser(user);
setIsDeleting(true);
};
const handleBulkDelete = async (users: User[]) => {
if (users.length === 0) return;
if (!confirm(`Are you sure you want to delete ${users.length} users?`))
return;
try {
const response = await fetch("/api/users", {
method: "DELETE",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ ids: users.map((u) => u.id) }),
});
if (response.ok) {
fetchUsers();
} else {
alert("Error deleting users");
} }
} catch (error) {
console.error(error);
}
};
const handleExport = () => { const handleExport = () => {
const csvContent = [ const csvContent = [
['Name', 'Email', 'Role', 'Phone', 'Membership', 'Status', 'Join Date', 'Last Visit'], [
...users.map(user => [ "Name",
"Email",
"Role",
"Phone",
"Membership",
"Status",
"Join Date",
"Last Visit",
],
...users.map((user) => [
`${user.firstName} ${user.lastName}`, `${user.firstName} ${user.lastName}`,
user.email, user.email,
user.role, user.role,
user.phone || '', user.phone || "",
user.client?.membershipType || '', user.client?.membershipType || "",
user.client?.membershipStatus || '', user.client?.membershipStatus || "",
user.client?.joinDate || user.createdAt, user.client?.joinDate || user.createdAt,
user.client?.lastVisit || '' user.client?.lastVisit || "",
]) ]),
].map(row => row.join(',')).join('\n') ]
.map((row) => row.join(","))
.join("\n");
const blob = new Blob([csvContent], { type: 'text/csv' }) const blob = new Blob([csvContent], { type: "text/csv" });
const url = window.URL.createObjectURL(blob) const url = window.URL.createObjectURL(blob);
const a = document.createElement('a') const a = document.createElement("a");
a.href = url a.href = url;
a.download = `users_${new Date().toISOString().split('T')[0]}.csv` a.download = `users_${new Date().toISOString().split("T")[0]}.csv`;
a.click() a.click();
window.URL.revokeObjectURL(url) window.URL.revokeObjectURL(url);
} };
const handleRefresh = () => { const handleRefresh = () => {
fetchUsers() fetchUsers();
};
const handleSaveEdit = async () => {
if (!editForm || !selectedUser) return;
try {
const response = await fetch("/api/users", {
method: "PUT",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ id: selectedUser.id, ...editForm }),
});
if (response.ok) {
setIsEditing(false);
setEditForm(null);
fetchUsers();
} else {
alert("Error updating user");
} }
} catch (error) {
console.error(error);
}
};
const handleDeleteConfirm = async () => {
if (!selectedUser) return;
try {
const response = await fetch(`/api/users?id=${selectedUser.id}`, {
method: "DELETE",
});
if (response.ok) {
setIsDeleting(false);
setSelectedUser(null);
fetchUsers();
} else {
alert("Error deleting user");
}
} catch (error) {
console.error(error);
}
};
return ( return (
<div className="space-y-6"> <div className="space-y-6">
@ -87,26 +184,40 @@ export function UserManagement() {
<h2 className="text-2xl font-bold">User Management</h2> <h2 className="text-2xl font-bold">User Management</h2>
<div className="flex gap-2"> <div className="flex gap-2">
<Button <Button
variant={filter === 'all' ? 'primary' : 'secondary'} variant={filter === "all" ? "primary" : "secondary"}
onClick={() => setFilter('all')} onClick={() => setFilter("all")}
> >
All Users All Users
</Button> </Button>
<Button <Button
variant={filter === 'client' ? 'primary' : 'secondary'} variant="secondary"
onClick={() => setFilter('client')} onClick={handleEditUser}
disabled={!selectedUser}
>
Edit User
</Button>
<Button
variant="secondary"
onClick={handleDeleteUser}
disabled={!selectedUser}
>
Delete User
</Button>
<Button
variant={filter === "client" ? "primary" : "secondary"}
onClick={() => setFilter("client")}
> >
Clients Clients
</Button> </Button>
<Button <Button
variant={filter === 'trainer' ? 'primary' : 'secondary'} variant={filter === "trainer" ? "primary" : "secondary"}
onClick={() => setFilter('trainer')} onClick={() => setFilter("trainer")}
> >
Trainers Trainers
</Button> </Button>
<Button <Button
variant={filter === 'admin' ? 'primary' : 'secondary'} variant={filter === "admin" ? "primary" : "secondary"}
onClick={() => setFilter('admin')} onClick={() => setFilter("admin")}
> >
Admins Admins
</Button> </Button>
@ -136,12 +247,140 @@ export function UserManagement() {
<CardContent className="p-0"> <CardContent className="p-0">
<UserGrid <UserGrid
users={users} users={users}
onUserSelect={handleUserSelect} onUserSelect={(user) => handleUserSelect(user)}
onEditUser={handleEditUser}
onDeleteUser={handleDeleteUser}
onBulkDelete={handleBulkDelete}
loading={loading} loading={loading}
/> />
</CardContent> </CardContent>
</Card> </Card>
{isEditing && editForm && (
<div className="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50">
<div className="bg-white p-6 rounded-lg shadow-lg max-w-md w-full">
<h3 className="text-lg font-semibold mb-4">Edit User</h3>
<form
onSubmit={(e) => {
e.preventDefault();
handleSaveEdit();
}}
>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">
First Name
</label>
<input
type="text"
value={editForm.firstName}
onChange={(e) =>
setEditForm({ ...editForm, firstName: e.target.value })
}
className="w-full border border-gray-300 rounded px-3 py-2"
required
/>
</div>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">
Last Name
</label>
<input
type="text"
value={editForm.lastName}
onChange={(e) =>
setEditForm({ ...editForm, lastName: e.target.value })
}
className="w-full border border-gray-300 rounded px-3 py-2"
required
/>
</div>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">Email</label>
<input
type="email"
value={editForm.email}
onChange={(e) =>
setEditForm({ ...editForm, email: e.target.value })
}
className="w-full border border-gray-300 rounded px-3 py-2"
required
/>
</div>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">Role</label>
<select
value={editForm.role}
onChange={(e) =>
setEditForm({ ...editForm, role: e.target.value })
}
className="w-full border border-gray-300 rounded px-3 py-2"
required
>
<option value="client">Client</option>
<option value="trainer">Trainer</option>
<option value="admin">Admin</option>
</select>
</div>
<div className="mb-4">
<label className="block text-sm font-medium mb-1">Phone</label>
<input
type="tel"
value={editForm.phone}
onChange={(e) =>
setEditForm({ ...editForm, phone: e.target.value })
}
className="w-full border border-gray-300 rounded px-3 py-2"
/>
</div>
<div className="flex justify-end gap-2">
<button
type="button"
onClick={() => {
setIsEditing(false);
setEditForm(null);
}}
className="px-4 py-2 bg-gray-300 text-gray-700 rounded hover:bg-gray-400"
>
Cancel
</button>
<button
type="submit"
className="px-4 py-2 bg-blue-600 text-white rounded hover:bg-blue-700"
>
Save
</button>
</div>
</form>
</div>
</div>
)}
{isDeleting && selectedUser && (
<div className="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50">
<div className="bg-white p-6 rounded-lg shadow-lg max-w-md w-full">
<h3 className="text-lg font-semibold mb-4">Delete User</h3>
<p className="mb-4">
Are you sure you want to delete {selectedUser.firstName}{" "}
{selectedUser.lastName}? This action cannot be undone.
</p>
<div className="flex justify-end gap-2">
<button
onClick={() => setIsDeleting(false)}
className="px-4 py-2 bg-gray-300 text-gray-700 rounded hover:bg-gray-400"
>
Cancel
</button>
<button
onClick={handleDeleteConfirm}
className="px-4 py-2 bg-red-600 text-white rounded hover:bg-red-700"
>
Delete
</button>
</div>
</div>
</div>
)}
{selectedUser && ( {selectedUser && (
<Card> <Card>
<CardHeader> <CardHeader>
@ -152,11 +391,26 @@ export function UserManagement() {
<div> <div>
<h4 className="font-medium mb-2">Basic Information</h4> <h4 className="font-medium mb-2">Basic Information</h4>
<div className="space-y-1 text-sm"> <div className="space-y-1 text-sm">
<p><span className="font-medium">Name:</span> {selectedUser.firstName} {selectedUser.lastName}</p> <p>
<p><span className="font-medium">Email:</span> {selectedUser.email}</p> <span className="font-medium">Name:</span>{" "}
<p><span className="font-medium">Phone:</span> {selectedUser.phone || 'N/A'}</p> {selectedUser.firstName} {selectedUser.lastName}
<p><span className="font-medium">Role:</span> {selectedUser.role}</p> </p>
<p><span className="font-medium">Joined:</span> {selectedUser.createdAt.toLocaleDateString()}</p> <p>
<span className="font-medium">Email:</span>{" "}
{selectedUser.email}
</p>
<p>
<span className="font-medium">Phone:</span>{" "}
{selectedUser.phone || "N/A"}
</p>
<p>
<span className="font-medium">Role:</span>{" "}
{selectedUser.role}
</p>
<p>
<span className="font-medium">Joined:</span>{" "}
{new Date(selectedUser.createdAt).toLocaleDateString()}
</p>
</div> </div>
</div> </div>
@ -164,10 +418,28 @@ export function UserManagement() {
<div> <div>
<h4 className="font-medium mb-2">Client Information</h4> <h4 className="font-medium mb-2">Client Information</h4>
<div className="space-y-1 text-sm"> <div className="space-y-1 text-sm">
<p><span className="font-medium">Membership:</span> {selectedUser.client.membershipType}</p> <p>
<p><span className="font-medium">Status:</span> {selectedUser.client.membershipStatus}</p> <span className="font-medium">Membership:</span>{" "}
<p><span className="font-medium">Member Since:</span> {selectedUser.client.joinDate.toLocaleDateString()}</p> {selectedUser.client.membershipType}
<p><span className="font-medium">Last Visit:</span> {selectedUser.client.lastVisit?.toLocaleDateString() || 'Never'}</p> </p>
<p>
<span className="font-medium">Status:</span>{" "}
{selectedUser.client.membershipStatus}
</p>
<p>
<span className="font-medium">Member Since:</span>{" "}
{new Date(
selectedUser.client.joinDate,
).toLocaleDateString()}
</p>
<p>
<span className="font-medium">Last Visit:</span>{" "}
{selectedUser.client.lastVisit
? new Date(
selectedUser.client.lastVisit,
).toLocaleDateString()
: "Never"}
</p>
</div> </div>
</div> </div>
)} )}
@ -176,5 +448,5 @@ export function UserManagement() {
</Card> </Card>
)} )}
</div> </div>
) );
} }

View File

@ -15,7 +15,7 @@
"moduleResolution": "bundler", "moduleResolution": "bundler",
"resolveJsonModule": true, "resolveJsonModule": true,
"isolatedModules": true, "isolatedModules": true,
"jsx": "preserve", "jsx": "react-jsx",
"incremental": true, "incremental": true,
"plugins": [ "plugins": [
{ {

File diff suppressed because it is too large Load Diff

View File

@ -14,16 +14,16 @@
"test": "jest" "test": "jest"
}, },
"dependencies": { "dependencies": {
"@expo/vector-icons": "^15.0.3", "@expo/vector-icons": "^15.0.0",
"@hookform/resolvers": "^3.3.0", "@hookform/resolvers": "^3.3.0",
"@tanstack/react-query": "^5.0.0", "@tanstack/react-query": "^5.0.0",
"ajv": "^8.12.0", "ajv": "^8.12.0",
"ajv-keywords": "^5.1.0", "ajv-keywords": "^5.1.0",
"axios": "^1.6.0", "axios": "^1.6.0",
"expo": "~54.0.0", "expo": "~54.0.23",
"expo-camera": "~17.0.9", "expo-camera": "~17.0.0",
"expo-linking": "~8.0.8", "expo-linking": "~8.0.0",
"expo-notifications": "~0.32.12", "expo-notifications": "~0.32.0",
"expo-router": "~6.0.14", "expo-router": "~6.0.14",
"expo-secure-store": "~15.0.7", "expo-secure-store": "~15.0.7",
"react": "19.1.0", "react": "19.1.0",
@ -44,6 +44,6 @@
"jest": "^29.2.1", "jest": "^29.2.1",
"@testing-library/react-native": "^12.4.0", "@testing-library/react-native": "^12.4.0",
"react-test-renderer": "19.1.0", "react-test-renderer": "19.1.0",
"babel-preset-expo": "~54.0.0" "babel-preset-expo": "~54.0.7"
} }
} }

885
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -21,12 +21,12 @@
"typecheck:mobile": "cd apps/mobile && npx tsc --noEmit" "typecheck:mobile": "cd apps/mobile && npx tsc --noEmit"
}, },
"devDependencies": { "devDependencies": {
"@typescript-eslint/eslint-plugin": "^6.0.0", "@typescript-eslint/eslint-plugin": "^8.46.3",
"@typescript-eslint/parser": "^6.0.0", "@typescript-eslint/parser": "^8.46.3",
"concurrently": "^8.2.2", "concurrently": "^9.2.1",
"eslint": "^8.45.0", "eslint": "^9.39.1",
"prettier": "^3.0.0", "prettier": "^3.6.2",
"typescript": "^5.0.0" "typescript": "^5.9.3"
}, },
"engines": { "engines": {
"node": ">=18.0.0", "node": ">=18.0.0",

1667
packages/database/package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

View File

@ -11,12 +11,12 @@
"db:studio": "drizzle-kit studio" "db:studio": "drizzle-kit studio"
}, },
"dependencies": { "dependencies": {
"drizzle-orm": "^0.29.0", "drizzle-orm": "^0.44.7",
"better-sqlite3": "^9.0.0", "better-sqlite3": "^12.4.1",
"@types/better-sqlite3": "^7.6.0" "@types/better-sqlite3": "^7.6.13"
}, },
"devDependencies": { "devDependencies": {
"typescript": "^5.0.0", "typescript": "^5.9.3",
"drizzle-kit": "^0.20.0" "drizzle-kit": "^0.31.6"
} }
} }

41
packages/shared/package-lock.json generated Normal file
View File

@ -0,0 +1,41 @@
{
"name": "@fitai/shared",
"version": "1.0.0",
"lockfileVersion": 3,
"requires": true,
"packages": {
"": {
"name": "@fitai/shared",
"version": "1.0.0",
"dependencies": {
"zod": "^4.1.12"
},
"devDependencies": {
"typescript": "^5.9.3"
}
},
"node_modules/typescript": {
"version": "5.9.3",
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
"integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==",
"dev": true,
"license": "Apache-2.0",
"bin": {
"tsc": "bin/tsc",
"tsserver": "bin/tsserver"
},
"engines": {
"node": ">=14.17"
}
},
"node_modules/zod": {
"version": "4.1.12",
"resolved": "https://registry.npmjs.org/zod/-/zod-4.1.12.tgz",
"integrity": "sha512-JInaHOamG8pt5+Ey8kGmdcAcg3OL9reK8ltczgHTAwNhMys/6ThXHityHxVV2p3fkw/c+MAvBHFVYHFZDmjMCQ==",
"license": "MIT",
"funding": {
"url": "https://github.com/sponsors/colinhacks"
}
}
}
}

View File

@ -9,9 +9,9 @@
"typecheck": "tsc --noEmit" "typecheck": "tsc --noEmit"
}, },
"dependencies": { "dependencies": {
"zod": "^3.22.0" "zod": "^4.1.12"
}, },
"devDependencies": { "devDependencies": {
"typescript": "^5.0.0" "typescript": "^5.9.3"
} }
} }