CreateNotebookDialog.tsx 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. import React, { useState } from 'react'
  2. import { X, Plus } from 'lucide-react'
  3. import { CreateGroupData } from '../types'
  4. import { useLanguage } from '../contexts/LanguageContext'
  5. import { useToast } from '../contexts/ToastContext'
  6. interface CreateNotebookDialogProps {
  7. isOpen: boolean
  8. onClose: () => void
  9. onCreate: (data: CreateGroupData) => Promise<void>
  10. }
  11. export const CreateNotebookDialog: React.FC<CreateNotebookDialogProps> = ({
  12. isOpen,
  13. onClose,
  14. onCreate,
  15. }) => {
  16. const { t } = useLanguage();
  17. const { showError } = useToast();
  18. const [name, setName] = useState('')
  19. const [description, setDescription] = useState('')
  20. const [isSubmitting, setIsSubmitting] = useState(false)
  21. if (!isOpen) return null
  22. const handleSubmit = async (e: React.FormEvent) => {
  23. e.preventDefault()
  24. if (!name.trim()) return
  25. try {
  26. setIsSubmitting(true)
  27. await onCreate({
  28. name: name.trim(),
  29. description: description.trim(),
  30. color: '#3b82f6', // Default color
  31. })
  32. // Reset form
  33. setName('')
  34. setDescription('')
  35. onClose()
  36. } catch (error) {
  37. console.error('Failed to create notebook:', error)
  38. showError(t('createFailedRetry'))
  39. } finally {
  40. setIsSubmitting(false)
  41. }
  42. }
  43. return (
  44. <div className="fixed inset-0 z-50 flex items-center justify-center bg-black/50 backdrop-blur-sm">
  45. <div className="bg-white rounded-xl shadow-xl w-full max-w-lg overflow-hidden animate-in fade-in zoom-in duration-200">
  46. <div className="flex justify-between items-center px-6 py-4 border-b">
  47. <h2 className="text-lg font-semibold text-slate-800">{t('createNotebookTitle')}</h2>
  48. <button
  49. onClick={onClose}
  50. className="text-slate-400 hover:text-slate-600 transition-colors"
  51. >
  52. <X size={20} />
  53. </button>
  54. </div>
  55. <form onSubmit={handleSubmit} className="p-6 space-y-4">
  56. <div>
  57. <label className="block text-sm font-medium text-slate-700 mb-1">
  58. {t('name')} <span className="text-red-500">*</span>
  59. </label>
  60. <input
  61. type="text"
  62. value={name}
  63. onChange={(e) => setName(e.target.value)}
  64. className="w-full px-3 py-2 border rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500"
  65. placeholder={t('namePlaceholder')}
  66. required
  67. autoFocus
  68. />
  69. </div>
  70. <div>
  71. <label className="block text-sm font-medium text-slate-700 mb-1">
  72. {t('shortDescription')}
  73. </label>
  74. <input
  75. type="text"
  76. value={description}
  77. onChange={(e) => setDescription(e.target.value)}
  78. className="w-full px-3 py-2 border rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500"
  79. placeholder={t('descPlaceholder')}
  80. />
  81. </div>
  82. <div className="flex justify-end pt-2">
  83. <button
  84. type="button"
  85. onClick={onClose}
  86. className="px-4 py-2 text-slate-600 hover:bg-slate-100 rounded-lg mr-2 transition-colors"
  87. >
  88. {t('cancel')}
  89. </button>
  90. <button
  91. type="submit"
  92. disabled={isSubmitting || !name.trim()}
  93. className="flex items-center gap-2 px-4 py-2 bg-blue-600 text-white rounded-lg hover:bg-blue-700 transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
  94. >
  95. <Plus size={18} />
  96. {isSubmitting ? t('creating') : t('createNow')}
  97. </button>
  98. </div>
  99. </form>
  100. </div>
  101. </div>
  102. )
  103. }