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'; const PUBLIC_MINIO_UPLOAD_URL = import.meta.env.PUBLIC_MINIO_UPLOAD_URL; const PUBLIC_TOPIC_API_URL = import.meta.env.PUBLIC_TOPIC_API_URL; const NewTopic = () => { // Form state const [formData, setFormData] = useState({ status: 'draft', category: '', title: '', content: '', imageUrl: '' }); // UI state const [isSubmitting, setIsSubmitting] = useState(false); 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'); // Upload file to MinIO const uploadToMinIO = async (file, onProgress) => { const formData = new FormData(); formData.append('file', file); formData.append('api_key', 'wweifwehfwfhwhtuyegbvijvbfvegfreyf'); try { const response = await fetch(PUBLIC_MINIO_UPLOAD_URL, { method: 'POST', body: formData, credentials: 'include' }); if (!response.ok) { throw new Error('Upload failed'); } const data = await response.json(); console.log(data.publicUrl) return data.publicUrl; } catch (error) { console.error('Upload error:', error); throw error; } }; // // Generate slug from title // useEffect(() => { // if (formData.title) { // const slug = formData.title // .toLowerCase() // .replace(/[^\w\s]/g, '') // .replace(/\s+/g, '-'); // setFormData(prev => ({ ...prev, slug })); // } // }, [formData.title]); const customImageCommand = { name: 'image', keyCommand: 'image', buttonProps: { 'aria-label': 'Insert image' }, icon: ( ), execute: () => { setImageDialogOpen(true); }, }; // Get all default commands and replace the image command const allCommands = commands.getCommands().map(cmd => { if (cmd.name === 'image') { return customImageCommand; } return cmd; }); // Handle image URL insertion 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 const handleImageFileSelect = (e) => { const file = e.target.files[0]; if (file) { setImageUploadFile(file); // Create preview const reader = new FileReader(); reader.onload = () => { setImageUploadPreview(reader.result); }; reader.readAsDataURL(file); } }; // Upload image file to MinIO and insert into editor const handleImageUpload = async () => { if (!imageUploadFile) return; console.log(imageUploadFile) try { setIsSubmitting(true); setUploadProgress(0); const uploadedUrl = await uploadToMinIO(imageUploadFile, (progress) => { setUploadProgress(progress); }); // Insert markdown for the uploaded image 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 const handleSubmit = async (e) => { e.preventDefault(); setIsSubmitting(true); setError(null); setUploadProgress(0); try { // Upload featured image if selected let imageUrl = formData.imageUrl; if (imageFile) { imageUrl = await uploadToMinIO(imageFile, (progress) => { console.log('imageUrl', imageUrl) setUploadProgress(progress); }); } // let finalImageUrl = imageUrl.publicUrl; // Prepare payload const payload = { ...formData, imageUrl }; // Submit to API const response = await fetch(`${PUBLIC_TOPIC_API_URL}?query=create-new-topic`, { 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 create topic'); } setSuccess(true); } catch (err) { setError(err.message); } finally { setIsSubmitting(false); setUploadProgress(0); } }; // Handle form field changes const handleChange = (e) => { const { name, value } = e.target; setFormData(prev => ({ ...prev, [name]: value })); }; // Handle featured image file selection const handleFileChange = (e) => { const file = e.target.files[0]; if (file) { setImageFile(file); // Preview image const reader = new FileReader(); reader.onload = () => { setFormData(prev => ({ ...prev, imageUrl: reader.result })); }; reader.readAsDataURL(file); } }; // Reset form const resetForm = () => { setFormData({ status: 'draft', category: '', title: '', slug: '', content: '', imageUrl: '' }); setImageFile(null); setSuccess(false); setError(null); }; // Success state if (success) { return (

Topic created successfully!

); } return (

Create New Topic

Start a new discussion in the SiliconPin community

{/* Title and Slug */}
{/*

This will be used in the topic URL

*/} {/* Content Editor with Preview Toggle */}
setFormData(prev => ({ ...prev, content: value || '' }))} height={400} preview={editorMode} commands={allCommands} />
{/* Image Upload Dialog */} Insert 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
)}
{/* Form Actions */}
{error && (
Error: {error}
)}
); }; export default NewTopic;