371 lines
13 KiB
Svelte
371 lines
13 KiB
Svelte
<script lang="ts">
|
|
import SettingsCard from '$lib/components/SettingsCard.svelte'
|
|
import Compass from '$lib/components/Compass.svelte'
|
|
import { imu } from '$lib/stores/imu'
|
|
import { Chart, registerables } from 'chart.js'
|
|
import { cubicOut } from 'svelte/easing'
|
|
import { slide } from 'svelte/transition'
|
|
import { onDestroy, onMount } from 'svelte'
|
|
import { socket, mpu } from '$lib/stores'
|
|
import { useFeatureFlags } from '$lib/stores/featureFlags'
|
|
import { Rotate3d } from '$lib/components/icons'
|
|
|
|
import { type IMUCalibrateData } from '$lib/platform_shared/message'
|
|
|
|
Chart.register(...registerables)
|
|
|
|
const features = useFeatureFlags()
|
|
let intervalId: ReturnType<typeof setInterval> | number
|
|
let isCalibrating = $state(false)
|
|
let calibrationResult = $state<IMUCalibrateData | null>(null)
|
|
|
|
let angleChartElement: HTMLCanvasElement = $state()!
|
|
let tempChartElement: HTMLCanvasElement = $state()!
|
|
let altitudeChartElement: HTMLCanvasElement = $state()!
|
|
let headingChartElement: HTMLCanvasElement = $state()!
|
|
|
|
let angleChart: Chart
|
|
let tempChart: Chart
|
|
let altitudeChart: Chart
|
|
let headingChart: Chart
|
|
|
|
const getChartColors = () => {
|
|
const style = getComputedStyle(document.body)
|
|
return {
|
|
primary: style.getPropertyValue('--color-primary'),
|
|
secondary: style.getPropertyValue('--color-secondary'),
|
|
accent: style.getPropertyValue('--color-accent'),
|
|
background: style.getPropertyValue('--color-background')
|
|
}
|
|
}
|
|
|
|
const createBaseChartConfig = (bgColor: string) => ({
|
|
maintainAspectRatio: false,
|
|
responsive: true,
|
|
plugins: {
|
|
legend: { display: true },
|
|
tooltip: { mode: 'index' as const, intersect: false }
|
|
},
|
|
elements: { point: { radius: 1 } },
|
|
scales: {
|
|
x: {
|
|
grid: { color: bgColor },
|
|
ticks: { color: bgColor },
|
|
display: false
|
|
},
|
|
y: {
|
|
type: 'linear' as const,
|
|
position: 'left' as const,
|
|
min: 0,
|
|
max: 10,
|
|
grid: { color: bgColor },
|
|
ticks: { color: bgColor },
|
|
border: { color: bgColor }
|
|
}
|
|
}
|
|
})
|
|
|
|
const initializeCharts = () => {
|
|
const colors = getChartColors()
|
|
const baseConfig = createBaseChartConfig(colors.background)
|
|
|
|
if (angleChartElement) {
|
|
angleChart = new Chart(angleChartElement, {
|
|
type: 'line',
|
|
data: {
|
|
datasets: [
|
|
{
|
|
label: 'x',
|
|
borderColor: colors.primary,
|
|
backgroundColor: colors.primary,
|
|
borderWidth: 2,
|
|
data: $imu.map(datapoint => datapoint.x),
|
|
yAxisID: 'y'
|
|
},
|
|
{
|
|
label: 'y',
|
|
borderColor: colors.secondary,
|
|
backgroundColor: colors.secondary,
|
|
borderWidth: 2,
|
|
data: $imu.map(datapoint => datapoint.y),
|
|
yAxisID: 'y'
|
|
},
|
|
{
|
|
label: 'z',
|
|
borderColor: colors.accent,
|
|
backgroundColor: colors.accent,
|
|
borderWidth: 2,
|
|
data: $imu.map(datapoint => datapoint.z),
|
|
yAxisID: 'y'
|
|
}
|
|
]
|
|
},
|
|
options: {
|
|
...baseConfig,
|
|
scales: {
|
|
...baseConfig.scales,
|
|
y: {
|
|
...baseConfig.scales.y,
|
|
title: {
|
|
display: true,
|
|
text: 'Angle [°]',
|
|
color: colors.background,
|
|
font: { size: 16, weight: 'bold' }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
if (tempChartElement) {
|
|
tempChart = new Chart(tempChartElement, {
|
|
type: 'line',
|
|
data: {
|
|
datasets: [
|
|
{
|
|
label: 'Barometer temperature',
|
|
borderColor: colors.secondary,
|
|
backgroundColor: colors.secondary,
|
|
borderWidth: 2,
|
|
data: $imu.map(datapoint => datapoint.bmpTemp),
|
|
yAxisID: 'y'
|
|
}
|
|
]
|
|
},
|
|
options: {
|
|
...baseConfig,
|
|
scales: {
|
|
...baseConfig.scales,
|
|
y: {
|
|
...baseConfig.scales.y,
|
|
title: {
|
|
display: true,
|
|
text: 'Temperature [C°]',
|
|
color: colors.background,
|
|
font: { size: 16, weight: 'bold' }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
if (altitudeChartElement) {
|
|
altitudeChart = new Chart(altitudeChartElement, {
|
|
type: 'line',
|
|
data: {
|
|
datasets: [
|
|
{
|
|
label: 'Altitude',
|
|
borderColor: colors.primary,
|
|
backgroundColor: colors.primary,
|
|
borderWidth: 2,
|
|
data: $imu.map(datapoint => datapoint.altitude),
|
|
yAxisID: 'y'
|
|
}
|
|
]
|
|
},
|
|
options: {
|
|
...baseConfig,
|
|
scales: {
|
|
...baseConfig.scales,
|
|
y: {
|
|
...baseConfig.scales.y,
|
|
title: {
|
|
display: true,
|
|
text: 'Altitude [M]',
|
|
color: colors.background,
|
|
font: { size: 16, weight: 'bold' }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
if (headingChartElement) {
|
|
headingChart = new Chart(headingChartElement, {
|
|
type: 'line',
|
|
data: {
|
|
datasets: [
|
|
{
|
|
label: 'Heading',
|
|
borderColor: colors.accent,
|
|
backgroundColor: colors.accent,
|
|
borderWidth: 2,
|
|
data: $imu.map(datapoint => datapoint.heading),
|
|
yAxisID: 'y'
|
|
}
|
|
]
|
|
},
|
|
options: {
|
|
...baseConfig,
|
|
scales: {
|
|
...baseConfig.scales,
|
|
y: {
|
|
...baseConfig.scales.y,
|
|
min: 0,
|
|
max: 360,
|
|
title: {
|
|
display: true,
|
|
text: 'Heading [°]',
|
|
color: colors.background,
|
|
font: { size: 16, weight: 'bold' }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
const updateChartData = (chart: Chart, data: number[]) => {
|
|
chart.data.labels = data
|
|
chart.data.datasets[0].data = data
|
|
chart.options.scales!.y!.min = Math.min(...data) - 1
|
|
chart.options.scales!.y!.max = Math.max(...data) + 1
|
|
chart.update('none')
|
|
}
|
|
|
|
const updateData = () => {
|
|
if ($features.imu && angleChart) {
|
|
const x = $imu.map(datapoint => datapoint.x)
|
|
const y = $imu.map(datapoint => datapoint.y)
|
|
const z = $imu.map(datapoint => datapoint.z)
|
|
|
|
angleChart.data.labels = Array.from({ length: $imu.length }, (_, i) => i + 1)
|
|
angleChart.data.datasets[0].data = x
|
|
angleChart.data.datasets[1].data = y
|
|
angleChart.data.datasets[2].data = z
|
|
|
|
const allValues = [...x, ...y, ...z]
|
|
angleChart.options.scales!.y!.min = Math.min(...allValues) - 1
|
|
angleChart.options.scales!.y!.max = Math.max(...allValues) + 1
|
|
angleChart.update('none')
|
|
}
|
|
|
|
if ($features.bmp && tempChart && altitudeChart) {
|
|
updateChartData(
|
|
tempChart,
|
|
$imu.map(datapoint => datapoint.bmpTemp)
|
|
)
|
|
updateChartData(
|
|
altitudeChart,
|
|
$imu.map(datapoint => datapoint.altitude)
|
|
)
|
|
}
|
|
|
|
if ($features.mag && headingChart) {
|
|
const headingData = $imu.map(datapoint => datapoint.heading)
|
|
headingChart.data.labels = headingData
|
|
headingChart.data.datasets[0].data = headingData
|
|
headingChart.update('none')
|
|
|
|
if ($imu.length > 0) {
|
|
mpu.set({ heading: $imu[$imu.length - 1].heading })
|
|
}
|
|
}
|
|
}
|
|
onMount(() => {
|
|
imu.listen()
|
|
initializeCharts()
|
|
intervalId = setInterval(updateData, 200)
|
|
})
|
|
|
|
onDestroy(() => {
|
|
imu.stop()
|
|
clearInterval(intervalId)
|
|
})
|
|
|
|
async function startCalibration() {
|
|
isCalibrating = true
|
|
calibrationResult = null
|
|
try {
|
|
const response = await socket.request({ imuCalibrateExecute: {} })
|
|
calibrationResult = response.imuCalibrateData ?? null
|
|
} finally {
|
|
isCalibrating = false
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<SettingsCard collapsible={false}>
|
|
{#snippet icon()}
|
|
<Rotate3d class="flex-shrink-0 mr-2 h-6 w-6 self-end" />
|
|
{/snippet}
|
|
{#snippet title()}
|
|
<span>IMU</span>
|
|
{/snippet}
|
|
|
|
<div class="flex items-center gap-2 mb-4">
|
|
<button
|
|
class="btn btn-sm btn-primary"
|
|
onclick={startCalibration}
|
|
disabled={isCalibrating || !$features.imu}
|
|
>
|
|
{#if isCalibrating}
|
|
<span class="loading loading-spinner loading-xs"></span>
|
|
Calibrating...
|
|
{:else}
|
|
Calibrate IMU
|
|
{/if}
|
|
</button>
|
|
{#if calibrationResult}
|
|
<span
|
|
class="badge"
|
|
class:badge-success={calibrationResult.success}
|
|
class:badge-error={!calibrationResult.success}
|
|
>
|
|
{calibrationResult.success ? 'Calibrated' : 'Failed'}
|
|
</span>
|
|
{/if}
|
|
</div>
|
|
|
|
{#if $features.imu}
|
|
<div class="w-full overflow-x-auto">
|
|
<div
|
|
class="flex w-full flex-col space-y-1 h-60"
|
|
transition:slide|local={{ duration: 300, easing: cubicOut }}
|
|
>
|
|
<canvas bind:this={angleChartElement}></canvas>
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
|
|
{#if $features.mag}
|
|
<div class="divider">Magnetometer</div>
|
|
<div class="flex flex-col lg:flex-row gap-4 items-center">
|
|
<Compass heading={$mpu.heading} />
|
|
<div class="flex-1 w-full overflow-x-auto">
|
|
<div
|
|
class="flex w-full flex-col space-y-1 h-60"
|
|
transition:slide|local={{ duration: 300, easing: cubicOut }}
|
|
>
|
|
<canvas bind:this={headingChartElement}></canvas>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
|
|
{#if $features.bmp}
|
|
<div class="divider">Barometer</div>
|
|
<div class="w-full overflow-x-auto">
|
|
<div
|
|
class="flex w-full flex-col space-y-1 h-60"
|
|
transition:slide|local={{ duration: 300, easing: cubicOut }}
|
|
>
|
|
<canvas bind:this={tempChartElement}></canvas>
|
|
</div>
|
|
</div>
|
|
<div class="w-full overflow-x-auto">
|
|
<div
|
|
class="flex w-full flex-col space-y-1 h-60"
|
|
transition:slide|local={{ duration: 300, easing: cubicOut }}
|
|
>
|
|
<canvas bind:this={altitudeChartElement}></canvas>
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
</SettingsCard>
|