Skip to content

Commit 39ccd3d

Browse files
feat: vehicles page
1 parent 89cb985 commit 39ccd3d

19 files changed

+705
-20
lines changed

.github/workflows/workflow.yaml

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
name: Continuos Integration
2+
3+
on:
4+
push:
5+
branches: ['main', 'dev']
6+
7+
jobs:
8+
continuos_integration:
9+
runs-on: ubuntu-latest
10+
11+
steps:
12+
- uses: actions/checkout@v3
13+
- name: Using Node.jsz
14+
uses: actions/setup-node@v2
15+
with:
16+
node_version: 20.x
17+
- name: run install and build
18+
run: |
19+
npm install
20+
npm run build

next.config.mjs

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,14 @@
11
/** @type {import('next').NextConfig} */
2-
const nextConfig = {};
2+
const nextConfig = {
3+
async redirects() {
4+
return [
5+
{
6+
source: "/",
7+
destination: "/app",
8+
permanent: true,
9+
},
10+
];
11+
},
12+
};
313

414
export default nextConfig;

package.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -11,12 +11,15 @@
1111
"dependencies": {
1212
"@hookform/resolvers": "^3.9.0",
1313
"@radix-ui/react-avatar": "^1.1.0",
14+
"@radix-ui/react-dialog": "^1.1.1",
1415
"@radix-ui/react-dropdown-menu": "^2.1.1",
1516
"@radix-ui/react-icons": "^1.3.0",
1617
"@radix-ui/react-label": "^2.1.0",
1718
"@radix-ui/react-radio-group": "^1.2.0",
1819
"@radix-ui/react-slot": "^1.1.0",
20+
"@radix-ui/react-tabs": "^1.1.0",
1921
"@radix-ui/react-toast": "^1.2.1",
22+
"@tanstack/react-table": "^8.20.5",
2023
"class-variance-authority": "^0.7.0",
2124
"clsx": "^2.1.1",
2225
"lucide-react": "^0.439.0",
@@ -29,7 +32,6 @@
2932
"sot": "file:",
3033
"tailwind-merge": "^2.5.2",
3134
"tailwindcss-animate": "^1.0.7",
32-
"three": "^0.168.0",
3335
"zod": "^3.23.8"
3436
},
3537
"devDependencies": {

src/app/app/_components/main-sidebar.tsx

+13-4
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,8 @@ import {
1212
SidebarNavMain,
1313
} from '@/components/dashboard/sidebar'
1414
import { usePathname } from 'next/navigation'
15-
import { GearIcon, MixerVerticalIcon } from '@radix-ui/react-icons'
15+
import { CarIcon } from "lucide-react"
16+
import { GearIcon, MixerVerticalIcon, DashboardIcon } from '@radix-ui/react-icons'
1617

1718
import { Session } from 'next-auth'
1819
import { UserDropdown } from './user-dropdown'
@@ -37,11 +38,19 @@ export function MainSidebar({ user }: MainSidebarProps) {
3738
<SidebarNav>
3839
<SidebarNavMain>
3940
<SidebarNavLink href="/app" active={isActive('/app')}>
40-
<GearIcon className="w-3 h-3 mr-3" />
41-
OS
41+
<GearIcon className="w-4 h-4 mr-3" />
42+
Ordens de Serviço
43+
</SidebarNavLink>
44+
<SidebarNavLink href="/app/vehicles">
45+
<CarIcon className="w-4 h-4 mr-3" />
46+
Veículos
47+
</SidebarNavLink>
48+
<SidebarNavLink href="/app/projects">
49+
<DashboardIcon className="w-4 h-4 mr-3" />
50+
Projetos
4251
</SidebarNavLink>
4352
<SidebarNavLink href="/app/settings">
44-
<MixerVerticalIcon className="w-3 h-3 mr-3" />
53+
<MixerVerticalIcon className="w-4 h-4 mr-3" />
4554
Configurações
4655
</SidebarNavLink>
4756
</SidebarNavMain>

src/app/app/_components/user-dropdown.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ export function UserDropdown({ user }: UserdropdwonProps) {
3636
/>
3737
<AvatarFallback>U</AvatarFallback>
3838
</Avatar>
39-
<div className="flex flex-col space-y-1 flex-1 text-left">
39+
<div className="flex flex-col space-y-1 max-w-max truncate flex-1 text-left">
4040
{user.name && (
4141
<p className="text-sm font-medium leading-none">{user.name}</p>
4242
)}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,289 @@
1+
"use client"
2+
3+
import * as React from "react"
4+
import {
5+
CaretSortIcon,
6+
ChevronDownIcon,
7+
DotsHorizontalIcon,
8+
} from "@radix-ui/react-icons"
9+
import {
10+
ColumnDef,
11+
ColumnFiltersState,
12+
SortingState,
13+
VisibilityState,
14+
flexRender,
15+
getCoreRowModel,
16+
getFilteredRowModel,
17+
getPaginationRowModel,
18+
getSortedRowModel,
19+
useReactTable,
20+
} from "@tanstack/react-table"
21+
22+
import { Button } from "@/components/ui/button"
23+
import {
24+
DropdownMenu,
25+
DropdownMenuCheckboxItem,
26+
DropdownMenuContent,
27+
DropdownMenuItem,
28+
DropdownMenuLabel,
29+
DropdownMenuSeparator,
30+
DropdownMenuTrigger,
31+
} from "@/components/ui/dropdown-menu"
32+
import { Input } from "@/components/ui/input"
33+
import {
34+
Table,
35+
TableBody,
36+
TableCell,
37+
TableHead,
38+
TableHeader,
39+
TableRow,
40+
} from "@/components/ui/table"
41+
42+
type Vehicle = {
43+
id: number;
44+
automaker: string;
45+
project: string;
46+
responsible: string;
47+
model: string;
48+
color: string;
49+
chassis: string;
50+
fleet: string | null;
51+
comments: string | null;
52+
image: string | null;
53+
nf_number: string;
54+
nf_emission_date: string;
55+
loan_expiration_date: string;
56+
};
57+
export const columns: ColumnDef<Vehicle>[] = [
58+
{
59+
accessorKey: "automaker",
60+
header: "Montadora",
61+
cell: ({ row }) => {
62+
const { automaker } = row.original
63+
64+
return <div>{automaker}</div>
65+
}
66+
},
67+
{
68+
accessorKey: "model",
69+
header: ({ column }) => {
70+
return (
71+
<Button
72+
variant="link"
73+
onClick={() => column.toggleSorting(column.getIsSorted() === "asc")}
74+
>
75+
Modelo
76+
<CaretSortIcon className="ml-2 h-4 w-4" />
77+
</Button>
78+
)
79+
},
80+
cell: ({ row }) => <div>{row.getValue("model")}</div>,
81+
},
82+
{
83+
accessorKey: "chassis",
84+
header: () => <div className="text-right">Chassi</div>,
85+
cell: ({ row }) => {
86+
87+
88+
return <div className="text-right font-medium">{row.original.chassis}</div>
89+
},
90+
},
91+
{
92+
id: "actions",
93+
enableHiding: false,
94+
cell: ({ row }) => {
95+
const vehicle = row.original
96+
97+
// const handleDeleteTodo = async (todo: Todo) => {
98+
// await deleteTodo({
99+
// id: todo.id
100+
// })
101+
// toast({
102+
// title: "Deletion Successful",
103+
// description: "The todo item has been successfully deleted."
104+
// })
105+
// }
106+
// const handleToggleDone = async (todo: Todo) => {
107+
// const doneAt = todo.doneAt ? null : new Date()
108+
// await upsertTodo({ id: todo.id, doneAt })
109+
// toast({
110+
// title: "Update Successful",
111+
// description: "The todo item has been successfully updated."
112+
// })
113+
// }
114+
return (
115+
<DropdownMenu>
116+
<DropdownMenuTrigger asChild>
117+
<Button variant="link" className="h-8 w-8 p-0">
118+
<span className="sr-only">Open menu</span>
119+
<DotsHorizontalIcon className="h-4 w-4" />
120+
</Button>
121+
</DropdownMenuTrigger>
122+
<DropdownMenuContent align="end">
123+
<DropdownMenuLabel>Actions</DropdownMenuLabel>
124+
<DropdownMenuItem
125+
onClick={() => navigator.clipboard.writeText(vehicle.id.toString())}
126+
>
127+
Copiar ID
128+
</DropdownMenuItem>
129+
<DropdownMenuSeparator />
130+
<DropdownMenuItem
131+
// onClick={() => handleDeleteVehicle(vehicle)}
132+
>
133+
Apagar
134+
</DropdownMenuItem>
135+
</DropdownMenuContent>
136+
</DropdownMenu>
137+
)
138+
},
139+
},
140+
]
141+
142+
type VehicleDataTable = {
143+
data: Vehicle[]
144+
}
145+
146+
export function VehicleDataTable({ data }: VehicleDataTable) {
147+
148+
const [sorting, setSorting] = React.useState<SortingState>([])
149+
const [columnFilters, setColumnFilters] = React.useState<ColumnFiltersState>(
150+
[]
151+
)
152+
const [columnVisibility, setColumnVisibility] =
153+
React.useState<VisibilityState>({})
154+
const [rowSelection, setRowSelection] = React.useState({})
155+
156+
const table = useReactTable({
157+
data,
158+
columns,
159+
onSortingChange: setSorting,
160+
onColumnFiltersChange: setColumnFilters,
161+
getCoreRowModel: getCoreRowModel(),
162+
getPaginationRowModel: getPaginationRowModel(),
163+
getSortedRowModel: getSortedRowModel(),
164+
getFilteredRowModel: getFilteredRowModel(),
165+
onColumnVisibilityChange: setColumnVisibility,
166+
onRowSelectionChange: setRowSelection,
167+
state: {
168+
sorting,
169+
columnFilters,
170+
columnVisibility,
171+
rowSelection,
172+
},
173+
})
174+
175+
return (
176+
<div className="w-full">
177+
<div className="flex items-center py-4">
178+
<Input
179+
placeholder="Filtrar montadora..."
180+
value={(table.getColumn("automaker")?.getFilterValue() as string) ?? ""}
181+
onChange={(event) =>
182+
table.getColumn("automaker")?.setFilterValue(event.target.value)
183+
}
184+
className="max-w-sm"
185+
/>
186+
<DropdownMenu>
187+
<DropdownMenuTrigger asChild>
188+
<Button variant="outline" className="ml-auto">
189+
Columns <ChevronDownIcon className="ml-2 h-4 w-4" />
190+
</Button>
191+
</DropdownMenuTrigger>
192+
<DropdownMenuContent align="end">
193+
{table
194+
.getAllColumns()
195+
.filter((column) => column.getCanHide())
196+
.map((column) => {
197+
return (
198+
<DropdownMenuCheckboxItem
199+
key={column.id}
200+
className="capitalize"
201+
checked={column.getIsVisible()}
202+
onCheckedChange={(value) =>
203+
column.toggleVisibility(!!value)
204+
}
205+
>
206+
{column.id}
207+
</DropdownMenuCheckboxItem>
208+
)
209+
})}
210+
</DropdownMenuContent>
211+
</DropdownMenu>
212+
</div>
213+
<div className="rounded-md border">
214+
<Table>
215+
<TableHeader>
216+
{table.getHeaderGroups().map((headerGroup) => (
217+
<TableRow key={headerGroup.id}>
218+
{headerGroup.headers.map((header) => {
219+
return (
220+
<TableHead key={header.id}>
221+
{header.isPlaceholder
222+
? null
223+
: flexRender(
224+
header.column.columnDef.header,
225+
header.getContext()
226+
)}
227+
</TableHead>
228+
)
229+
})}
230+
</TableRow>
231+
))}
232+
</TableHeader>
233+
<TableBody>
234+
{table.getRowModel().rows?.length ? (
235+
table.getRowModel().rows.map((row) => (
236+
<TableRow
237+
key={row.id}
238+
data-state={row.getIsSelected() && "selected"}
239+
>
240+
{row.getVisibleCells().map((cell) => (
241+
<TableCell key={cell.id}>
242+
{flexRender(
243+
cell.column.columnDef.cell,
244+
cell.getContext()
245+
)}
246+
</TableCell>
247+
))}
248+
</TableRow>
249+
))
250+
) : (
251+
<TableRow>
252+
<TableCell
253+
colSpan={columns.length}
254+
className="h-24 text-center"
255+
>
256+
No results.
257+
</TableCell>
258+
</TableRow>
259+
)}
260+
</TableBody>
261+
</Table>
262+
</div>
263+
<div className="flex items-center justify-end space-x-2 py-4">
264+
<div className="flex-1 text-sm text-muted-foreground">
265+
{table.getFilteredSelectedRowModel().rows.length} of{" "}
266+
{table.getFilteredRowModel().rows.length} row(s) selected.
267+
</div>
268+
<div className="space-x-2">
269+
<Button
270+
variant="outline"
271+
size="sm"
272+
onClick={() => table.previousPage()}
273+
disabled={!table.getCanPreviousPage()}
274+
>
275+
Previous
276+
</Button>
277+
<Button
278+
variant="outline"
279+
size="sm"
280+
onClick={() => table.nextPage()}
281+
disabled={!table.getCanNextPage()}
282+
>
283+
Next
284+
</Button>
285+
</div>
286+
</div>
287+
</div>
288+
)
289+
}

0 commit comments

Comments
 (0)