"use client"; import { useCallback, useEffect, useMemo, useRef, type SetStateAction, type PointerEvent as ReactPointerEvent } from "react"; import { useParams, useRouter } from "next/navigation"; import { useShallow } from "zustand/react/shallow"; import Map 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 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 { 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 { useProjectCommands } from "@/uhm/lib/editor/project/useProjectCommands"; 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"; 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 { 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, hoveredGeometryId, 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, hoveredGeometryId: state.hoveredGeometryId, 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, status: typeof e.status === "number" ? e.status : 1, geometry_count: 0, })) .filter((e) => e.id.length > 0 && e.name.length > 0); }, [snapshotEntities]); const entities = useMemo( () => mergeEntitySearchResults(entityCatalog, snapshotEntitiesAsEntities), [entityCatalog, snapshotEntitiesAsEntities] ); 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]); // Timeline filter: only affects persisted snapshot features. // New features created in the current session remain visible regardless of time range. const timelineVisibleDraft = useMemo(() => { // Nếu ở mode replay, sử dụng replayDraft thay vì main draft const activeDraft = mode === "replay" ? editor.replayDraft : editor.mainDraft; if (!timelineFilterEnabled) return activeDraft; const year = clampYearToFixedRange(Math.trunc(timelineDraftYear)); return { ...activeDraft, features: activeDraft.features.filter((feature) => { if (!editor.hasPersistedFeature(feature.properties.id)) return true; return isFeatureVisibleAtYear(feature, year); }), }; }, [editor, mode, timelineDraftYear, timelineFilterEnabled]); 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 hoveredGeometryHighlight = useMemo(() => { if (!hoveredGeometryId) return null; const feature = editor.draft.features.find( (item) => String(item.properties.id) === hoveredGeometryId ); if (!feature) return null; return { type: "FeatureCollection", features: [feature], } as FeatureCollection; }, [editor.draft.features, hoveredGeometryId]); 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 sectionCommands = useProjectCommands({ editor, store: editorStoreApi, emptyFeatureCollection: EMPTY_FEATURE_COLLECTION, pendingSaveCount, }); const { openSectionForEditing, commitSection, submitCurrentSection, restoreCommit, } = sectionCommands; const setMode = useCallback((m: EditorMode, featureId?: string | number) => { 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); editor.switchReplayContext(triggerId, selectedFeatureIds); } else if (m !== "replay") { if (mode === "replay") { editor.closeReplayContext(); setSelectedFeatureIds([]); } setReplayFeatureId(null); setHideOutside(false); } internalSetMode(m); }, [internalSetMode, mode, editor, selectedFeatureIds, setHideOutside, setReplayFeatureId, setSelectedFeatureIds]); const effectiveGeometryVisibility = useMemo(() => { const visibility: Record = { ...geometryVisibility }; if (mode === "replay" && replayFeatureId) { // Ẩn chính geo được chọn làm replay (marker kịch bản) visibility[String(replayFeatureId)] = false; if (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; }, [geometryVisibility, mode, replayFeatureId, hideOutside, editor.draft.features]); const onToggleHideOutside = useCallback(() => { setHideOutside((prev) => !prev); }, [setHideOutside]); 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 handleTimelineYearChange = (nextYear: number) => { setTimelineDraftYear(clampYearToFixedRange(Math.trunc(nextYear))); }; 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, updated_at: wiki.updated_at, }, ...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, status: 1, 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, status: 1, 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, slug: null, description, status: 1, }, ...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]); }; return ( {mode !== "replay" ? ( <> { setLeftPanelWidth((prev) => clampNumber(prev + deltaX, 220, 520)); }} /> > ) : ( <> { setLeftPanelWidth((prev) => clampNumber(prev + deltaX, 220, 520)); }} /> > )} {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. Xoa submission pending de unlock router.push("/user/projects")} style={{ padding: "10px 12px", borderRadius: 6, border: "1px solid #334155", background: "#111827", color: "white", cursor: "pointer", }} > Quay lai danh sach projects ) : null} {!blockedPendingSubmissionId ? ( {isBackgroundVisibilityReady ? ( ) : ( )} {mode !== "replay" && ( )} ) : null} {mode !== "replay" ? ( <> { // 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} handleAddEntityRefToProject(e)} style={{ border: "none", background: "#111827", color: "#93c5fd", cursor: "pointer", borderRadius: 6, padding: "6px 8px", fontSize: 12, fontWeight: 700, }} title="Add entity ref to project snapshot" > Add ))} {!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} handleAddWikiRefToProject(w)} style={{ border: "none", background: "#111827", color: "#93c5fd", cursor: "pointer", borderRadius: 6, padding: "6px 8px", fontSize: 12, fontWeight: 700, }} title="Add wiki ref to project snapshot" > Add ))} {!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 ?? "?"}` : ""} handleImportGeoFromSearch(item, geo)} style={{ border: "none", background: "#111827", color: "#93c5fd", cursor: "pointer", borderRadius: 6, padding: "6px 8px", fontSize: 12, fontWeight: 700, flex: "0 0 auto", }} title="Import geometry into current editor draft" > Import ))} ) : ( No geometry linked. )} ))} {!isGeoSearching && geoSearchResults.length === 0 ? ( No results. ) : null} ) : null} {selectedFeature ? ( setMode("replay", id)} /> ) : null} } /> > ) : ( <> { setRightPanelWidth((prev) => clampNumber(prev - deltaX, 260, 720)); }} /> > )} ); } 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() : "", slug: typeof e.slug === "string" ? e.slug : null, description: e.description == null ? null : String(e.description), status: typeof e.status === "number" ? e.status : null, })) .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) => ({ geometry_id: replay.geometry_id, detail: Array.isArray(replay.detail) ? replay.detail : [], replay_features: normalizeReplayFeatureCollection(replay.replay_features), })) .sort((a, b) => a.geometry_id.localeCompare(b.geometry_id)); } function normalizeReplayFeatureCollection(input: FeatureCollection | null | undefined) { const features = Array.isArray(input?.features) ? input.features : []; return { type: "FeatureCollection" as const, features: features .filter((feature) => feature && feature.properties && (typeof feature.properties.id === "string" || typeof feature.properties.id === "number")) .map((feature) => ({ type: "Feature" as const, properties: { ...feature.properties, id: String(feature.properties.id), }, geometry: feature.geometry, })) .sort((a, b) => String(a.properties.id).localeCompare(String(b.properties.id))), }; } 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; }
{blockedPendingSubmissionId}