Skip to content

Commit

Permalink
WIP
Browse files Browse the repository at this point in the history
  • Loading branch information
DTTerastar committed Dec 29, 2024
1 parent 7497073 commit d7cc99c
Show file tree
Hide file tree
Showing 9 changed files with 625 additions and 1 deletion.
2 changes: 1 addition & 1 deletion .clinerules
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
Test via `dotnet watch --project src`
Don't fix warnings and typescript issues unless it relates to your task
src/ui is svelete 5 please don't use old syntax
src/ui is svelete 5 please don't use old syntax
58 changes: 58 additions & 0 deletions src/ui/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions src/ui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
"@types/cookie": "^1.0.0",
"@types/d3": "^7.4.3",
"@types/node": "^22.9.1",
"@types/three": "^0.171.0",
"@typescript-eslint/eslint-plugin": "^8.15.0",
"@typescript-eslint/parser": "^8.15.0",
"autoprefixer": "^10.4.20",
Expand All @@ -41,6 +42,7 @@
"svelte-check": "^4.0.9",
"svelte-eslint-parser": "^0.43.0",
"svelte-table": "^0.6.3",
"three": "^0.171.0",
"tslib": "^2.8.1",
"typescript": "^5.6.3",
"typescript-eslint": "^8.18.2",
Expand Down
148 changes: 148 additions & 0 deletions src/ui/src/lib/3d/Devices.svelte
Original file line number Diff line number Diff line change
@@ -0,0 +1,148 @@
<script lang="ts">
import { onMount, onDestroy } from 'svelte';
import { devices } from '$lib/stores';
import * as THREE from 'three';
import { CSS2DObject } from 'three/examples/jsm/renderers/CSS2DRenderer.js';
import type { Group } from 'three';
import type { Device } from '$lib/types';
export let groupPivot: Group;
// Position adjustments
const X_POS_ADJ = 1.5;
const Y_POS_ADJ = 5;
// Animation settings
const PULSE_SPEED = 2; // Oscillations per second
const PULSE_MIN = 0.8;
const PULSE_MAX = 1.2;
const geoSphere = new THREE.SphereGeometry(0.2, 32, 16);
const trackerMaterials = [
new THREE.MeshStandardMaterial({
emissive: 0xff0000,
emissiveIntensity: 2,
transparent: true,
opacity: 0.8
}),
new THREE.MeshStandardMaterial({
emissive: 0xffbb00,
emissiveIntensity: 2,
transparent: true,
opacity: 0.8
}),
new THREE.MeshStandardMaterial({
emissive: 0xffee00,
emissiveIntensity: 2,
transparent: true,
opacity: 0.8
}),
];
let deviceGroup: THREE.Group | null = null;
let trackingSpheres: THREE.Mesh[] = [];
let trackerLabels: { [key: string]: HTMLDivElement } = {};
let startTime = performance.now();
$: if ($devices && groupPivot) {
updateDevices($devices);
}
function cleanupDeviceGroup() {
if (!deviceGroup) return;
deviceGroup.traverse(child => {
if ((child as any).geometry) {
(child as any).geometry.dispose();
}
if ((child as any).material) {
(child as any).material.dispose();
}
// Clean up CSS2D labels
if (child instanceof CSS2DObject) {
const element = child.element;
if (element && element.parentNode) {
element.parentNode.removeChild(element);
}
}
});
groupPivot.remove(deviceGroup);
deviceGroup = null;
trackingSpheres = [];
trackerLabels = {};
}
function updateDevices(devices: Device[]) {
cleanupDeviceGroup();
const newDeviceGroup = new THREE.Group();
newDeviceGroup.name = 'DeviceGroup';
devices.forEach(device => {
if (!device.location) return;
const trackName = device.id;
const confidence = device.confidence || 0;
const fixes = device.fixes || 0;
const position = device.location;
if (confidence <= 1) return;
const material = trackerMaterials[trackingSpheres.length % trackerMaterials.length];
const newSphere = new THREE.Mesh(geoSphere, material);
newSphere.name = trackName;
newSphere.position.set(position.x - X_POS_ADJ, position.y - Y_POS_ADJ, position.z);
trackingSpheres.push(newSphere);
newDeviceGroup.add(newSphere);
const labelDivEle = document.createElement('div');
labelDivEle.style.color = '#ffffff';
labelDivEle.style.fontFamily = 'Arial';
labelDivEle.style.fontSize = '0.8rem';
labelDivEle.style.marginTop = '-1em';
const labelDivLine1 = document.createElement('div');
const displayName = device.name || device.id;
labelDivLine1.textContent = displayName.length > 15 ? (displayName.substring(0, 14) + '...') : displayName;
const labelDivLine2 = document.createElement('div');
labelDivLine2.textContent = `${confidence}% (${fixes} fixes)`;
labelDivEle.append(labelDivLine1, labelDivLine2);
trackerLabels[trackName] = labelDivLine2;
const labelElement = new CSS2DObject(labelDivEle);
labelElement.name = trackName + '#label';
labelElement.position.set(position.x - X_POS_ADJ, position.y - Y_POS_ADJ, position.z);
newDeviceGroup.add(labelElement);
});
deviceGroup = newDeviceGroup;
groupPivot.add(deviceGroup);
}
// Update pulse scale based on time
function updatePulse() {
const elapsed = (performance.now() - startTime) / 1000;
const phase = (elapsed * PULSE_SPEED * Math.PI) % (Math.PI * 2);
const scale = PULSE_MIN + (Math.sin(phase) + 1) * (PULSE_MAX - PULSE_MIN) / 2;
trackingSpheres.forEach((sphere) => {
sphere.scale.set(scale, scale, scale);
});
}
// Export update function for parent to call during animation
export function update() {
if (deviceGroup) {
updatePulse();
}
}
onDestroy(() => {
cleanupDeviceGroup();
});
</script>
Loading

0 comments on commit d7cc99c

Please sign in to comment.