import React, { useState, useEffect, useRef } from "react"; import MDEditor, { commands } from '@uiw/react-md-editor'; import { Input } from './ui/input'; import { Label } from './ui/label'; import { Select, SelectTrigger, SelectValue, SelectContent, SelectItem } from './ui/select'; import { Button } from './ui/button'; import { Separator } from './ui/separator'; import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogTrigger } from './ui/dialog'; import { CustomTabs } from './ui/tabs'; import Loader from "./ui/loader"; const TOPIC_API_URL = 'https://host-api.cs1.hz.siliconpin.com/v1/topics/'; const MINIO_UPLOAD_URL = 'https://your-minio-api-endpoint/upload'; const urlParams = new URLSearchParams(window.location.search); const slug = urlParams.get('slug'); // console.log('find slug from url', slug); export default function EditTopic (){ // const { slug } = useParams(); // const navigate = useNavigate(); // Form state const [formData, setFormData] = useState({ status: 'draft', category: '', slug: '', title: '', content: '', imageUrl: '' }); // UI state const [isSubmitting, setIsSubmitting] = useState(false); const [isLoading, setIsLoading] = useState(true); const [error, setError] = useState(null); const [success, setSuccess] = useState(false); const [imageFile, setImageFile] = useState(null); const [uploadProgress, setUploadProgress] = useState(0); const [imageDialogOpen, setImageDialogOpen] = useState(false); const [imageUrlInput, setImageUrlInput] = useState(''); const [imageUploadFile, setImageUploadFile] = useState(null); const [imageUploadPreview, setImageUploadPreview] = useState(''); const [editorMode, setEditorMode] = useState('edit'); // Fetch topic data on component mount useEffect(() => { const fetchTopic = async () => { try { const response = await fetch(`${TOPIC_API_URL}?query=get-single-topic&slug=${slug}`, { credentials: 'include' }); if (!response.ok) { throw new Error('Failed to fetch topic'); } const data = await response.json(); const topic = data.data[0] console.log('Single topic data', data) setFormData({ status: topic.status || 'draft', category: topic.category || '', title: topic.title || '', slug: topic.slug || '', content: topic.content || '', imageUrl: topic.imageUrl || '' }); } catch (err) { setError(err.message); } finally { setIsLoading(false); } }; fetchTopic(); }, [slug]); // Upload file to MinIO (same as NewTopic) const uploadToMinIO = async (file, onProgress) => { const formData = new FormData(); formData.append('file', file); formData.append('bucket', 'siliconpin-uploads'); formData.append('folder', 'topic-images'); try { const response = await fetch(MINIO_UPLOAD_URL, { method: 'POST', body: formData, credentials: 'include', }); if (!response.ok) { throw new Error('Upload failed'); } const data = await response.json(); return data.url; } catch (error) { console.error('Upload error:', error); throw error; } }; // Generate slug from title (same as NewTopic) // useEffect(() => { // if (formData.title) { // const newSlug = formData.title // .toLowerCase() // .replace(/[^\w\s]/g, '') // .replace(/\s+/g, '-'); // setFormData(prev => ({ ...prev, slug: newSlug })); // } // }, [formData.title]); // Custom image command for MDEditor (same as NewTopic) const customImageCommand = { name: 'image', keyCommand: 'image', buttonProps: { 'aria-label': 'Insert image' }, icon: ( ), execute: () => { setImageDialogOpen(true); }, }; // Get all commands (same as NewTopic) const allCommands = commands.getCommands().map(cmd => { if (cmd.name === 'image') { return customImageCommand; } return cmd; }); // Handle image URL insertion (same as NewTopic) const handleInsertImageUrl = () => { if (imageUrlInput) { const imgMarkdown = `![Image](${imageUrlInput})`; const textarea = document.querySelector('.w-md-editor-text-input'); if (textarea) { const startPos = textarea.selectionStart; const endPos = textarea.selectionEnd; const currentValue = formData.content; const newValue = currentValue.substring(0, startPos) + imgMarkdown + currentValue.substring(endPos); setFormData(prev => ({ ...prev, content: newValue })); } setImageDialogOpen(false); setImageUrlInput(''); } }; // Handle image file selection (same as NewTopic) const handleImageFileSelect = (e) => { const file = e.target.files[0]; if (file) { setImageUploadFile(file); const reader = new FileReader(); reader.onload = () => { setImageUploadPreview(reader.result); }; reader.readAsDataURL(file); } }; // Upload image file (same as NewTopic) const handleImageUpload = async () => { if (!imageUploadFile) return; try { setIsSubmitting(true); setUploadProgress(0); const uploadedUrl = await uploadToMinIO(imageUploadFile, (progress) => { setUploadProgress(progress); }); const imgMarkdown = `![Image](${uploadedUrl})`; const textarea = document.querySelector('.w-md-editor-text-input'); if (textarea) { const startPos = textarea.selectionStart; const endPos = textarea.selectionEnd; const currentValue = formData.content; const newValue = currentValue.substring(0, startPos) + imgMarkdown + currentValue.substring(endPos); setFormData(prev => ({ ...prev, content: newValue })); } setImageDialogOpen(false); setImageUploadFile(null); setImageUploadPreview(''); } catch (error) { setError('Failed to upload image: ' + error.message); } finally { setIsSubmitting(false); } }; // Form submission for EDIT const handleSubmit = async (e) => { e.preventDefault(); setIsSubmitting(true); setError(null); setUploadProgress(0); try { // Upload new featured image if selected let imageUrl = formData.imageUrl; if (imageFile) { imageUrl = await uploadToMinIO(imageFile, (progress) => { setUploadProgress(progress); }); } // Prepare payload const payload = { ...formData, imageUrl }; // Submit to API (PUT request for update) const response = await fetch(`${TOPIC_API_URL}?query=update-topic&slug=${formData.slug}`, { method: 'POST', headers: { 'Content-Type': 'application/json', }, credentials: 'include', body: JSON.stringify(payload) }); if (!response.ok) { const errorData = await response.json(); throw new Error(errorData.message || 'Failed to update topic'); } setSuccess(true); setTimeout(() => { navigate(`/topic/${formData.slug}`); }, 1500); } catch (err) { setError(err.message); } finally { setIsSubmitting(false); setUploadProgress(0); } }; // Handle form field changes (same as NewTopic) const handleChange = (e) => { const { name, value } = e.target; setFormData(prev => ({ ...prev, [name]: value })); }; // Handle featured image file selection (same as NewTopic) const handleFileChange = (e) => { const file = e.target.files[0]; if (file) { setImageFile(file); const reader = new FileReader(); reader.onload = () => { setFormData(prev => ({ ...prev, imageUrl: reader.result })); }; reader.readAsDataURL(file); } }; // Loading state if (isLoading) { return ( ); } // Success state if (success) { return (

Topic has been successfully updated!

You are being automatically redirected to the topic page...

); } return (

Edit Topic

{formData.title}

{/* Title and Slug */}

This will be used in the topic URL

{/* Content Editor */}
setFormData(prev => ({ ...prev, content: value || '' }))} height={400} preview={editorMode} commands={allCommands} />
{/* Image Upload Dialog (same as NewTopic) */} Add Image setImageUrlInput(e.target.value)} />
) }, { label: "Upload", value: "upload", content: (
{imageUploadPreview && (
Preview
)} {uploadProgress > 0 && uploadProgress < 100 && (
)}
) } ]} /> {/* Featured Image Upload */}
{uploadProgress > 0 && uploadProgress < 100 && (
)} {formData.imageUrl && (
Preview

Current Image

)}
{/* Form Actions */}
{error && (
Error: {error}
)}
); };