Files
home_hub/frontend/src/pages/NotesPage.tsx
T
gilles ec87bc091d feat(sse): sync temps réel multi-appareils via Server-Sent Events v0.5.8
- Broadcaster asyncio.Queue avec keepalive 25s (prévient timeout proxy)
- Endpoint GET /api/events/stream (StreamingResponse text/event-stream)
- Broadcast notes_changed / todos_changed / shopping_changed sur toutes mutations
- Hook useServerEvents: EventSource avec reconnexion automatique (3s)
- Pages Notes, Todos, Shopping abonnées aux événements SSE
- nginx: location SSE dédiée (proxy_buffering off, timeout 24h)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-25 20:12:02 +02:00

502 lines
23 KiB
TypeScript

import { useState, useEffect, useCallback, useRef } from 'react'
import { useServerEvents } from '../hooks/useServerEvents'
import type { Note, NoteFilters } from '../api/notes'
import { fetchNotes, createNote, updateNote, deleteNote, addAttachment, deleteAttachment } from '../api/notes'
import Modal from '../components/Modal'
import NoteForm from '../components/notes/NoteForm'
import { useActionButton } from '../contexts/ActionButtonContext'
const noSelect: React.CSSProperties = { userSelect: 'none' }
const inputStyle: React.CSSProperties = {
background: 'var(--bg-3)',
border: '1px solid var(--bg-5)',
borderRadius: 8,
padding: '6px 10px',
color: 'var(--ink-1)',
fontFamily: 'var(--font-ui)',
fontSize: 13,
}
const actionBtnStyle: React.CSSProperties = {
padding: '5px 10px',
borderRadius: 6,
border: '1px solid var(--bg-5)',
background: 'var(--bg-4)',
color: 'var(--ink-3)',
cursor: 'pointer',
fontSize: 13,
minHeight: 32,
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
}
function formatDate(iso: string) {
return new Date(iso).toLocaleDateString('fr-FR', { day: '2-digit', month: 'short', year: 'numeric' })
}
// Formate le texte inline : **gras**, *italique*, `code`
function inlineFmt(text: string): React.ReactNode {
const parts = text.split(/(\*\*[^*]+\*\*|\*[^*]+\*|`[^`]+`)/)
return (
<>
{parts.map((p, i) => {
if (p.startsWith('**') && p.endsWith('**')) return <strong key={i}>{p.slice(2, -2)}</strong>
if (p.startsWith('*') && p.endsWith('*')) return <em key={i} style={{ color: 'var(--ink-3)' }}>{p.slice(1, -1)}</em>
if (p.startsWith('`') && p.endsWith('`')) return <code key={i} style={{ background: 'var(--bg-4)', borderRadius: 3, padding: '0 4px', fontFamily: 'var(--font-mono)', fontSize: '0.88em' }}>{p.slice(1, -1)}</code>
return p || null
})}
</>
)
}
// Renderer pseudo-markdown ligne par ligne
function renderMarkdown(text: string): React.ReactNode {
const lines = text.split('\n')
const nodes: React.ReactNode[] = []
let i = 0
while (i < lines.length) {
const line = lines[i]
if (line.startsWith('```')) {
const lang = line.slice(3).trim()
const code: string[] = []
i++
while (i < lines.length && !lines[i].startsWith('```')) { code.push(lines[i]); i++ }
nodes.push(
<pre key={i} style={{ background: 'var(--bg-4)', border: '1px solid var(--bg-5)', borderRadius: 6, padding: '8px 12px', margin: '6px 0', fontFamily: 'var(--font-mono)', fontSize: 12, overflowX: 'auto', color: 'var(--ink-1)', whiteSpace: 'pre' }}>
{lang && <div style={{ color: 'var(--ink-4)', fontSize: 10, marginBottom: 4, textTransform: 'uppercase', letterSpacing: 0.5 }}>{lang}</div>}
{code.join('\n')}
</pre>
)
} else if (line.startsWith('# ')) {
nodes.push(<div key={i} style={{ fontWeight: 700, fontSize: 15, color: 'var(--accent)', marginTop: 10, marginBottom: 2, fontFamily: 'var(--font-ui)' }}>{inlineFmt(line.slice(2))}</div>)
} else if (line.startsWith('## ')) {
nodes.push(<div key={i} style={{ fontWeight: 700, fontSize: 14, color: 'var(--ink-1)', marginTop: 8, marginBottom: 2, paddingBottom: 3, borderBottom: '1px solid var(--bg-5)', fontFamily: 'var(--font-ui)' }}>{inlineFmt(line.slice(3))}</div>)
} else if (line.startsWith('### ')) {
nodes.push(<div key={i} style={{ fontWeight: 600, fontSize: 13, color: 'var(--ink-1)', marginTop: 6, marginBottom: 1, fontFamily: 'var(--font-ui)' }}>{inlineFmt(line.slice(4))}</div>)
} else if (line.startsWith('- ') || line.startsWith('* ')) {
nodes.push(
<div key={i} style={{ display: 'flex', gap: 6, color: 'var(--ink-2)', lineHeight: 1.6, fontFamily: 'var(--font-ui)', fontSize: 13 }}>
<span style={{ color: 'var(--accent)', flexShrink: 0 }}></span>
<span>{inlineFmt(line.slice(2))}</span>
</div>
)
} else if (/^\d+\.\s/.test(line)) {
const m = line.match(/^(\d+)\.\s(.*)/)
nodes.push(
<div key={i} style={{ display: 'flex', gap: 6, color: 'var(--ink-2)', lineHeight: 1.6, fontFamily: 'var(--font-ui)', fontSize: 13 }}>
<span style={{ color: 'var(--accent)', flexShrink: 0, minWidth: 18, textAlign: 'right' }}>{m?.[1]}.</span>
<span>{inlineFmt(m?.[2] ?? '')}</span>
</div>
)
} else if (line.startsWith('> ')) {
nodes.push(
<div key={i} style={{ borderLeft: '3px solid var(--accent)', paddingLeft: 10, color: 'var(--ink-3)', fontStyle: 'italic', margin: '4px 0', lineHeight: 1.6, fontFamily: 'var(--font-ui)', fontSize: 13 }}>
{inlineFmt(line.slice(2))}
</div>
)
} else if (line === '---' || line === '***') {
nodes.push(<div key={i} style={{ borderBottom: '1px solid var(--bg-5)', margin: '8px 0' }} />)
} else if (line.trim() === '') {
nodes.push(<div key={i} style={{ height: 5 }} />)
} else {
nodes.push(
<div key={i} style={{ color: 'var(--ink-2)', lineHeight: 1.6, fontFamily: 'var(--font-ui)', fontSize: 13 }}>
{inlineFmt(line)}
</div>
)
}
i++
}
return <>{nodes}</>
}
type NoteState = 'semi' | 'expanded' | 'collapsed'
function NoteCard({ note, onEdit, onDelete, onAddPhoto, onAddAudio, onAddVideo, onDeleteAtt }: {
note: Note
onEdit: () => void
onDelete: () => void
onAddPhoto: (file: File) => void
onAddAudio: (file: File) => void
onAddVideo: (file: File) => void
onDeleteAtt: (attId: string) => void
}) {
const [state, setState] = useState<NoteState>('semi')
const photoRef = useRef<HTMLInputElement>(null)
const audioRef = useRef<HTMLInputElement>(null)
const videoRef = useRef<HTMLInputElement>(null)
const [recording, setRecording] = useState(false)
const recorderRef = useRef<MediaRecorder | null>(null)
const chunksRef = useRef<Blob[]>([])
const images = note.attachments.filter(a => a.file_type === 'image')
const audios = note.attachments.filter(a => a.file_type === 'audio')
const videos = note.attachments.filter(a => a.file_type === 'video')
function cycleState() {
setState(s => s === 'semi' ? 'expanded' : s === 'expanded' ? 'collapsed' : 'semi')
}
const stateIcon = state === 'semi' ? 'fa-chevron-down' : state === 'expanded' ? 'fa-minus' : 'fa-chevron-right'
const stateTitle = state === 'semi' ? 'Tout afficher' : state === 'expanded' ? 'Réduire' : 'Développer'
async function startRecord() {
try {
const stream = await navigator.mediaDevices.getUserMedia({ audio: true })
const mimeType = ['audio/webm', 'audio/mp4', 'audio/ogg'].find(t => MediaRecorder.isTypeSupported(t)) ?? ''
const recorder = mimeType ? new MediaRecorder(stream, { mimeType }) : new MediaRecorder(stream)
chunksRef.current = []
recorder.ondataavailable = e => { if (e.data.size > 0) chunksRef.current.push(e.data) }
recorder.onstop = () => {
stream.getTracks().forEach(t => t.stop())
const type = recorder.mimeType || mimeType || 'audio/webm'
const ext = type.includes('mp4') ? 'm4a' : 'webm'
const blob = new Blob(chunksRef.current, { type })
onAddAudio(new File([blob], `enregistrement.${ext}`, { type }))
}
recorder.start()
recorderRef.current = recorder
setRecording(true)
} catch { /* micro non disponible */ }
}
function stopRecord() {
recorderRef.current?.stop()
setRecording(false)
}
const toggleBtn = (
<button
onClick={cycleState}
title={stateTitle}
style={{ background: 'transparent', border: 'none', color: 'var(--ink-4)', cursor: 'pointer', padding: '2px 6px', borderRadius: 4, fontSize: 13, flexShrink: 0, ...noSelect }}
>
<i className={`fa-solid ${stateIcon}`} />
</button>
)
const metaLine = (
<div style={{ display: 'flex', gap: 6, flexWrap: 'wrap', alignItems: 'center', ...noSelect }}>
<span style={{ color: 'var(--ink-4)', fontSize: 11, fontFamily: 'var(--font-mono)' }}>{formatDate(note.created_at)}</span>
{note.category && <span style={{ background: 'var(--bg-4)', color: 'var(--info)', fontSize: 11, fontFamily: 'var(--font-ui)', borderRadius: 999, padding: '1px 7px' }}>{note.category}</span>}
{note.tags.map(t => <span key={t} style={{ background: 'var(--bg-5)', color: 'var(--ink-3)', fontSize: 11, fontFamily: 'var(--font-ui)', borderRadius: 999, padding: '1px 7px' }}>{t}</span>)}
{images.length > 0 && <i className="fa-solid fa-image" style={{ color: 'var(--ink-4)', fontSize: 11 }} title={`${images.length} photo(s)`} />}
{audios.length > 0 && <i className="fa-solid fa-microphone" style={{ color: 'var(--ink-4)', fontSize: 11 }} title={`${audios.length} audio(s)`} />}
{videos.length > 0 && <i className="fa-solid fa-video" style={{ color: 'var(--ink-4)', fontSize: 11 }} title={`${videos.length} vidéo(s)`} />}
{note.gps_lat != null && (
<i className="fa-solid fa-location-dot" style={{ color: 'var(--ok)', fontSize: 12 }} title={`${note.gps_lat.toFixed(4)}, ${note.gps_lon?.toFixed(4)}`} />
)}
</div>
)
const mediaSection = (
<>
{images.length > 0 && (
<div style={{ display: 'flex', gap: 6, flexWrap: 'wrap' }}>
{images.map(img => (
<div key={img.id} style={{ position: 'relative' }}>
<img src={`/media/${img.thumbnail_path ?? img.file_path}`} alt="" style={{ width: 80, height: 80, objectFit: 'cover', borderRadius: 6 }} />
<button onClick={() => onDeleteAtt(img.id)} style={{ position: 'absolute', top: 2, right: 2, background: 'rgba(0,0,0,0.65)', border: 'none', color: '#fff', borderRadius: '50%', width: 20, height: 20, cursor: 'pointer', fontSize: 10, display: 'flex', alignItems: 'center', justifyContent: 'center', padding: 0 }}></button>
</div>
))}
</div>
)}
{audios.length > 0 && (
<div style={{ display: 'flex', flexDirection: 'column', gap: 4 }}>
{audios.map(aud => (
<div key={aud.id} style={{ display: 'flex', alignItems: 'center', gap: 6, background: 'var(--bg-4)', borderRadius: 6, padding: '4px 8px', minWidth: 0 }}>
<audio
src={`/media/${aud.file_path}`}
controls
style={{ height: 32, flex: 1, minWidth: 0, width: '100%' }}
onLoadedMetadata={e => { (e.target as HTMLAudioElement).volume = 0.5 }}
/>
<button onClick={() => onDeleteAtt(aud.id)} style={{ background: 'transparent', border: 'none', color: 'var(--err)', cursor: 'pointer', fontSize: 14, flexShrink: 0 }}></button>
</div>
))}
</div>
)}
{videos.length > 0 && (
<div style={{ display: 'flex', flexDirection: 'column', gap: 6 }}>
{videos.map(vid => (
<div key={vid.id} style={{ position: 'relative' }}>
<video src={`/media/${vid.file_path}`} controls playsInline style={{ width: '100%', maxHeight: 220, borderRadius: 6, background: '#000', display: 'block' }} />
<button onClick={() => onDeleteAtt(vid.id)} style={{ position: 'absolute', top: 6, right: 6, background: 'rgba(0,0,0,0.65)', border: 'none', color: '#fff', borderRadius: '50%', width: 22, height: 22, cursor: 'pointer', fontSize: 11, display: 'flex', alignItems: 'center', justifyContent: 'center', padding: 0 }}></button>
</div>
))}
</div>
)}
</>
)
const actionButtons = (
<div style={{ display: 'flex', gap: 6 }}>
<input ref={photoRef} type="file" accept="image/*" style={{ display: 'none' }} onChange={e => { const f = e.target.files?.[0]; if (f) onAddPhoto(f); e.target.value = '' }} />
<button onClick={() => photoRef.current?.click()} title="Photo" style={actionBtnStyle}><i className="fa-solid fa-camera" /></button>
<input ref={audioRef} type="file" accept="audio/*" style={{ display: 'none' }} onChange={e => { const f = e.target.files?.[0]; if (f) onAddAudio(f); e.target.value = '' }} />
<button
onClick={recording ? stopRecord : startRecord}
title={recording ? "Arrêter" : "Enregistrer"}
style={{ ...actionBtnStyle, background: recording ? 'var(--err)' : actionBtnStyle.background, color: recording ? '#fff' : actionBtnStyle.color, border: recording ? 'none' : actionBtnStyle.border }}
><i className={`fa-solid fa-${recording ? 'stop' : 'microphone'}`} /></button>
<input ref={videoRef} type="file" accept="video/*" style={{ display: 'none' }} onChange={e => { const f = e.target.files?.[0]; if (f) onAddVideo(f); e.target.value = '' }} />
<button onClick={() => videoRef.current?.click()} title="Vidéo" style={actionBtnStyle}><i className="fa-solid fa-video" /></button>
<div style={{ flex: 1 }} />
<button onClick={onEdit} title="Éditer" style={{ ...actionBtnStyle, background: 'var(--bg-5)', border: 'none' }}><i className="fa-solid fa-pen" /></button>
<button onClick={onDelete} title="Supprimer" style={{ ...actionBtnStyle, background: 'transparent', color: 'var(--err)' }}><i className="fa-solid fa-xmark" /></button>
</div>
)
// ─── COLLAPSED ───────────────────────────────────────────────────────────────
if (state === 'collapsed') {
return (
<div className="glass" style={{ borderRadius: 10, padding: '8px 14px' }}>
<div style={{ display: 'flex', alignItems: 'center', gap: 8 }}>
<div style={{ flex: 1, minWidth: 0, ...noSelect }}>
<span style={{ color: 'var(--ink-2)', fontFamily: 'var(--font-ui)', fontSize: 13, fontWeight: note.title ? 600 : 400 }}>
{note.title || note.content.slice(0, 60).replace(/\n/g, ' ')}
</span>
<span style={{ color: 'var(--ink-4)', fontFamily: 'var(--font-mono)', fontSize: 11, marginLeft: 8 }}>
{formatDate(note.created_at)}
</span>
</div>
{toggleBtn}
</div>
</div>
)
}
// ─── SEMI (défaut) ───────────────────────────────────────────────────────────
if (state === 'semi') {
return (
<div className="glass" style={{ borderRadius: 10, padding: '12px 14px', display: 'flex', flexDirection: 'column', gap: 8 }}>
<div style={{ display: 'flex', alignItems: 'flex-start', gap: 8 }}>
<div style={{ flex: 1, minWidth: 0, ...noSelect }}>
{note.title && (
<div style={{ color: 'var(--ink-1)', fontFamily: 'var(--font-ui)', fontWeight: 600, fontSize: 14, marginBottom: 4 }}>
{note.title}
</div>
)}
<div style={{
color: 'var(--ink-2)', fontFamily: 'var(--font-ui)', fontSize: 13, lineHeight: 1.5,
display: '-webkit-box', WebkitLineClamp: 3, WebkitBoxOrient: 'vertical', overflow: 'hidden',
} as React.CSSProperties}>
{note.content}
</div>
</div>
{toggleBtn}
</div>
{metaLine}
{actionButtons}
</div>
)
}
// ─── EXPANDED ────────────────────────────────────────────────────────────────
return (
<div className="glass" style={{ borderRadius: 10, padding: '12px 14px', display: 'flex', flexDirection: 'column', gap: 10 }}>
<div style={{ display: 'flex', alignItems: 'flex-start', gap: 8, ...noSelect }}>
<div style={{ flex: 1, minWidth: 0 }}>
{note.title && (
<div style={{ color: 'var(--ink-1)', fontFamily: 'var(--font-ui)', fontWeight: 700, fontSize: 15 }}>
{note.title}
</div>
)}
</div>
{toggleBtn}
</div>
<div>{renderMarkdown(note.content)}</div>
{mediaSection}
{metaLine}
{actionButtons}
</div>
)
}
// ─── PAGE ─────────────────────────────────────────────────────────────────────
export default function NotesPage() {
const [notes, setNotes] = useState<Note[]>([])
const [loading, setLoading] = useState(true)
const [error, setError] = useState<string | null>(null)
const [showForm, setShowForm] = useState(false)
const [editingNote, setEditingNote] = useState<Note | null>(null)
const [filters, setFilters] = useState<NoteFilters>({})
const [searchInput, setSearchInput] = useState('')
const searchTimer = useRef<ReturnType<typeof setTimeout> | null>(null)
const { setActionButton } = useActionButton()
useEffect(() => {
setActionButton(
<button
onClick={() => setShowForm(true)}
aria-label="Nouvelle note"
style={{ width: 56, height: 56, borderRadius: '50%', background: 'var(--accent)', color: '#1d2021', border: 'none', fontSize: 24, cursor: 'pointer', boxShadow: '0 4px 16px rgba(0,0,0,0.5)', display: 'flex', alignItems: 'center', justifyContent: 'center' }}
>+</button>
)
return () => setActionButton(null)
}, [setActionButton])
const load = useCallback(async () => {
setLoading(true)
setError(null)
try {
setNotes(await fetchNotes(filters))
} catch {
setError('Erreur de chargement')
} finally {
setLoading(false)
}
}, [filters])
useEffect(() => { void load() }, [load])
useServerEvents({ notes_changed: () => void load() })
function handleSearchChange(val: string) {
setSearchInput(val)
if (searchTimer.current) clearTimeout(searchTimer.current)
searchTimer.current = setTimeout(() => {
setFilters(f => ({ ...f, q: val || undefined }))
}, 300)
}
async function handleCreate(data: Parameters<typeof createNote>[0]) {
await createNote(data)
setShowForm(false)
void load()
}
async function handleUpdate(id: string, data: Parameters<typeof updateNote>[1]) {
await updateNote(id, data)
setEditingNote(null)
void load()
}
async function handleDelete(id: string) {
if (!confirm('Supprimer cette note ?')) return
try { await deleteNote(id); void load() }
catch { setError('Erreur lors de la suppression') }
}
async function handleAddPhoto(noteId: string, file: File) {
try { await addAttachment(noteId, file); void load() }
catch { setError('Erreur upload photo') }
}
async function handleAddAudio(noteId: string, file: File) {
try { await addAttachment(noteId, file); void load() }
catch { setError('Erreur upload audio') }
}
async function handleAddVideo(noteId: string, file: File) {
try { await addAttachment(noteId, file); void load() }
catch { setError('Erreur upload vidéo') }
}
async function handleDeleteAtt(noteId: string, attId: string) {
try { await deleteAttachment(noteId, attId); void load() }
catch { setError('Erreur suppression pièce jointe') }
}
const hasActiveFilters = filters.has_photo || filters.has_audio || filters.has_video || filters.has_gps
const noteGrid = (cols: string) => (
<div style={{ display: 'grid', gridTemplateColumns: cols, gap: 10, alignItems: 'start' }}>
{notes.map(note => (
<NoteCard
key={note.id}
note={note}
onEdit={() => setEditingNote(note)}
onDelete={() => void handleDelete(note.id)}
onAddPhoto={f => void handleAddPhoto(note.id, f)}
onAddAudio={f => void handleAddAudio(note.id, f)}
onAddVideo={f => void handleAddVideo(note.id, f)}
onDeleteAtt={attId => void handleDeleteAtt(note.id, attId)}
/>
))}
</div>
)
return (
<div className="p-4">
<div style={{ display: 'flex', alignItems: 'center', gap: 8, marginBottom: 16 }}>
<h1 style={{ color: 'var(--accent)', fontFamily: 'var(--font-mono)', margin: 0, flex: 1, ...noSelect }}>Notes</h1>
{/* Bouton ajout visible sur laptop uniquement */}
<button
className="hidden lg:flex"
onClick={() => setShowForm(true)}
style={{ alignItems: 'center', gap: 8, padding: '8px 16px', borderRadius: 8, border: 'none', background: 'var(--accent)', color: '#1d2021', fontFamily: 'var(--font-ui)', fontSize: 13, fontWeight: 600, cursor: 'pointer', ...noSelect }}
>
<i className="fa-solid fa-plus" /> Nouvelle note
</button>
</div>
{/* Barre recherche + filtres */}
<div style={{ display: 'flex', gap: 8, marginBottom: 12, flexWrap: 'wrap', alignItems: 'center' }}>
<input
style={{ ...inputStyle, flex: 1, minWidth: 160 }}
placeholder="Rechercher…"
value={searchInput}
onChange={e => handleSearchChange(e.target.value)}
/>
{([
{ key: 'has_photo', icon: 'fa-image', label: 'Photo' },
{ key: 'has_audio', icon: 'fa-microphone', label: 'Audio' },
{ key: 'has_video', icon: 'fa-video', label: 'Vidéo' },
{ key: 'has_gps', icon: 'fa-location-dot', label: 'GPS' },
] as const).map(({ key, icon, label }) => {
const active = filters[key] === true
return (
<button
key={key}
onClick={() => setFilters(f => ({ ...f, [key]: active ? undefined : true }))}
style={{ padding: '5px 10px', borderRadius: 999, border: 'none', background: active ? 'var(--accent)' : 'var(--bg-3)', color: active ? '#1d2021' : 'var(--ink-3)', cursor: 'pointer', fontFamily: 'var(--font-ui)', fontSize: 12, display: 'flex', alignItems: 'center', gap: 5, ...noSelect }}
>
<i className={`fa-solid ${icon}`} style={{ fontSize: 11 }} />{label}
</button>
)
})}
{hasActiveFilters && (
<button
onClick={() => setFilters(f => ({ ...f, has_photo: undefined, has_audio: undefined, has_video: undefined, has_gps: undefined }))}
style={{ padding: '5px 10px', borderRadius: 999, border: 'none', background: 'transparent', color: 'var(--err)', cursor: 'pointer', fontSize: 12, ...noSelect }}
><i className="fa-solid fa-xmark" /> Filtres</button>
)}
</div>
{error && (
<p style={{ color: 'var(--err)', background: 'var(--bg-3)', borderRadius: 8, padding: '8px 12px', marginBottom: 12, fontFamily: 'var(--font-ui)', fontSize: 13 }}>{error}</p>
)}
{showForm && (
<Modal title="Nouvelle note" onClose={() => setShowForm(false)}>
<NoteForm onSubmit={handleCreate} onCancel={() => setShowForm(false)} />
</Modal>
)}
{editingNote && (
<Modal title="Modifier la note" onClose={() => setEditingNote(null)}>
<NoteForm initialValues={editingNote} onSubmit={data => handleUpdate(editingNote.id, data)} onCancel={() => setEditingNote(null)} submitLabel="Enregistrer" />
</Modal>
)}
{loading && <p style={{ color: 'var(--ink-3)', textAlign: 'center', padding: 24, ...noSelect }}>Chargement</p>}
{!loading && notes.length === 0 && (
<p style={{ color: 'var(--ink-3)', textAlign: 'center', marginTop: 40, ...noSelect }}>Aucune note</p>
)}
<div className="block lg:hidden">
{noteGrid('1fr')}
</div>
<div className="hidden lg:block">
{noteGrid('repeat(3, 1fr)')}
</div>
</div>
)
}