1
+ import { Button } from 'components/ui/button' ;
2
+ import { Dialog , DialogContent , DialogFooter , DialogHeader , DialogTitle } from 'components/ui/dialog' ;
3
+ import { Input } from 'components/ui/input' ;
4
+ import { Label } from 'components/ui/label' ;
5
+ import { Select , SelectContent , SelectItem , SelectTrigger , SelectValue } from 'components/ui/select' ;
6
+ import { useToast } from 'components/ui/use-toast' ;
7
+ import React , { useEffect , useState } from 'react' ;
8
+ import { createCategory , getCategories } from 'services/api' ;
9
+ import { Category } from 'services/types' ;
10
+
11
+ interface CategoryManagerProps {
12
+ onCategorySelect : ( categoryId : number ) => void ;
13
+ selectedCategoryId ?: number ;
14
+ }
15
+
16
+ const COLORS = [ '#FF0000' , '#00FF00' , '#0000FF' , '#FFFF00' , '#FF00FF' , '#00FFFF' , '#FFA500' , '#800080' , '#008000' , '#FFC0CB' ] ;
17
+
18
+ const CategoryManager : React . FC < CategoryManagerProps > = ( { onCategorySelect, selectedCategoryId } ) => {
19
+ const [ categories , setCategories ] = useState < Category [ ] > ( [ ] ) ;
20
+ const [ isDialogOpen , setIsDialogOpen ] = useState ( false ) ;
21
+ const [ newCategoryName , setNewCategoryName ] = useState ( '' ) ;
22
+ const [ newCategoryColor , setNewCategoryColor ] = useState ( COLORS [ 0 ] ) ;
23
+ const { toast } = useToast ( ) ;
24
+
25
+ useEffect ( ( ) => {
26
+ fetchCategories ( ) ;
27
+ } , [ ] ) ;
28
+
29
+ const fetchCategories = async ( ) => {
30
+ try {
31
+ const fetchedCategories = await getCategories ( ) ;
32
+ setCategories ( fetchedCategories ) ;
33
+ } catch ( error ) {
34
+ console . error ( 'Error fetching categories:' , error ) ;
35
+ toast ( {
36
+ title : 'Error' ,
37
+ description : 'Failed to fetch categories' ,
38
+ variant : 'destructive' ,
39
+ } ) ;
40
+ }
41
+ } ;
42
+
43
+ const handleCreateCategory = async ( ) => {
44
+ try {
45
+ const newCategory = await createCategory ( {
46
+ name : newCategoryName ,
47
+ color : newCategoryColor ,
48
+ } ) ;
49
+ await fetchCategories ( ) ;
50
+ setIsDialogOpen ( false ) ;
51
+ setNewCategoryName ( '' ) ;
52
+ setNewCategoryColor ( COLORS [ 0 ] ) ;
53
+ toast ( {
54
+ title : 'Success' ,
55
+ description : 'Category created successfully' ,
56
+ } ) ;
57
+ onCategorySelect ( newCategory . id ) ;
58
+ } catch ( error ) {
59
+ console . error ( 'Error creating category:' , error ) ;
60
+ toast ( {
61
+ title : 'Error' ,
62
+ description : 'Failed to create category' ,
63
+ variant : 'destructive' ,
64
+ } ) ;
65
+ }
66
+ } ;
67
+
68
+ const handleSelectChange = ( value : string ) => {
69
+ if ( value === 'create' ) {
70
+ setIsDialogOpen ( true ) ;
71
+ } else {
72
+ onCategorySelect ( Number ( value ) ) ;
73
+ }
74
+ } ;
75
+
76
+ return (
77
+ < >
78
+ < Select onValueChange = { handleSelectChange } value = { selectedCategoryId ?. toString ( ) } >
79
+ < SelectTrigger >
80
+ < SelectValue placeholder = "Select a category" />
81
+ </ SelectTrigger >
82
+ < SelectContent >
83
+ { categories . map ( ( category ) => (
84
+ < SelectItem key = { category . id } value = { category . id . toString ( ) } >
85
+ < span style = { { color : category . color } } > { category . name } </ span >
86
+ </ SelectItem >
87
+ ) ) }
88
+ < SelectItem value = "create" > Create Category</ SelectItem >
89
+ </ SelectContent >
90
+ </ Select >
91
+
92
+ < Dialog open = { isDialogOpen } onOpenChange = { setIsDialogOpen } >
93
+ < DialogContent >
94
+ < DialogHeader >
95
+ < DialogTitle > Create Category</ DialogTitle >
96
+ </ DialogHeader >
97
+ < div className = "space-y-4" >
98
+ < div >
99
+ < Label htmlFor = "categoryName" > Name</ Label >
100
+ < Input
101
+ id = "categoryName"
102
+ value = { newCategoryName }
103
+ onChange = { ( e ) => setNewCategoryName ( e . target . value ) }
104
+ />
105
+ </ div >
106
+ < div >
107
+ < Label htmlFor = "categoryColor" > Color</ Label >
108
+ < Select onValueChange = { setNewCategoryColor } value = { newCategoryColor } >
109
+ < SelectTrigger >
110
+ < SelectValue placeholder = "Select a color" />
111
+ </ SelectTrigger >
112
+ < SelectContent >
113
+ { COLORS . map ( ( color ) => (
114
+ < SelectItem key = { color } value = { color } >
115
+ < div className = "flex items-center" >
116
+ < div className = "w-4 h-4 mr-2 rounded-full" style = { { backgroundColor : color } } > </ div >
117
+ { color }
118
+ </ div >
119
+ </ SelectItem >
120
+ ) ) }
121
+ </ SelectContent >
122
+ </ Select >
123
+ </ div >
124
+ </ div >
125
+ < DialogFooter >
126
+ < Button onClick = { ( ) => setIsDialogOpen ( false ) } > Cancel</ Button >
127
+ < Button onClick = { handleCreateCategory } > Create</ Button >
128
+ </ DialogFooter >
129
+ </ DialogContent >
130
+ </ Dialog >
131
+ </ >
132
+ ) ;
133
+ } ;
134
+
135
+ export default CategoryManager ;
0 commit comments