EditNotebookDialog.tsx 4.6 KB

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