refactor: updating outline

This commit is contained in:
2025-08-22 14:01:44 +05:30
parent 0b7ef13a81
commit c78f9edb5c
5 changed files with 133 additions and 72 deletions

View File

@@ -75,13 +75,13 @@ function MainScene() {
clearComparisonProduct(); clearComparisonProduct();
setIsVersionSaved(false); setIsVersionSaved(false);
} }
}, [activeModule]) }, [activeModule, clearComparisonProduct, setIsVersionSaved])
useEffect(() => { useEffect(() => {
if (versionHistory.length > 0) { if (versionHistory.length > 0) {
setSelectedVersion(versionHistory[0]) setSelectedVersion(versionHistory[0])
} }
}, [versionHistory]) }, [setSelectedVersion, versionHistory])
const handleSelectVersion = (option: string) => { const handleSelectVersion = (option: string) => {
const version = versionHistory.find((version) => version.versionName === option); const version = versionHistory.find((version) => version.versionName === option);
@@ -140,7 +140,7 @@ function MainScene() {
{!selectedUser && ( {!selectedUser && (
<> <>
<KeyPressListener /> <KeyPressListener />
{/* {loadingProgress > 0 && <LoadingPage progress={loadingProgress} />} */} {loadingProgress > 0 && <LoadingPage progress={loadingProgress} />}
{!isPlaying && ( {!isPlaying && (
<> <>
{toggleThreeD && !isVersionSaved && <ModuleToggle />} {toggleThreeD && !isVersionSaved && <ModuleToggle />}
@@ -155,7 +155,7 @@ function MainScene() {
)} )}
{(isPlaying) && {(isPlaying) &&
activeModule === "simulation" && activeModule === "simulation" &&
loadingProgress == 0 && <SimulationPlayer />} loadingProgress === 0 && <SimulationPlayer />}
{(isPlaying) && {(isPlaying) &&
activeModule !== "simulation" && <ControlsPlayer />} activeModule !== "simulation" && <ControlsPlayer />}
@@ -188,7 +188,7 @@ function MainScene() {
} }
onDragOver={(event) => event.preventDefault()} onDragOver={(event) => event.preventDefault()}
> >
{/* <Scene layout="Main Layout" /> */} <Scene layout="Main Layout" />
</div> </div>
{selectedProduct && selectedVersion && isVersionSaved && !isPlaying && activeModule === "simulation" && ( {selectedProduct && selectedVersion && isVersionSaved && !isPlaying && activeModule === "simulation" && (

View File

@@ -4,54 +4,92 @@ import DropDownList from "../../ui/list/DropDownList";
import { useSceneContext } from "../../../modules/scene/sceneContext"; import { useSceneContext } from "../../../modules/scene/sceneContext";
import { isPointInsidePolygon } from "../../../functions/isPointInsidePolygon"; import { isPointInsidePolygon } from "../../../functions/isPointInsidePolygon";
interface AssetData {
id: string;
name: string;
position?: [];
rotation?: {};
}
interface ZoneData { interface ZoneData {
id: string; id: string;
name: string; name: string;
assets: { id: string; name: string; position?: []; rotation?: {} }[]; assets: AssetData[];
} }
const Outline: React.FC = () => { const Outline: React.FC = () => {
const [searchValue, setSearchValue] = useState<string>(""); const [searchValue, setSearchValue] = useState<string>("");
const [zoneDataList, setZoneDataList] = useState<ZoneData[]>([]); const [sceneAssetsDataList, setSceneAssetsDataList] = useState<
const [buildingsList, setBuildingsList] = useState<{ id: string; name: string }[]>([]); ZoneData[] | AssetData[]
const [isLayersOpen, setIsLayersOpen] = useState(true); >([]);
const [isBuildingsOpen, setIsBuildingsOpen] = useState(false); // const [buildingsList, setBuildingsList] = useState<{ id: string; name: string }[]>([]);
const [isZonesOpen, setIsZonesOpen] = useState(false); // const [isLayersOpen, setIsLayersOpen] = useState(true);
// const [isBuildingsOpen, setIsBuildingsOpen] = useState(false);
const [isZonesOpen, setIsZonesOpen] = useState(true);
const { assetStore, zoneStore } = useSceneContext(); const { assetStore, zoneStore } = useSceneContext();
const { assets } = assetStore(); const { assets } = assetStore();
const { zones } = zoneStore(); const { zones } = zoneStore();
useEffect(() => { useEffect(() => {
const updatedZoneList: ZoneData[] = zones?.map((zone: any) => { const assignedAssets = new Set<string>();
const polygon2D = zone.points.map((p: any) => [p.position[0], p.position[2]]);
const assetsInZone = assets.filter((item: any) => {
const [x, , z] = item.position;
return isPointInsidePolygon([x, z], polygon2D as [number, number][]);
})
.map((item: any) => ({
id: item.modelUuid,
name: item.modelName,
position: item.position,
rotation: item.rotation,
}));
const updatedZoneList: ZoneData[] =
return { zones?.map((zone: any) => {
id: zone.zoneUuid, const polygon2D = zone.points.map((p: any) => [
name: zone.zoneName, p.position[0],
assets: assetsInZone, p.position[2],
}; ]);
}); const assetsInZone = assets
.filter((item: any) => {
const [x, , z] = item.position;
const inside = isPointInsidePolygon(
[x, z],
polygon2D as [number, number][]
);
if (inside) assignedAssets.add(item.modelUuid);
return inside;
})
.map((item: any) => ({
id: item.modelUuid,
name: item.modelName,
position: item.position,
rotation: item.rotation,
}));
setZoneDataList(updatedZoneList); return {
id: zone.zoneUuid,
name: zone.zoneName,
assets: assetsInZone,
};
}) ?? [];
// Collect unassigned assets
const unassignedAssets = assets
.filter((item: any) => !assignedAssets.has(item.modelUuid))
.map((item: any) => ({
id: item.modelUuid,
name: item.modelName,
position: item.position,
rotation: item.rotation,
}));
// Add as a separate "zone"
if (unassignedAssets.length > 0) {
updatedZoneList.push({
id: "unassigned-zone",
name: "Unassigned",
assets: unassignedAssets,
});
}
setSceneAssetsDataList(updatedZoneList);
}, [zones, assets]); }, [zones, assets]);
const handleSearchChange = (value: string) => { const handleSearchChange = (value: string) => {
setSearchValue(value); setSearchValue(value);
}; };
const dropdownItems = [{ id: "1", name: "Ground Floor" }]; // const dropdownItems = [{ id: "1", name: "Ground Floor" }];
return ( return (
<div className="outline-container"> <div className="outline-container">
@@ -63,7 +101,7 @@ const Outline: React.FC = () => {
</div> </div>
) : ( ) : (
<div className="outline-content-container"> <div className="outline-content-container">
<section className="outline-section"> {/* <section className="outline-section">
<DropDownList <DropDownList
value="Layers" value="Layers"
items={dropdownItems} items={dropdownItems}
@@ -73,19 +111,19 @@ const Outline: React.FC = () => {
showFocusIcon={true} showFocusIcon={true}
remove remove
/> />
</section> </section> */}
<section className="outline-section overflow"> <section className="outline-section overflow">
<DropDownList {/* <DropDownList
value="Buildings" value="Buildings"
items={buildingsList} items={buildingsList}
isOpen={isBuildingsOpen} isOpen={isBuildingsOpen}
onToggle={() => setIsBuildingsOpen((prev) => !prev)} onToggle={() => setIsBuildingsOpen((prev) => !prev)}
showKebabMenu={false} showKebabMenu={false}
showAddIcon={false} showAddIcon={false}
/> /> */}
<DropDownList <DropDownList
value="Zones" value="Zones"
items={zoneDataList} items={sceneAssetsDataList}
isOpen={isZonesOpen} isOpen={isZonesOpen}
onToggle={() => setIsZonesOpen((prev) => !prev)} onToggle={() => setIsZonesOpen((prev) => !prev)}
showKebabMenu={false} showKebabMenu={false}

View File

@@ -1,9 +1,9 @@
import React, { useState } from 'react' import { useState } from 'react'
import NavigateCatagory from '../../NavigateCatagory' // import NavigateCatagory from '../../NavigateCatagory'
import { EyeIcon, ForkLiftIcon, KebabIcon, LocationPinIcon, RightHalfFillCircleIcon } from '../../../../../icons/ExportCommonIcons'; import { EyeIcon, ForkLiftIcon, KebabIcon, LocationPinIcon, RightHalfFillCircleIcon } from '../../../../../icons/ExportCommonIcons';
import assetImage from "../../../../../../assets/image/asset-image.png" import assetImage from "../../../../../../assets/image/asset-image.png"
const AssetManagement = () => { const AssetManagement = () => {
const [selectedCategory, setSelectedCategory] = useState("All Assets"); // const [selectedCategory, setSelectedCategory] = useState("All Assets");
const [expandedAssetId, setExpandedAssetId] = useState<string | null>(null); const [expandedAssetId, setExpandedAssetId] = useState<string | null>(null);
const dummyAssets = [ const dummyAssets = [
@@ -56,7 +56,7 @@ const AssetManagement = () => {
<div className='assetManagement-container assetManagement-wrapper'> <div className='assetManagement-container assetManagement-wrapper'>
{dummyAssets.map((asset, index) => ( {dummyAssets.map((asset, index) => (
<div className={`assetManagement-card-wrapper ${expandedAssetId === asset.id ? "openViewMore" : ""}`}> <div className={`assetManagement-card-wrapper ${expandedAssetId === asset.id ? "openViewMore" : ""}`} key={index}>
<header> <header>
<div className="header-wrapper"> <div className="header-wrapper">

View File

@@ -12,9 +12,7 @@ import {
LockIcon, LockIcon,
RemoveIcon, RemoveIcon,
} from "../../icons/ExportCommonIcons"; } from "../../icons/ExportCommonIcons";
import { import { useZoneAssetId } from "../../../store/builder/store";
useZoneAssetId,
} from "../../../store/builder/store";
import { zoneCameraUpdate } from "../../../services/visulization/zone/zoneCameraUpdation"; import { zoneCameraUpdate } from "../../../services/visulization/zone/zoneCameraUpdation";
import { setAssetsApi } from "../../../services/factoryBuilder/asset/floorAsset/setAssetsApi"; import { setAssetsApi } from "../../../services/factoryBuilder/asset/floorAsset/setAssetsApi";
import { useParams } from "react-router-dom"; import { useParams } from "react-router-dom";
@@ -47,7 +45,9 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
const { zoneAssetId, setZoneAssetId } = useZoneAssetId(); const { zoneAssetId, setZoneAssetId } = useZoneAssetId();
const { setSubModule } = useSubModuleStore(); const { setSubModule } = useSubModuleStore();
const [expandedZones, setExpandedZones] = useState<Record<string, boolean>>({}); const [expandedZones, setExpandedZones] = useState<Record<string, boolean>>(
{}
);
const { projectId } = useParams(); const { projectId } = useParams();
const { assetStore } = useSceneContext(); const { assetStore } = useSceneContext();
const { setName } = assetStore(); const { setName } = assetStore();
@@ -55,7 +55,7 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
const { selectedVersionStore } = useVersionContext(); const { selectedVersionStore } = useVersionContext();
const { selectedVersion } = selectedVersionStore(); const { selectedVersion } = selectedVersionStore();
const { zoneStore } = useSceneContext(); const { zoneStore } = useSceneContext();
const { zones, setZoneName } = zoneStore() const { zones, setZoneName } = zoneStore();
useEffect(() => { useEffect(() => {
useSelectedZoneStore.getState().setSelectedZone({ useSelectedZoneStore.getState().setSelectedZone({
@@ -78,14 +78,24 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
}; };
async function handleSelectZone(id: string) { async function handleSelectZone(id: string) {
try { try {
if (selectedZone?.zoneUuid === id) { if (selectedZone?.zoneUuid === id || id === 'unassigned-zone') {
return; return;
} }
setSubModule("zoneProperties"); setSubModule("zoneProperties");
let response = await getZoneData(id, organization, projectId, selectedVersion?.versionId || ""); let response = await getZoneData(
id,
organization,
projectId,
selectedVersion?.versionId || ""
);
console.log("response: ", response?.zoneName);
if (!response) return;
setSelectedZone({ setSelectedZone({
zoneName: response?.zoneName, zoneName: response?.zoneName,
activeSides: response?.activeSides ?? [], activeSides: response?.activeSides ?? [],
@@ -121,10 +131,15 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
zoneUuid: selectedZone.zoneUuid, zoneUuid: selectedZone.zoneUuid,
zoneName: newName, zoneName: newName,
}; };
const response = await zoneCameraUpdate(zonesdata, organization, projectId, selectedVersion?.versionId || ""); const response = await zoneCameraUpdate(
zonesdata,
organization,
projectId,
selectedVersion?.versionId || ""
);
if (response.message === "zone updated") { if (response.message === "zone updated") {
setSelectedZone((prev) => ({ ...prev, zoneName: newName })); setSelectedZone((prev) => ({ ...prev, zoneName: newName }));
setZoneName(selectedZone.zoneUuid, newName) setZoneName(selectedZone.zoneUuid, newName);
// setZones((prevZones: any[]) => // setZones((prevZones: any[]) =>
// prevZones.map((zone) => // prevZones.map((zone) =>
// zone.zoneUuid === selectedZone.zoneUuid // zone.zoneUuid === selectedZone.zoneUuid
@@ -140,7 +155,7 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
let response = await setAssetsApi({ let response = await setAssetsApi({
modelUuid: zoneAssetId.id, modelUuid: zoneAssetId.id,
modelName: newName, modelName: newName,
projectId projectId,
}); });
// console.log("response: ", response); // console.log("response: ", response);
@@ -159,11 +174,15 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
let drag = false; let drag = false;
let isLeftMouseDown = false; let isLeftMouseDown = false;
const contextClassNames = ["list-wrapper", "zone-properties-container", "list-container"]; const contextClassNames = [
"list-wrapper",
"zone-properties-container",
"list-container",
];
const isOutsideClick = (target: EventTarget | null) => { const isOutsideClick = (target: EventTarget | null) => {
if (!(target instanceof HTMLElement)) return true; if (!(target instanceof HTMLElement)) return true;
return !contextClassNames.some(className => return !contextClassNames.some((className) =>
target.closest(`.${className}`) target.closest(`.${className}`)
); );
}; };
@@ -189,38 +208,37 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
if (isOutsideClick(evt.target)) { if (isOutsideClick(evt.target)) {
// Clear selected zone // Clear selected zone
setSelectedZone({ setSelectedZone({
zoneUuid: '', zoneUuid: "",
zoneName: '', zoneName: "",
activeSides: [], activeSides: [],
panelOrder: [], panelOrder: [],
lockedPanels: [], lockedPanels: [],
widgets: [], widgets: [],
zoneViewPortTarget: [], zoneViewPortTarget: [],
zoneViewPortPosition: [] zoneViewPortPosition: [],
}); });
setZoneAssetId({ setZoneAssetId({
id: '', id: "",
name: '', name: "",
}); });
setSubModule("properties") setSubModule("properties");
} }
} }
}; };
if (selectedZone.zoneName! === '' && activeModule === 'Builder') { if (selectedZone.zoneName! === "" && activeModule === "Builder") {
document.addEventListener('mousedown', onMouseDown); document.addEventListener("mousedown", onMouseDown);
document.addEventListener('mousemove', onMouseMove); document.addEventListener("mousemove", onMouseMove);
document.addEventListener('mouseup', onMouseUp); document.addEventListener("mouseup", onMouseUp);
} }
return () => { return () => {
document.removeEventListener('mousedown', onMouseDown); document.removeEventListener("mousedown", onMouseDown);
document.removeEventListener('mousemove', onMouseMove); document.removeEventListener("mousemove", onMouseMove);
document.removeEventListener('mouseup', onMouseUp); document.removeEventListener("mouseup", onMouseUp);
}; };
}, [selectedZone, activeModule]); }, [selectedZone, activeModule]);
return ( return (
<> <>
{items?.length > 0 ? ( {items?.length > 0 ? (
@@ -234,7 +252,11 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
toggleZoneExpansion(item.id); toggleZoneExpansion(item.id);
}} }}
> >
<div className={`list-item ${selectedZone.zoneUuid === item.id ? "active" : ""}`}> <div
className={`list-item ${
selectedZone.zoneUuid === item.id ? "active" : ""
}`}
>
<div className="zone-header"> <div className="zone-header">
<button className="value" id="zone-name"> <button className="value" id="zone-name">
<RenameInput <RenameInput
@@ -276,7 +298,9 @@ const List: React.FC<ListProps> = ({ items = [], remove }) => {
{item.assets.map((asset) => ( {item.assets.map((asset) => (
<li <li
key={`asset-${asset.id}`} key={`asset-${asset.id}`}
className={`list-container asset-item ${zoneAssetId?.id === asset.id ? "active" : ""}`} className={`list-container asset-item ${
zoneAssetId?.id === asset.id ? "active" : ""
}`}
> >
<div className="list-item"> <div className="list-item">
<button <button

View File

@@ -4,7 +4,6 @@ import {
useSocketStore, useSocketStore,
useOrganization, useOrganization,
useUserName, useUserName,
useSaveVersion,
useProjectName, useProjectName,
useActiveTool, useActiveTool,
} from "../store/builder/store"; } from "../store/builder/store";
@@ -33,7 +32,7 @@ const Project: React.FC = () => {
const { setUserName } = useUserName(); const { setUserName } = useUserName();
const { setOrganization } = useOrganization(); const { setOrganization } = useOrganization();
const { projectId } = useParams(); const { projectId } = useParams();
const { projectName, setProjectName } = useProjectName(); const { setProjectName } = useProjectName();
const { userId, email, organization, userName } = getUserData(); const { userId, email, organization, userName } = getUserData();
const { selectedUser } = useSelectedUserStore(); const { selectedUser } = useSelectedUserStore();
const { isLogListVisible } = useLogger(); const { isLogListVisible } = useLogger();