"use client"; import { useCallback, useEffect, useMemo, useRef, useState, type SetStateAction, type PointerEvent as ReactPointerEvent } from "react"; import { useParams, useRouter } from "next/navigation"; import { useShallow } from "zustand/react/shallow"; import Map, { type MapHandle } from "@/uhm/components/Map"; import Editor from "@/uhm/components/Editor"; import BackgroundLayersPanel from "@/uhm/components/editor/BackgroundLayersPanel"; import TimelineBar from "@/uhm/components/ui/TimelineBar"; import SelectedGeometryPanel from "@/uhm/components/editor/SelectedGeometryPanel"; import ReplayTimelineSidebar from "@/uhm/components/editor/ReplayTimelineSidebar"; import ReplayEffectsSidebar from "@/uhm/components/editor/ReplayEffectsSidebar"; import ReplayPreviewOverlay from "@/uhm/components/editor/ReplayPreviewOverlay"; import PublicWikiSidebar from "@/uhm/components/wiki/PublicWikiSidebar"; import WikiSidebarPanel from "@/uhm/components/wiki/WikiSidebarPanel"; import ProjectEntityRefsPanel from "@/uhm/components/editor/ProjectEntityRefsPanel"; import EntityWikiBindingsPanel from "@/uhm/components/editor/EntityWikiBindingsPanel"; import GeometryBindingPanel from "@/uhm/components/editor/GeometryBindingPanel"; import { Entity, fetchEntities, searchEntitiesByName } from "@/uhm/api/entities"; import { ApiError } from "@/uhm/api/http"; import { fetchCurrentUser } from "@/uhm/api/auth"; import { ProjectCommit } from "@/uhm/api/projects"; import { fetchWikiById, searchWikisByTitle, type Wiki } from "@/uhm/api/wikis"; import { searchGeometriesByEntityName, type EntityGeometriesSearchItem, type EntityGeometrySearchGeo } from "@/uhm/api/geometries"; import type { EntitySnapshot } from "@/uhm/types/entities"; import { Feature, FeatureCollection, Geometry, useEditorState, } from "@/uhm/lib/editor/state/useEditorState"; import { EditorMode } from "@/uhm/lib/editor/session/sessionTypes"; import { getDefaultTypeIdForFeature, normalizeFeatureBindingIds, normalizeFeatureEntityIds, uniqueEntityIds, } from "@/uhm/lib/editor/snapshot/editorSnapshot"; import { buildClientEntityId, mergeEntitySearchResults, } from "@/uhm/lib/editor/entity/entityBinding"; import { buildFeatureEntityPatch } from "@/uhm/lib/editor/entity/entityBinding"; import { loadBackgroundLayerVisibilityFromStorage, } from "@/uhm/lib/editor/background/backgroundVisibilityStorage"; import { deepClone } from "@/uhm/lib/editor/draft/draftDiff"; import { useProjectCommands } from "@/uhm/lib/editor/project/useProjectCommands"; import { useReplayPreview } from "@/uhm/lib/replay/useReplayPreview"; import { EMPTY_FEATURE_COLLECTION } from "@/uhm/lib/map/geo/constants"; import { FIXED_TIMELINE_RANGE, clampYearToFixedRange } from "@/uhm/lib/utils/timeline"; import { useFeatureCommands } from "./featureCommands"; import { deleteSubmission } from "@/uhm/api/projects"; import type { WikiSnapshot } from "@/uhm/types/wiki"; import type { BattleReplay, EntityWikiLinkSnapshot } from "@/uhm/types/projects"; import UnifiedSearchBar from "@/uhm/components/ui/UnifiedSearchBar"; import { EditorStoreProvider, useEditorStore, useEditorStoreApi, } from "@/uhm/store/editorStore"; const CURRENT_YEAR = new Date().getUTCFullYear(); const DEFAULT_EDITOR_USER_ID = "local-editor"; type ReplayPreviewSession = { replay: BattleReplay; draft: FeatureCollection; wikis: WikiSnapshot[]; selectedStageId: number | null; selectedStepIndex: number | null; timelineYear: number; timelineFilterEnabled: boolean; mapViewState: ReturnType; }; export default function Page() { return ( ); } function EditorPageContent() { const params = useParams(); const router = useRouter(); const editorStoreApi = useEditorStoreApi(); const projectId = String(params.id || ""); const openedProjectIdRef = useRef(null); const entityFormStatusTimeoutRef = useRef(null); const geoBindingStatusTimeoutRef = useRef(null); const localCreatedEntityIdsRef = useRef>(new Set()); const lastSelectedFeatureIdRef = useRef(null); const mapHandleRef = useRef(null); const { mode, internalSetMode, initialData, isSaving, isSubmitting, isOpeningSection, setIsOpeningSection, commitTitle, setCommitTitle, activeSection, projectState, sectionCommits, baselineSnapshot, entityCatalog, setEntityCatalog, snapshotEntities, setSnapshotEntities, entityStatus, setEntityStatus, selectedFeatureIds, setSelectedFeatureIds, entityForm, setEntityForm, selectedGeometryEntityIds, setSelectedGeometryEntityIds, geometryMetaForm, setGeometryMetaForm, setIsEntitySubmitting, setEntityFormStatus, entitySearchResults, setEntitySearchResults, isEntitySearchLoading, setIsEntitySearchLoading, timelineDraftYear, setTimelineDraftYear, backgroundVisibility, setBackgroundVisibility, isBackgroundVisibilityReady, setIsBackgroundVisibilityReady, snapshotWikis, setSnapshotWikis, snapshotEntityWikiLinks, setSnapshotEntityWikiLinks, blockedPendingSubmissionId, setBlockedPendingSubmissionId, searchKind, setSearchKind, searchQuery, setSearchQuery, searchQueryDraft, setSearchQueryDraft, wikiSearchResults, setWikiSearchResults, isWikiSearching, setIsWikiSearching, geoSearchResults, setGeoSearchResults, isGeoSearching, setIsGeoSearching, setRequestedActiveWikiId, leftPanelWidth, setLeftPanelWidth, rightPanelWidth, setRightPanelWidth, timelineFilterEnabled, setTimelineFilterEnabled, geometryBindingFilterEnabled, setGeoBindingStatus, geometryFocusRequest, setGeometryFocusRequest, replayFeatureId, setReplayFeatureId, hideOutside, setHideOutside, geometryVisibility, } = useEditorStore(useShallow((state) => ({ mode: state.mode, internalSetMode: state.setMode, initialData: state.initialData, isSaving: state.isSaving, isSubmitting: state.isSubmitting, isOpeningSection: state.isOpeningSection, setIsOpeningSection: state.setIsOpeningSection, commitTitle: state.commitTitle, setCommitTitle: state.setCommitTitle, activeSection: state.activeSection, projectState: state.projectState, sectionCommits: state.sectionCommits, baselineSnapshot: state.baselineSnapshot, entityCatalog: state.entityCatalog, setEntityCatalog: state.setEntityCatalog, snapshotEntities: state.snapshotEntities, setSnapshotEntities: state.setSnapshotEntities, entityStatus: state.entityStatus, setEntityStatus: state.setEntityStatus, selectedFeatureIds: state.selectedFeatureIds, setSelectedFeatureIds: state.setSelectedFeatureIds, entityForm: state.entityForm, setEntityForm: state.setEntityForm, selectedGeometryEntityIds: state.selectedGeometryEntityIds, setSelectedGeometryEntityIds: state.setSelectedGeometryEntityIds, geometryMetaForm: state.geometryMetaForm, setGeometryMetaForm: state.setGeometryMetaForm, setIsEntitySubmitting: state.setIsEntitySubmitting, setEntityFormStatus: state.setEntityFormStatus, entitySearchResults: state.entitySearchResults, setEntitySearchResults: state.setEntitySearchResults, isEntitySearchLoading: state.isEntitySearchLoading, setIsEntitySearchLoading: state.setIsEntitySearchLoading, timelineDraftYear: state.timelineDraftYear, setTimelineDraftYear: state.setTimelineDraftYear, backgroundVisibility: state.backgroundVisibility, setBackgroundVisibility: state.setBackgroundVisibility, isBackgroundVisibilityReady: state.isBackgroundVisibilityReady, setIsBackgroundVisibilityReady: state.setIsBackgroundVisibilityReady, snapshotWikis: state.snapshotWikis, setSnapshotWikis: state.setSnapshotWikis, snapshotEntityWikiLinks: state.snapshotEntityWikiLinks, setSnapshotEntityWikiLinks: state.setSnapshotEntityWikiLinks, blockedPendingSubmissionId: state.blockedPendingSubmissionId, setBlockedPendingSubmissionId: state.setBlockedPendingSubmissionId, searchKind: state.searchKind, setSearchKind: state.setSearchKind, searchQuery: state.searchQuery, setSearchQuery: state.setSearchQuery, searchQueryDraft: state.searchQueryDraft, setSearchQueryDraft: state.setSearchQueryDraft, wikiSearchResults: state.wikiSearchResults, setWikiSearchResults: state.setWikiSearchResults, isWikiSearching: state.isWikiSearching, setIsWikiSearching: state.setIsWikiSearching, geoSearchResults: state.geoSearchResults, setGeoSearchResults: state.setGeoSearchResults, isGeoSearching: state.isGeoSearching, setIsGeoSearching: state.setIsGeoSearching, setRequestedActiveWikiId: state.setRequestedActiveWikiId, leftPanelWidth: state.leftPanelWidth, setLeftPanelWidth: state.setLeftPanelWidth, rightPanelWidth: state.rightPanelWidth, setRightPanelWidth: state.setRightPanelWidth, timelineFilterEnabled: state.timelineFilterEnabled, setTimelineFilterEnabled: state.setTimelineFilterEnabled, geometryBindingFilterEnabled: state.geometryBindingFilterEnabled, setGeoBindingStatus: state.setGeoBindingStatus, geometryFocusRequest: state.geometryFocusRequest, setGeometryFocusRequest: state.setGeometryFocusRequest, replayFeatureId: state.replayFeatureId, setReplayFeatureId: state.setReplayFeatureId, hideOutside: state.hideOutside, setHideOutside: state.setHideOutside, geometryVisibility: state.geometryVisibility, }))); // Counter để bỏ qua response cũ khi user gõ search entity liên tục. const entitySearchRequestRef = useRef(0); const wikiSearchRequestRef = useRef(0); const geoSearchRequestRef = useRef(0); const snapshotEntitiesRef = useRef(snapshotEntities); const snapshotWikisRef = useRef(snapshotWikis); const snapshotEntityWikiLinksRef = useRef(snapshotEntityWikiLinks); useEffect(() => { snapshotEntitiesRef.current = snapshotEntities; }, [snapshotEntities]); useEffect(() => { snapshotWikisRef.current = snapshotWikis; }, [snapshotWikis]); useEffect(() => { snapshotEntityWikiLinksRef.current = snapshotEntityWikiLinks; }, [snapshotEntityWikiLinks]); const editor = useEditorState(initialData, { snapshotUndo: { snapshotEntitiesRef, setSnapshotEntities, snapshotWikisRef, setSnapshotWikis, snapshotEntityWikiLinksRef, setSnapshotEntityWikiLinks, }, initialReplays: baselineSnapshot?.replays, mode: mode, }); const setSnapshotWikisUndoable = useCallback( (next: SetStateAction) => { editor.setSnapshotWikis(next, "Cập nhật wiki"); }, [editor] ); const setSnapshotEntityWikiLinksUndoable = useCallback( (next: SetStateAction) => { editor.setSnapshotEntityWikiLinks(next, "Cập nhật entity-wiki"); }, [editor] ); const snapshotEntitiesAsEntities = useMemo(() => { const rows = snapshotEntities || []; return rows .filter((e) => e && e.operation !== "delete") .map((e) => ({ id: String(e.id || ""), name: String(e.name || "").trim() || String(e.id || ""), description: e.description ?? null, geometry_count: 0, })) .filter((e) => e.id.length > 0 && e.name.length > 0); }, [snapshotEntities]); const entities = useMemo( () => mergeEntitySearchResults(entityCatalog, snapshotEntitiesAsEntities), [entityCatalog, snapshotEntitiesAsEntities] ); const [replaySelection, setReplaySelection] = useState<{ stageId: number | null; stepIndex: number | null; }>({ stageId: null, stepIndex: null, }); const [previewSession, setPreviewSession] = useState(null); const [previewAutoplayMode, setPreviewAutoplayMode] = useState<"start" | "selection" | null>(null); const [previewWikiCache, setPreviewWikiCache] = useState>({}); const [previewWikiError, setPreviewWikiError] = useState(null); const [isPreviewWikiLoading, setIsPreviewWikiLoading] = useState(false); const handleReplaySelectionChange = useCallback((stageId: number | null, stepIndex: number | null) => { setReplaySelection({ stageId, stepIndex }); }, []); const getCurrentMapInstance = useCallback(() => mapHandleRef.current?.getMap() ?? null, []); const getCurrentMapViewState = useCallback(() => mapHandleRef.current?.getViewState() ?? null, []); const isReplayEditMode = mode === "replay"; const isReplayPreviewMode = mode === "replay_preview"; const entitiesRef = useRef(entities); useEffect(() => { entitiesRef.current = entities; }, [entities]); useEffect(() => { const localCreatedIds = localCreatedEntityIdsRef.current; if (!localCreatedIds.size) return; const snapshotIds = new Set((snapshotEntities || []).map((entity) => String(entity.id || ""))); setEntityCatalog((prev) => { let changed = false; const next = (prev || []).filter((entity) => { const id = String(entity?.id || ""); const shouldDrop = localCreatedIds.has(id) && !snapshotIds.has(id); if (shouldDrop) { changed = true; localCreatedIds.delete(id); return false; } return true; }); return changed ? next : prev; }); }, [snapshotEntities, setEntityCatalog]); const handleTimelineYearChange = useCallback((nextYear: number) => { setTimelineDraftYear(clampYearToFixedRange(Math.trunc(nextYear))); }, [setTimelineDraftYear]); const replayPreview = useReplayPreview({ replay: previewSession?.replay || null, draft: previewSession?.draft || EMPTY_FEATURE_COLLECTION, getMapInstance: getCurrentMapInstance, initialTimelineYear: previewSession?.timelineYear ?? timelineDraftYear, initialTimelineFilterEnabled: previewSession?.timelineFilterEnabled ?? timelineFilterEnabled, initialMapViewState: previewSession?.mapViewState ?? null, selectedStageId: previewSession?.selectedStageId ?? replaySelection.stageId, selectedStepIndex: previewSession?.selectedStepIndex ?? replaySelection.stepIndex, onSelectStep: () => {}, }); const replayPreviewDraft = useMemo(() => { const sourceDraft = previewSession?.draft || EMPTY_FEATURE_COLLECTION; if (!isReplayPreviewMode || replayPreview.hiddenGeometryIds.length === 0) { return sourceDraft; } const hiddenIds = new Set(replayPreview.hiddenGeometryIds); return { ...sourceDraft, features: sourceDraft.features.filter( (feature) => !hiddenIds.has(String(feature.properties.id)) ), }; }, [isReplayPreviewMode, previewSession?.draft, replayPreview.hiddenGeometryIds]); const activeTimelineYear = isReplayPreviewMode ? replayPreview.timelineYear : timelineDraftYear; const activeTimelineFilterEnabled = isReplayPreviewMode ? replayPreview.timelineFilterEnabled : timelineFilterEnabled; // Timeline filter: only affects persisted snapshot features. // New features created in the current session remain visible regardless of time range. const timelineVisibleDraft = useMemo(() => { const activeDraft = isReplayPreviewMode ? replayPreviewDraft : isReplayEditMode ? editor.replayDraft : editor.mainDraft; if (!activeTimelineFilterEnabled) return activeDraft; const year = clampYearToFixedRange(Math.trunc(activeTimelineYear)); return { ...activeDraft, features: activeDraft.features.filter((feature) => { if (!editor.hasPersistedFeature(feature.properties.id)) return true; return isFeatureVisibleAtYear(feature, year); }), }; }, [ activeTimelineFilterEnabled, activeTimelineYear, editor, isReplayEditMode, isReplayPreviewMode, replayPreviewDraft, ]); const selectedFeatures = useMemo(() => { if (!selectedFeatureIds || selectedFeatureIds.length === 0) return []; return selectedFeatureIds .map(id => editor.draft.features.find(f => String(f.properties.id) === String(id))) .filter(Boolean) as Feature[]; }, [selectedFeatureIds, editor.draft.features]); const isMultiEditValid = useMemo(() => { if (selectedFeatures.length <= 1) return true; const firstShape = selectedFeatures[0].geometry.type; return selectedFeatures.every(f => f.geometry.type === firstShape); }, [selectedFeatures]); const selectedFeature = selectedFeatures.length > 0 && isMultiEditValid ? selectedFeatures[0] : null; const geometryChoices = useMemo(() => { const createdGeometryIds = new Set(); for (const [id, change] of editor.changes.entries()) { if (change.action === "create") createdGeometryIds.add(String(id)); } const rows = (editor.draft.features || []) .filter((f) => f && f.properties && (typeof f.properties.id === "string" || typeof f.properties.id === "number")) .map((f) => { const id = String(f.properties.id); const semantic = String(f.properties.type || getDefaultTypeIdForFeature(f) || "").trim(); const label = semantic.length ? `${semantic} (${f.geometry.type})` : f.geometry.type; return { id, label, isNew: createdGeometryIds.has(id) || !editor.hasPersistedFeature(f.properties.id), }; }); rows.sort((a, b) => a.id.localeCompare(b.id)); return rows; }, [editor]); const selectedGeometryBindingIds = useMemo(() => { if (!selectedFeature) return []; return normalizeFeatureBindingIds(selectedFeature); }, [selectedFeature]); const wikiChoices = useMemo(() => { return (snapshotWikis || []) .filter((wiki) => wiki && wiki.operation !== "delete") .map((wiki) => ({ id: String(wiki.id || ""), label: `${(wiki.title || "").trim() || "Untitled wiki"} (${String(wiki.id || "")})`, })) .filter((wiki) => wiki.id.length > 0); }, [snapshotWikis]); const wikiDirty = useMemo(() => { const prev = normalizeWikisForCompare(baselineSnapshot?.wikis); const next = normalizeWikisForCompare(snapshotWikis); try { return JSON.stringify(prev) !== JSON.stringify(next); } catch { return true; } }, [baselineSnapshot?.wikis, snapshotWikis]); const entitiesDirty = useMemo(() => { const prev = normalizeEntitiesForCompare(baselineSnapshot?.entities); const next = normalizeEntitiesForCompare(snapshotEntities); try { return JSON.stringify(prev) !== JSON.stringify(next); } catch { return true; } }, [baselineSnapshot?.entities, snapshotEntities]); const entityWikiDirty = useMemo(() => { const prev = normalizeEntityWikiLinksForCompare(baselineSnapshot?.entity_wiki); const next = normalizeEntityWikiLinksForCompare(snapshotEntityWikiLinks); try { return JSON.stringify(prev) !== JSON.stringify(next); } catch { return true; } }, [snapshotEntityWikiLinks, baselineSnapshot?.entity_wiki]); const replayDirty = useMemo(() => { const prev = normalizeReplaysForCompare(baselineSnapshot?.replays); const next = normalizeReplaysForCompare(editor.effectiveReplays); try { return JSON.stringify(prev) !== JSON.stringify(next); } catch { return true; } }, [baselineSnapshot?.replays, editor.effectiveReplays]); const pendingSaveCount = editor.changeCount + (wikiDirty ? 1 : 0) + (entitiesDirty ? 1 : 0) + (entityWikiDirty ? 1 : 0) + (replayDirty ? 1 : 0); const activeReplayStages = useMemo( () => editor.activeReplayDraft?.detail || [], [editor.activeReplayDraft?.detail] ); const sectionCommands = useProjectCommands({ editor, store: editorStoreApi, emptyFeatureCollection: EMPTY_FEATURE_COLLECTION, pendingSaveCount, }); const { openSectionForEditing, commitSection, submitCurrentSection, restoreCommit, } = sectionCommands; const exitReplayPreview = useCallback(() => { replayPreview.resetPreview(); setPreviewAutoplayMode(null); setPreviewSession(null); internalSetMode("replay"); }, [internalSetMode, replayPreview.resetPreview]); const openReplayPreview = useCallback((autoplayMode: "start" | "selection") => { if (!editor.activeReplayDraft) return; setPreviewSession({ replay: deepClone(editor.activeReplayDraft), draft: deepClone(editor.replayDraft), wikis: deepClone(snapshotWikis), selectedStageId: replaySelection.stageId, selectedStepIndex: replaySelection.stepIndex, timelineYear: timelineDraftYear, timelineFilterEnabled, mapViewState: getCurrentMapViewState(), }); setPreviewAutoplayMode(autoplayMode); setSelectedFeatureIds([]); internalSetMode("replay_preview"); }, [ editor.activeReplayDraft, editor.replayDraft, getCurrentMapViewState, internalSetMode, replaySelection.stageId, replaySelection.stepIndex, setSelectedFeatureIds, snapshotWikis, timelineDraftYear, timelineFilterEnabled, ]); const setMode = useCallback((m: EditorMode, featureId?: string | number) => { if (m === "replay_preview") { return; } if (mode === "replay_preview") { replayPreview.resetPreview(); setPreviewAutoplayMode(null); setPreviewSession(null); if (m === "replay") { internalSetMode("replay"); return; } editor.closeReplayContext(); setSelectedFeatureIds([]); setReplayFeatureId(null); setHideOutside(false); setReplaySelection({ stageId: null, stepIndex: null }); internalSetMode(m); return; } if (m === "replay" && featureId) { // QUY TẮC: Geo chọn đầu tiên là geo main. const triggerId = selectedFeatureIds.length > 0 ? selectedFeatureIds[0] : featureId; setReplayFeatureId(triggerId); setReplaySelection({ stageId: null, stepIndex: null }); editor.switchReplayContext(triggerId, selectedFeatureIds); setSelectedFeatureIds([]); } else if (m !== "replay") { if (mode === "replay") { editor.closeReplayContext(); setSelectedFeatureIds([]); } setReplayFeatureId(null); setHideOutside(false); setReplaySelection({ stageId: null, stepIndex: null }); } internalSetMode(m); }, [ editor, internalSetMode, mode, replayPreview.resetPreview, selectedFeatureIds, setHideOutside, setReplayFeatureId, setSelectedFeatureIds, ]); useEffect(() => { if (!activeReplayStages.length) { if (replaySelection.stageId != null || replaySelection.stepIndex != null) { setReplaySelection({ stageId: null, stepIndex: null }); } return; } const targetStage = activeReplayStages.find((stage) => stage.id === replaySelection.stageId) || activeReplayStages[0]; const nextStageId = targetStage.id; let nextStepIndex: number | null = null; if (targetStage.steps.length > 0) { if ( replaySelection.stageId === targetStage.id && replaySelection.stepIndex != null && replaySelection.stepIndex >= 0 && replaySelection.stepIndex < targetStage.steps.length ) { nextStepIndex = replaySelection.stepIndex; } else { nextStepIndex = 0; } } if ( nextStageId !== replaySelection.stageId || nextStepIndex !== replaySelection.stepIndex ) { setReplaySelection({ stageId: nextStageId, stepIndex: nextStepIndex, }); } }, [activeReplayStages, replaySelection.stageId, replaySelection.stepIndex]); useEffect(() => { if (!isReplayPreviewMode || !previewSession || !previewAutoplayMode) return; if (previewAutoplayMode === "selection") { replayPreview.playFromSelection(); } else { replayPreview.playFromStart(); } setPreviewAutoplayMode(null); }, [ isReplayPreviewMode, previewAutoplayMode, previewSession, replayPreview.playFromSelection, replayPreview.playFromStart, ]); useEffect(() => { setPreviewWikiCache({}); setPreviewWikiError(null); setIsPreviewWikiLoading(false); }, [previewSession]); const replayPreviewActiveStepLabel = useMemo(() => { if ( replayPreview.activeCursor.stageId == null || replayPreview.activeCursor.stepIndex == null ) { return null; } return `Stage #${replayPreview.activeCursor.stageId} · Step ${replayPreview.activeCursor.stepIndex + 1}`; }, [replayPreview.activeCursor.stageId, replayPreview.activeCursor.stepIndex]); const replayPreviewWikiRows = previewSession?.wikis || []; const replayPreviewActiveWikiSnapshot = useMemo(() => { if (!replayPreview.activeWikiId) return null; return replayPreviewWikiRows.find((item) => item.id === replayPreview.activeWikiId) || null; }, [replayPreview.activeWikiId, replayPreviewWikiRows]); useEffect(() => { if (!isReplayPreviewMode || !replayPreview.sidebarOpen) { setPreviewWikiError(null); setIsPreviewWikiLoading(false); return; } const activeWikiId = String(replayPreview.activeWikiId || "").trim(); if (!activeWikiId.length) { setPreviewWikiError(null); setIsPreviewWikiLoading(false); return; } const localWiki = replayPreviewWikiRows.find((item) => item.id === activeWikiId) || null; if (!localWiki) { setPreviewWikiError("Không tìm thấy wiki trong snapshot preview."); setIsPreviewWikiLoading(false); return; } if (typeof localWiki.doc === "string") { setPreviewWikiError(null); setIsPreviewWikiLoading(false); return; } if (previewWikiCache[activeWikiId]) { setPreviewWikiError(null); setIsPreviewWikiLoading(false); return; } let disposed = false; setPreviewWikiError(null); setIsPreviewWikiLoading(true); void fetchWikiById(activeWikiId) .then((row) => { if (disposed) return; setPreviewWikiCache((prev) => ({ ...prev, [activeWikiId]: row })); }) .catch((err) => { if (disposed) return; setPreviewWikiError(err instanceof Error ? err.message : "Không tải được wiki preview."); }) .finally(() => { if (!disposed) { setIsPreviewWikiLoading(false); } }); return () => { disposed = true; }; }, [ isReplayPreviewMode, previewWikiCache, replayPreview.activeWikiId, replayPreview.sidebarOpen, replayPreviewWikiRows, ]); const replayPreviewActiveWiki = useMemo(() => { const snapshotWiki = replayPreviewActiveWikiSnapshot; if (!snapshotWiki) return null; if (typeof snapshotWiki.doc === "string") { return { id: snapshotWiki.id, project_id: projectId, title: snapshotWiki.title, slug: snapshotWiki.slug ?? null, content: snapshotWiki.doc || "", }; } return previewWikiCache[snapshotWiki.id] || null; }, [previewWikiCache, projectId, replayPreviewActiveWikiSnapshot]); const handleReplayPreviewWikiLinkRequest = useCallback(({ slug }: { slug: string; rect: DOMRect }) => { const nextSlug = String(slug || "").trim(); if (!nextSlug.length) return; const match = replayPreviewWikiRows.find((item) => String(item.slug || "").trim() === nextSlug) || null; if (!match) { setPreviewWikiError(`Wiki /wiki/${nextSlug} không có trong snapshot preview.`); return; } setPreviewWikiError(null); replayPreview.openWikiPanelById(match.id); }, [replayPreview.openWikiPanelById, replayPreviewWikiRows]); const effectiveGeometryVisibility = useMemo(() => { const visibility: Record = { ...geometryVisibility }; if ((isReplayEditMode || isReplayPreviewMode) && replayFeatureId) { // Ẩn chính geo được chọn làm replay (marker kịch bản) visibility[String(replayFeatureId)] = false; if (isReplayEditMode && hideOutside) { // Trong mode replay, ta chỉ hiển thị những gì có trong draft của replay đó const currentReplayFeatureIds = new Set(editor.draft.features.map(f => String(f.properties.id))); // Ẩn tất cả các geo KHÔNG nằm trong draft replay hiện tại Object.keys(visibility).forEach(fid => { if (fid === String(replayFeatureId)) { visibility[fid] = false; } else { visibility[fid] = currentReplayFeatureIds.has(fid); } }); } } return visibility; }, [ editor.draft.features, geometryVisibility, hideOutside, isReplayEditMode, isReplayPreviewMode, replayFeatureId, ]); const openProject = useCallback(async () => { if (!projectId) return; try { setIsOpeningSection(true); setEntityStatus(null); setBlockedPendingSubmissionId(null); await openSectionForEditing(projectId); setEntityStatus(null); } catch (err) { if (err instanceof ApiError) { // Only bounce to login when the session is truly unauthenticated. // Token refresh is handled centrally; if we still get 401 here, refresh likely failed/expired. if (err.status === 401) { router.replace("/signin"); return; } // Pending submission blocks editor in BE. We parse the pending id to offer delete/unlock. if (err.status === 409) { try { const payload = JSON.parse(err.body || "{}"); if (payload?.pending_submission_id) { setBlockedPendingSubmissionId(String(payload.pending_submission_id)); setEntityStatus("Project đang có submission PENDING. Hãy xoa submission đó để unlock editor."); return; } } catch { // fallthrough } } setEntityStatus(`Mở project thất bại: ${err.body || err.message}`); } else { console.error("Open project failed", err); setEntityStatus("Mở project thất bại."); } } finally { setIsOpeningSection(false); } }, [openSectionForEditing, projectId, router, setBlockedPendingSubmissionId, setEntityStatus, setIsOpeningSection]); const unlockByDeletingPendingSubmission = useCallback(async () => { if (!blockedPendingSubmissionId) return; const confirmed = window.confirm("Xoa submission PENDING de unlock editor? Hanh dong nay khong the hoan tac."); if (!confirmed) return; try { setIsOpeningSection(true); setEntityStatus(null); await deleteSubmission(blockedPendingSubmissionId); setBlockedPendingSubmissionId(null); await openProject(); } catch (err) { if (err instanceof ApiError) { setEntityStatus(`Khong the xoa submission: ${err.body || err.message}`); } else { setEntityStatus("Khong the xoa submission."); } } finally { setIsOpeningSection(false); } }, [blockedPendingSubmissionId, openProject, setBlockedPendingSubmissionId, setEntityStatus, setIsOpeningSection]); useEffect(() => { let disposed = false; async function ensureAuthenticated() { try { await fetchCurrentUser(); } catch (err) { if (disposed) return; if (err instanceof ApiError && err.status === 401) { // Only redirect when refresh token/session is no longer usable. router.replace("/signin"); return; } console.error("Ensure authenticated failed", err); } } ensureAuthenticated(); return () => { disposed = true; }; }, [router]); useEffect(() => { if (!projectId) return; if (openedProjectIdRef.current === projectId) return; openProject() .then(() => { openedProjectIdRef.current = projectId; }) .catch(() => { // allow retry if openProject threw outside its try/catch (should be rare) openedProjectIdRef.current = null; }); }, [openProject, projectId]); useEffect(() => { let disposed = false; async function loadEntities() { try { const rows = await fetchEntities(); if (disposed) return; setEntityCatalog((prev) => { const byId = new globalThis.Map(); for (const row of prev || []) { if (!row?.id) continue; byId.set(String(row.id), row); } for (const row of rows || []) { if (!row?.id) continue; // Prefer the freshest backend payload on conflicts. byId.set(String(row.id), row); } return Array.from(byId.values()); }); setEntityStatus(null); } catch (err) { if (disposed) return; console.error("Load entities failed", err); setEntityStatus("Không tải được danh sách entity."); } } loadEntities(); return () => { disposed = true; }; }, [setEntityCatalog, setEntityStatus]); useEffect(() => { if (searchKind !== "entity") { setEntitySearchResults([]); setIsEntitySearchLoading(false); return; } const keyword = searchQuery.trim(); if (!keyword.length) { setEntitySearchResults([]); setIsEntitySearchLoading(false); return; } let disposed = false; const requestId = ++entitySearchRequestRef.current; const timeoutId = window.setTimeout(async () => { const keywordLower = keyword.toLowerCase(); const localMatches = entitiesRef.current .filter((entity) => entity.name.toLowerCase().includes(keywordLower) || (entity.description || "").toLowerCase().includes(keywordLower) ) .map((entity) => ({ ...entity, geometry_count: typeof entity.geometry_count === "number" ? entity.geometry_count : 0, })); setIsEntitySearchLoading(true); try { const rows = await searchEntitiesByName(keyword, { limit: 30 }); if (disposed || requestId !== entitySearchRequestRef.current) return; // Centralize: merge search results into the shared entity catalog so UI stays consistent. setEntityCatalog((prev) => { const byId = new globalThis.Map(); for (const row of prev || []) { if (!row?.id) continue; byId.set(String(row.id), row); } for (const row of rows || []) { if (!row?.id) continue; byId.set(String(row.id), row); } return Array.from(byId.values()); }); const mergedRows = mergeEntitySearchResults(rows, localMatches); setEntitySearchResults(mergedRows); } catch (err) { if (disposed || requestId !== entitySearchRequestRef.current) return; console.error("Search entity by name failed", err); setEntitySearchResults(localMatches); } finally { if (!disposed && requestId === entitySearchRequestRef.current) { setIsEntitySearchLoading(false); } } }, 220); return () => { disposed = true; window.clearTimeout(timeoutId); }; }, [ searchKind, searchQuery, setEntityCatalog, setEntitySearchResults, setIsEntitySearchLoading, ]); useEffect(() => { if (searchKind !== "wiki") { setWikiSearchResults([]); setIsWikiSearching(false); return; } const keyword = searchQuery.trim(); if (!keyword.length) { setWikiSearchResults([]); setIsWikiSearching(false); return; } let disposed = false; const requestId = ++wikiSearchRequestRef.current; const timeoutId = window.setTimeout(async () => { setIsWikiSearching(true); try { const rows = await searchWikisByTitle(keyword, { limit: 12 }); if (disposed || requestId !== wikiSearchRequestRef.current) return; setWikiSearchResults(rows); } catch (err) { if (disposed || requestId !== wikiSearchRequestRef.current) return; console.error("Search wikis failed", err); setWikiSearchResults([]); } finally { if (!disposed && requestId === wikiSearchRequestRef.current) { setIsWikiSearching(false); } } }, 250); return () => { disposed = true; window.clearTimeout(timeoutId); }; }, [searchKind, searchQuery, setIsWikiSearching, setWikiSearchResults]); useEffect(() => { if (searchKind !== "geo") { setGeoSearchResults([]); setIsGeoSearching(false); return; } const keyword = searchQuery.trim(); if (!keyword.length) { setGeoSearchResults([]); setIsGeoSearching(false); return; } let disposed = false; const requestId = ++geoSearchRequestRef.current; const timeoutId = window.setTimeout(async () => { setIsGeoSearching(true); try { const res = await searchGeometriesByEntityName(keyword, { limit: 24 }); if (disposed || requestId !== geoSearchRequestRef.current) return; setGeoSearchResults(res.items || []); } catch (err) { if (disposed || requestId !== geoSearchRequestRef.current) return; console.error("Search geometries by entity name failed", err); setGeoSearchResults([]); } finally { if (!disposed && requestId === geoSearchRequestRef.current) { setIsGeoSearching(false); } } }, 260); return () => { disposed = true; window.clearTimeout(timeoutId); }; }, [searchKind, searchQuery, setGeoSearchResults, setIsGeoSearching]); useEffect(() => { if (!selectedFeatureIds || selectedFeatureIds.length === 0) return; const stillExistIds = selectedFeatureIds.filter(id => timelineVisibleDraft.features.some(feature => String(feature.properties.id) === String(id)) ); if (stillExistIds.length !== selectedFeatureIds.length) { setSelectedFeatureIds(stillExistIds); } }, [timelineVisibleDraft, selectedFeatureIds, setSelectedFeatureIds]); useEffect(() => { if (!selectedFeature) { setSelectedGeometryEntityIds([]); setGeometryMetaForm({ type_key: "", time_start: "", time_end: "", binding: "", }); setEntityFormStatus(null); lastSelectedFeatureIdRef.current = null; return; } const featureEntityIds = normalizeFeatureEntityIds(selectedFeature); const nextTypeKey = typeof selectedFeature.properties.type === "string" && selectedFeature.properties.type.trim().length ? selectedFeature.properties.type : getDefaultTypeIdForFeature(selectedFeature); const currentId = String(selectedFeature.properties.id); setSelectedGeometryEntityIds(featureEntityIds); setGeometryMetaForm({ type_key: nextTypeKey, time_start: selectedFeature.properties.time_start != null ? String(selectedFeature.properties.time_start) : "", time_end: selectedFeature.properties.time_end != null ? String(selectedFeature.properties.time_end) : "", binding: normalizeFeatureBindingIds(selectedFeature).join(", "), }); // Only clear status when switching to a different geometry, not when patching metadata/bindings // on the same selected geometry (otherwise messages will blink). if (lastSelectedFeatureIdRef.current !== currentId) { setEntityFormStatus(null); } lastSelectedFeatureIdRef.current = currentId; }, [ selectedFeature, setEntityFormStatus, setGeometryMetaForm, setSelectedGeometryEntityIds, ]); const flashEntityFormStatus = useCallback((msg: string | null, timeoutMs = 3000) => { if (entityFormStatusTimeoutRef.current) { window.clearTimeout(entityFormStatusTimeoutRef.current); entityFormStatusTimeoutRef.current = null; } setEntityFormStatus(msg); if (msg && timeoutMs > 0) { entityFormStatusTimeoutRef.current = window.setTimeout(() => { setEntityFormStatus(null); entityFormStatusTimeoutRef.current = null; }, timeoutMs); } }, [setEntityFormStatus]); const flashGeoBindingStatus = useCallback((msg: string | null, timeoutMs = 3000) => { if (geoBindingStatusTimeoutRef.current) { window.clearTimeout(geoBindingStatusTimeoutRef.current); geoBindingStatusTimeoutRef.current = null; } setGeoBindingStatus(msg); if (msg && timeoutMs > 0) { geoBindingStatusTimeoutRef.current = window.setTimeout(() => { setGeoBindingStatus(null); geoBindingStatusTimeoutRef.current = null; }, timeoutMs); } }, [setGeoBindingStatus]); useEffect(() => { setBackgroundVisibility(loadBackgroundLayerVisibilityFromStorage()); setIsBackgroundVisibilityReady(true); }, [setBackgroundVisibility, setIsBackgroundVisibilityReady]); const handleAddEntityRefToProject = useCallback((entity: Entity) => { const id = String(entity.id || "").trim(); if (!id) return; editor.setSnapshotEntities((prev) => { if (prev.some((e) => String(e.id) === id)) return prev; return [ { id, source: "ref", operation: "reference", name: entity.name, description: entity.description ?? null, }, ...prev, ]; }, `Thêm entity ref #${id}`); // Keep entity catalog centralized as a single in-memory list. setEntityCatalog((prev) => { const byId = new globalThis.Map(); for (const row of prev || []) { if (!row?.id) continue; byId.set(String(row.id), row); } byId.set(id, entity); return Array.from(byId.values()); }); }, [editor, setEntityCatalog]); const handleUpdateEntityInProject = useCallback((entityId: string, payload: { name: string; description: string | null }) => { const id = String(entityId || "").trim(); if (!id) return; const nextName = String(payload?.name || "").trim(); if (!nextName.length) { flashEntityFormStatus("Ten entity la bat buoc."); return; } const nextDescription = payload?.description == null ? null : String(payload.description); editor.setSnapshotEntities((prev) => prev.map((e) => { if (!e || String(e.id) !== id) return e; const source = e.source === "inline" ? "inline" : "ref"; const operation = source === "ref" ? "reference" : e.operation === "create" ? "create" : "update"; return { ...e, id, source, operation, name: nextName, description: nextDescription, }; }), `Cap nhat entity #${id}`); flashEntityFormStatus("Da cap nhat entity. Commit khi san sang.", 3000); }, [editor, flashEntityFormStatus]); const handleToggleBindEntityForSelectedGeometry = useCallback((entityId: string, nextChecked: boolean) => { if (!selectedFeatures || selectedFeatures.length === 0) { flashEntityFormStatus("Chưa chọn geometry để bind entity."); return; } if (!isMultiEditValid) { flashEntityFormStatus("Không thể bind entity cho nhiều geometry khác loại."); return; } const id = String(entityId || "").trim(); if (!id) return; const nextEntityIds = (() => { const prev = selectedGeometryEntityIds; const has = prev.includes(id); if (nextChecked) { if (has) return prev; return uniqueEntityIds([...prev, id]); } if (!has) return prev; return prev.filter((x) => x !== id); })(); setIsEntitySubmitting(true); flashEntityFormStatus(null, 0); try { editor.patchFeaturePropertiesBatch( selectedFeatures.map((feature) => ({ id: feature.properties.id, patch: buildFeatureEntityPatch(feature, nextEntityIds, entities), })), nextChecked ? "Bind entity vào GEO" : "Unbind entity khỏi GEO" ); setSelectedGeometryEntityIds(nextEntityIds); flashEntityFormStatus( nextChecked ? "Đã bind entity vào geometry. Commit khi sẵn sàng." : "Đã unbind entity khỏi geometry. Commit khi sẵn sàng.", 3000 ); } finally { setIsEntitySubmitting(false); } }, [ editor, entities, flashEntityFormStatus, selectedFeatures, isMultiEditValid, selectedGeometryEntityIds, setIsEntitySubmitting, setSelectedGeometryEntityIds, ]); const handleToggleBindGeometryForSelectedGeometry = useCallback((geoId: string, nextChecked: boolean) => { if (!selectedFeatures || selectedFeatures.length === 0) { flashGeoBindingStatus("Chưa chọn geometry để bind."); return; } if (!isMultiEditValid) { flashGeoBindingStatus("Không thể bind geometry cho nhiều geometry khác loại."); return; } const id = String(geoId || "").trim(); if (!id) return; if (selectedFeatures.some(f => String(f.properties.id) === id)) return; setIsEntitySubmitting(true); flashGeoBindingStatus(null, 0); try { const bindingPatches = selectedFeatures.map((feature) => { const prevBindingIds = normalizeFeatureBindingIds(feature); const has = prevBindingIds.includes(id); const nextBindingIds = (() => { if (nextChecked) { if (has) return prevBindingIds; return [...prevBindingIds, id]; } if (!has) return prevBindingIds; return prevBindingIds.filter((x) => x !== id); })(); return { id: feature.properties.id, patch: { binding: nextBindingIds }, }; }); editor.patchFeaturePropertiesBatch( bindingPatches, nextChecked ? "Bind geometry vào GEO" : "Unbind geometry khỏi GEO" ); // Assume selectedFeature (the first one) reflects the representative binding in UI const firstFeaturePrevBindings = normalizeFeatureBindingIds(selectedFeatures[0]); const firstFeatureHas = firstFeaturePrevBindings.includes(id); const nextBindingIdsForUI = (() => { if (nextChecked) return firstFeatureHas ? firstFeaturePrevBindings : [...firstFeaturePrevBindings, id]; return firstFeatureHas ? firstFeaturePrevBindings.filter(x => x !== id) : firstFeaturePrevBindings; })(); setGeometryMetaForm((prev) => ({ ...prev, binding: nextBindingIdsForUI.join(", ") })); flashGeoBindingStatus( nextChecked ? "Đã bind geometry vào binding. Commit khi sẵn sàng." : "Đã gỡ binding geometry. Commit khi sẵn sàng.", 3000 ); } finally { setIsEntitySubmitting(false); } }, [ editor, flashGeoBindingStatus, selectedFeatures, isMultiEditValid, setGeometryMetaForm, setIsEntitySubmitting, ]); const handleFocusGeometryFromBindingPanel = useCallback((geoId: string) => { const id = String(geoId || "").trim(); if (!id) return; const feature = editor.draft.features.find((item) => String(item.properties.id) === id) || null; if (!feature) { flashGeoBindingStatus("Không tìm thấy geometry để zoom."); return; } const visibleInCurrentTimeline = timelineVisibleDraft.features.some( (item) => String(item.properties.id) === id ); if (timelineFilterEnabled && !visibleInCurrentTimeline) { setTimelineFilterEnabled(false); } setSelectedFeatureIds([feature.properties.id]); setGeometryFocusRequest((prev) => ({ key: (prev?.key ?? 0) + 1, collection: { type: "FeatureCollection", features: [feature], }, })); }, [ editor.draft.features, flashGeoBindingStatus, setGeometryFocusRequest, setSelectedFeatureIds, setTimelineFilterEnabled, timelineFilterEnabled, timelineVisibleDraft.features, ]); const handleAddWikiRefToProject = useCallback((wiki: Wiki) => { const id = String(wiki.id || "").trim(); if (!id) return; const title = (wiki.title || "").trim() || "Untitled wiki"; editor.setSnapshotWikis((prev) => { if (prev.some((w) => w.id === id)) return prev; return [ { id, source: "ref", operation: "reference", title, doc: null, }, ...prev, ]; }, `Thêm wiki ref #${id}`); setRequestedActiveWikiId(id); }, [editor, setRequestedActiveWikiId]); const handleImportGeoFromSearch = useCallback(( entityItem: EntityGeometriesSearchItem, geo: EntityGeometrySearchGeo ) => { const geoId = String(geo?.id || "").trim(); if (!geoId) return; // Ensure the geometry stays selectable even if it doesn't match the current timeline year. setTimelineFilterEnabled(false); const importedEntity: Entity = { id: entityItem.entity_id, name: (entityItem.name || "").trim() || entityItem.entity_id, description: (entityItem.description || "").trim() || null, geometry_count: 0, }; const existing = editor.draft.features.find((f) => String(f.properties.id) === geoId) || null; if (existing) { // Keep entity store consistent: importing/selecting a geo implies the entity should exist in snapshot + catalog. handleAddEntityRefToProject(importedEntity); setSelectedFeatureIds([existing.properties.id]); flashEntityFormStatus("Đã chọn geometry từ kết quả search.", 3000); return; } const geometry = normalizeGeoSearchGeometry(geo.draw_geometry); if (!geometry) { flashEntityFormStatus("Không import được: draw_geometry không hợp lệ.", 3000); return; } const bindingIds = normalizeGeoSearchBindingIds(geo.binding); const typeKey = geo.type || null; const feature: Feature = { type: "Feature", properties: { id: geoId, type: typeKey, time_start: typeof geo.time_start === "number" ? geo.time_start : null, time_end: typeof geo.time_end === "number" ? geo.time_end : null, binding: bindingIds.length ? bindingIds : undefined, entity_id: entityItem.entity_id, entity_ids: [entityItem.entity_id], entity_name: (entityItem.name || "").trim() || entityItem.entity_id, entity_names: [(entityItem.name || "").trim() || entityItem.entity_id], }, geometry, }; editor.createFeatureWithSnapshotEntities( feature, (prev) => { if (prev.some((e) => String(e.id) === importedEntity.id)) return prev; return [ { id: importedEntity.id, source: "ref", operation: "reference", name: importedEntity.name, description: importedEntity.description ?? null, }, ...prev, ]; }, `Import GEO #${geoId}` ); setEntityCatalog((prev) => { const byId = new globalThis.Map(); for (const row of prev || []) { if (!row?.id) continue; byId.set(String(row.id), row); } byId.set(importedEntity.id, importedEntity); return Array.from(byId.values()); }); setSelectedFeatureIds([feature.properties.id]); flashEntityFormStatus("Đã import geometry từ search GEO. Commit khi sẵn sàng.", 3000); }, [ editor, flashEntityFormStatus, handleAddEntityRefToProject, setEntityCatalog, setSelectedFeatureIds, setTimelineFilterEnabled, ]); const featureCommands = useFeatureCommands({ editor, selectedFeatures, geometryMetaForm, setGeometryMetaForm, selectedGeometryEntityIds, setSelectedGeometryEntityIds, entities, setIsEntitySubmitting, setEntityFormStatus, }); const handleCreateEntityOnly = async () => { const name = entityForm.name.trim(); if (!name) { setEntityFormStatus("Tên entity là bắt buộc."); return; } const description = entityForm.description.trim() || null; const normalizedName = name.toLowerCase(); const duplicatedName = entities.some((entity) => entity.name.trim().toLowerCase() === normalizedName); if (duplicatedName) { setEntityFormStatus("Tên entity đã tồn tại."); return; } const entityId = buildClientEntityId(); const createdEntity: Entity = { id: entityId, name, description, geometry_count: 0, }; setIsEntitySubmitting(true); setEntityFormStatus(null); try { editor.setSnapshotEntities((prev) => { if (prev.some((e) => String(e.id) === entityId)) return prev; return [ { id: entityId, source: "inline", operation: "create", name, description, }, ...prev, ]; }, `Tạo entity #${entityId}`); localCreatedEntityIdsRef.current.add(entityId); setEntityCatalog((prev) => { const byId = new globalThis.Map(); for (const row of prev || []) { if (!row?.id) continue; byId.set(String(row.id), row); } byId.set(entityId, createdEntity); return Array.from(byId.values()); }); setEntityForm((prev) => ({ ...prev, name: "", description: "", })); setEntityStatus(null); setEntityFormStatus("Đã tạo entity mới (local). Commit khi sẵn sàng."); } finally { setIsEntitySubmitting(false); } }; const headCommit = projectState?.head_commit_id ? sectionCommits.find((commit) => commit.id === projectState.head_commit_id) || null : null; const handleCreateFeature = (feature: Feature) => { editor.createFeature(feature); setSelectedFeatureIds([feature.properties.id]); }; const mapLabelContextDraft = isReplayPreviewMode ? previewSession?.draft || EMPTY_FEATURE_COLLECTION : editor.draft; return (
{!isReplayEditMode && !isReplayPreviewMode ? ( <> { setLeftPanelWidth((prev) => clampNumber(prev + deltaX, 220, 520)); }} /> ) : isReplayEditMode ? ( <> setMode("select")} isPreviewPlaying={false} previewPlaybackSpeed={1} onPlayPreviewFromStart={() => openReplayPreview("start")} onPlayPreviewFromSelection={() => openReplayPreview("selection")} onStopPreview={() => {}} onResetPreview={() => {}} /> { setLeftPanelWidth((prev) => clampNumber(prev + deltaX, 220, 520)); }} /> ) : null} {blockedPendingSubmissionId ? (

Editor dang bi khoa

Project nay dang co submission o trang thai PENDING (id:{" "} {blockedPendingSubmissionId}). Theo BE moi, khi submission dang pending thi khong duoc tao submission/commit moi va khong duoc vao editor.
) : null} {!blockedPendingSubmissionId ? (
{isBackgroundVisibilityReady ? ( ) : (
)} {isReplayPreviewMode ? ( ) : null} {isReplayPreviewMode && replayPreview.sidebarOpen ? ( ) : null} {!isReplayPreviewMode || replayPreview.timelineVisible ? ( ) : null}
) : null} {!isReplayEditMode && !isReplayPreviewMode ? ( <> { // dragging handle (between map and right panel): moving right increases right panel width setRightPanelWidth((prev) => clampNumber(prev - deltaX, 260, 720)); }} /> { setSearchKind(next); setSearchQuery(""); setSearchQueryDraft(""); }} query={searchQuery} onQueryChange={setSearchQuery} onLocalQueryChange={setSearchQueryDraft} /> {searchKind === "entity" && searchQueryDraft.trim().length > 0 ? (
Entity Results
{isEntitySearchLoading ? "Searching…" : `${entitySearchResults.length} results`}
{entitySearchResults.slice(0, 8).map((e) => (
{e.name}
{e.id}
))} {!isEntitySearchLoading && entitySearchResults.length === 0 ? (
No results.
) : null}
) : null} {searchKind === "wiki" && searchQueryDraft.trim().length > 0 ? (
Wiki Results
{isWikiSearching ? "Searching…" : `${wikiSearchResults.length} results`}
{wikiSearchResults.slice(0, 8).map((w) => (
{(w.title || "").trim() || "Untitled wiki"}
{w.id}
))} {!isWikiSearching && wikiSearchResults.length === 0 ? (
No results.
) : null}
) : null} {searchKind === "geo" && searchQueryDraft.trim().length > 0 ? (
Geo Results
{isGeoSearching ? "Searching…" : `${geoSearchResults.length} entities`}
{geoSearchResults.slice(0, 6).map((item) => (
{item.name?.trim() || item.entity_id}
{item.entity_id}
{Array.isArray(item.geometries) ? item.geometries.length : 0} geos
{item.description?.trim() ? (
{item.description.trim()}
) : null} {Array.isArray(item.geometries) && item.geometries.length ? (
{item.geometries.map((geo) => (
#{geo.id}
type: {geo.type || "unknown"}{" "} {geo.time_start != null || geo.time_end != null ? `| time: ${geo.time_start ?? "?"} → ${geo.time_end ?? "?"}` : ""}
))}
) : (
No geometry linked.
)}
))} {!isGeoSearching && geoSearchResults.length === 0 ? (
No results.
) : null}
) : null} {selectedFeature ? ( setMode("replay", id)} /> ) : null}
} /> ) : isReplayEditMode ? ( <> { setRightPanelWidth((prev) => clampNumber(prev - deltaX, 260, 720)); }} /> String(id))} currentTimelineYear={timelineDraftYear} geometryChoices={geometryChoices} wikiChoices={wikiChoices} getCurrentMapViewState={getCurrentMapViewState} onMutateReplay={editor.mutateActiveReplay} /> ) : null}
); } function ResizeHandle({ onDrag, title, }: { onDrag: (deltaX: number) => void; title: string; }) { const handlePointerDown = (event: ReactPointerEvent) => { // Only horizontal resize event.preventDefault(); const startX = event.clientX; let lastX = startX; const onMove = (e: PointerEvent) => { const dx = e.clientX - lastX; if (dx !== 0) { onDrag(dx); lastX = e.clientX; } }; const onUp = () => { window.removeEventListener("pointermove", onMove); window.removeEventListener("pointerup", onUp); }; window.addEventListener("pointermove", onMove); window.addEventListener("pointerup", onUp); }; return (
); } function clampNumber(value: number, min: number, max: number): number { if (value < min) return min; if (value > max) return max; return value; } function formatCommitTitle(commit: ProjectCommit): string { return commit.edit_summary?.trim() || `Commit ${commit.id.slice(0, 8)}`; } function isFeatureVisibleAtYear(feature: Feature, year: number): boolean { const start = feature.properties.time_start; const end = feature.properties.time_end; if (typeof start === "number" && Number.isFinite(start) && year < start) return false; if (typeof end === "number" && Number.isFinite(end) && year > end) return false; return true; } function normalizeWikisForCompare(input: WikiSnapshot[] | null | undefined) { const list = Array.isArray(input) ? input : []; const normalized = list .filter((w) => w && typeof w.id === "string" && w.id.trim().length > 0) .filter((w) => { if (w.source === "ref") return true; if (w.operation === "create" || w.operation === "update" || w.operation === "delete") return true; const title = typeof w.title === "string" ? w.title.trim() : ""; const doc = typeof w.doc === "string" ? w.doc.trim() : ""; return title.length > 0 || (w.doc !== null && doc.length > 0); }) .map((w) => ({ id: w.id, source: w.source, title: typeof w.title === "string" ? w.title.trim() : "", slug: typeof w.slug === "string" ? w.slug : null, doc: w.doc === null ? null : typeof w.doc === "string" ? w.doc.trim() : null, })) .sort((a, b) => a.id.localeCompare(b.id)); return normalized; } function normalizeEntitiesForCompare(input: EntitySnapshot[] | null | undefined) { const list = Array.isArray(input) ? input : []; const normalized = list .filter((e) => e && (typeof e.id === "string" || typeof e.id === "number")) .map((e) => ({ id: String(e.id), source: e.source, name: typeof e.name === "string" ? e.name.trim() : "", description: e.description == null ? null : String(e.description), })) .sort((a, b) => a.id.localeCompare(b.id)); return normalized; } function normalizeEntityWikiLinksForCompare(input: Array<{ entity_id: string; wiki_id: string; operation?: string }> | null | undefined) { const list = Array.isArray(input) ? input : []; const normalized = list .filter((l) => l && typeof l.entity_id === "string" && typeof l.wiki_id === "string") .map((l) => ({ entity_id: l.entity_id, wiki_id: l.wiki_id, operation: l.operation === "delete" ? "delete" : "binding", })) .sort((a, b) => (a.entity_id + a.wiki_id).localeCompare(b.entity_id + b.wiki_id)); return normalized; } function normalizeReplaysForCompare(input: BattleReplay[] | null | undefined) { const list = Array.isArray(input) ? input : []; return list .filter((replay) => replay && typeof replay.geometry_id === "string" && replay.geometry_id.trim().length > 0) .map((replay) => ({ id: typeof replay.id === "string" ? replay.id : replay.geometry_id, geometry_id: replay.geometry_id, target_geometry_ids: normalizeReplayTargetGeometryIdsForCompare( replay.target_geometry_ids, replay.geometry_id ), detail: Array.isArray(replay.detail) ? replay.detail : [], })) .sort((a, b) => a.geometry_id.localeCompare(b.geometry_id)); } function normalizeReplayTargetGeometryIdsForCompare( input: string[] | null | undefined, geometryId: string ) { const orderedIds: string[] = []; const seen = new Set(); const pushId = (rawId: string | number | null | undefined) => { if (rawId == null) return; const id = String(rawId).trim(); if (!id || seen.has(id)) return; seen.add(id); orderedIds.push(id); }; pushId(geometryId); for (const rawId of input || []) pushId(rawId); return orderedIds; } function normalizeGeoSearchGeometry(value: unknown): Geometry | null { if (!value || typeof value !== "object") return null; const g = value as Record; if (typeof g.type !== "string") return null; if (!("coordinates" in g)) return null; return value as Geometry; } function normalizeGeoSearchBindingIds(value: unknown): string[] { if (!Array.isArray(value)) return []; const deduped: string[] = []; const seen = new Set(); for (const rawId of value) { if (typeof rawId !== "string" && typeof rawId !== "number") continue; const id = String(rawId).trim(); if (!id || seen.has(id)) continue; seen.add(id); deduped.push(id); } return deduped; }