small fixes for tools

This commit is contained in:
vcoppe
2025-10-24 20:07:15 +02:00
parent 9c83dcafa7
commit 6db8696a36
8 changed files with 241 additions and 196 deletions

View File

@@ -20,7 +20,7 @@
variant="outline" variant="outline"
class="whitespace-normal h-fit" class="whitespace-normal h-fit"
disabled={!validSelection} disabled={!validSelection}
onclick={async () => { onclick={() => {
if ($map) { if ($map) {
fileActions.addElevationToSelection($map); fileActions.addElevationToSelection($map);
} }

View File

@@ -13,7 +13,7 @@
} from '$lib/units'; } from '$lib/units';
import { CalendarDate, type DateValue } from '@internationalized/date'; import { CalendarDate, type DateValue } from '@internationalized/date';
import { CalendarClock, CirclePlay, CircleStop, CircleX, Timer, Zap } from '@lucide/svelte'; import { CalendarClock, CirclePlay, CircleStop, CircleX, Timer, Zap } from '@lucide/svelte';
import { tick } from 'svelte'; import { untrack } from 'svelte';
import { i18n } from '$lib/i18n.svelte'; import { i18n } from '$lib/i18n.svelte';
import { import {
ListFileItem, ListFileItem,
@@ -87,7 +87,7 @@
$effect(() => { $effect(() => {
if ($gpxStatistics && $velocityUnits && $distanceUnits) { if ($gpxStatistics && $velocityUnits && $distanceUnits) {
setGPXData(); untrack(() => setGPXData());
} }
}); });
@@ -204,7 +204,9 @@
min={0.01} min={0.01}
disabled={!canUpdate} disabled={!canUpdate}
bind:value={speed} bind:value={speed}
onchange={updateDataFromSpeed} onchange={() => {
untrack(() => updateDataFromSpeed());
}}
class="text-sm" class="text-sm"
/> />
<span class="text-sm shrink-0"> <span class="text-sm shrink-0">
@@ -221,7 +223,9 @@
bind:value={speed} bind:value={speed}
showHours={false} showHours={false}
disabled={!canUpdate} disabled={!canUpdate}
onChange={updateDataFromSpeed} onChange={() => {
untrack(() => updateDataFromSpeed());
}}
/> />
<span class="text-sm shrink-0"> <span class="text-sm shrink-0">
{#if $distanceUnits === 'imperial'} {#if $distanceUnits === 'imperial'}
@@ -243,7 +247,9 @@
<TimePicker <TimePicker
bind:value={movingTime} bind:value={movingTime}
disabled={!canUpdate} disabled={!canUpdate}
onChange={updateDataFromTotalTime} onChange={() => {
untrack(() => updateDataFromTotalTime());
}}
/> />
</div> </div>
</div> </div>
@@ -258,18 +264,19 @@
locale={i18n.lang} locale={i18n.lang}
placeholder={i18n._('toolbar.time.pick_date')} placeholder={i18n._('toolbar.time.pick_date')}
class="w-fit grow" class="w-fit grow"
onValueChange={async () => { onchange={() => {
await tick(); untrack(() => updateEnd());
updateEnd();
}} }}
/> />
<input <Input
type="time" type="time"
step={1} step={1}
disabled={!canUpdate} disabled={!canUpdate}
bind:value={startTime} bind:value={startTime}
class="w-fit" class="w-fit"
onchange={updateEnd} onchange={() => {
untrack(() => updateEnd());
}}
/> />
</div> </div>
<Label class="flex flex-row"> <Label class="flex flex-row">
@@ -283,18 +290,19 @@
locale={i18n.lang} locale={i18n.lang}
placeholder={i18n._('toolbar.time.pick_date')} placeholder={i18n._('toolbar.time.pick_date')}
class="w-fit grow" class="w-fit grow"
onValueChange={async () => { onchange={() => {
await tick(); untrack(() => updateStart());
updateStart();
}} }}
/> />
<input <Input
type="time" type="time"
step={1} step={1}
disabled={!canUpdate} disabled={!canUpdate}
bind:value={endTime} bind:value={endTime}
class="w-fit" class="w-fit"
onchange={updateStart} onchange={() => {
untrack(() => updateStart());
}}
/> />
</div> </div>
{#if $gpxStatistics.global.time.moving === 0 || $gpxStatistics.global.time.moving === undefined} {#if $gpxStatistics.global.time.moving === 0 || $gpxStatistics.global.time.moving === undefined}
@@ -400,15 +408,3 @@
{/if} {/if}
</Help> </Help>
</div> </div>
<style lang="postcss">
@reference "../../../../app.css";
div :global(input[type='time']) {
/*
Style copy-pasted from shadcn-svelte Input.
Needed to use native time input to avoid a bug with 2-level bind:value.
*/
@apply flex h-10 rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50;
}
</style>

View File

@@ -2,7 +2,7 @@
import { Label } from '$lib/components/ui/label/index.js'; import { Label } from '$lib/components/ui/label/index.js';
import { Button } from '$lib/components/ui/button'; import { Button } from '$lib/components/ui/button';
import { Slider } from '$lib/components/ui/slider'; import { Slider } from '$lib/components/ui/slider';
import { ListItem, ListRootItem } from '$lib/components/file-list/file-list'; import { ListRootItem } from '$lib/components/file-list/file-list';
import Help from '$lib/components/Help.svelte'; import Help from '$lib/components/Help.svelte';
import { Funnel } from '@lucide/svelte'; import { Funnel } from '@lucide/svelte';
import { i18n } from '$lib/i18n.svelte'; import { i18n } from '$lib/i18n.svelte';
@@ -10,13 +10,11 @@
import { onDestroy } from 'svelte'; import { onDestroy } from 'svelte';
import { getURLForLanguage } from '$lib/utils'; import { getURLForLanguage } from '$lib/utils';
import { selection } from '$lib/logic/selection'; import { selection } from '$lib/logic/selection';
import { minTolerance, ReducedGPXLayerCollection, tolerance } from './reduce'; import { minTolerance, ReducedGPXLayerCollection, tolerance } from './reduce.svelte';
let props: { class?: string } = $props(); let props: { class?: string } = $props();
let sliderValue = $state([50]); let sliderValue = $state([50]);
let maxPoints = $state(0);
let currentPoints = $state(0);
const maxTolerance = 10000; const maxTolerance = 10000;
let validSelection = $derived( let validSelection = $derived(
@@ -46,7 +44,7 @@
</Label> </Label>
<Label class="flex flex-row justify-between"> <Label class="flex flex-row justify-between">
<span>{i18n._('toolbar.reduce.number_of_points')}</span> <span>{i18n._('toolbar.reduce.number_of_points')}</span>
<span class="font-normal">{currentPoints}/{maxPoints}</span> <span class="font-normal">{reducedLayers.currentPoints}/{reducedLayers.maxPoints}</span>
</Label> </Label>
<Button variant="outline" disabled={!validSelection} onclick={() => reducedLayers.reduce()}> <Button variant="outline" disabled={!validSelection} onclick={() => reducedLayers.reduce()}>
<Funnel size="16" class="mr-1" /> <Funnel size="16" class="mr-1" />

View File

@@ -5,7 +5,7 @@ import { GPXFileStateCollectionObserver, type GPXFileState } from '$lib/logic/fi
import { selection } from '$lib/logic/selection'; import { selection } from '$lib/logic/selection';
import { ramerDouglasPeucker, TrackPoint, type SimplifiedTrackPoint } from 'gpx'; import { ramerDouglasPeucker, TrackPoint, type SimplifiedTrackPoint } from 'gpx';
import type { GeoJSONSource } from 'mapbox-gl'; import type { GeoJSONSource } from 'mapbox-gl';
import { get, writable } from 'svelte/store'; import { get, writable, type Writable } from 'svelte/store';
export const minTolerance = 0.1; export const minTolerance = 0.1;
@@ -53,14 +53,16 @@ export const tolerance = writable<number>(0);
export class ReducedGPXLayerCollection { export class ReducedGPXLayerCollection {
private _layers: Map<string, ReducedGPXLayer> = new Map(); private _layers: Map<string, ReducedGPXLayer> = new Map();
private _simplified: Map<string, [ListItem, number, SimplifiedTrackPoint[]]>; private _simplified: Map<string, [ListItem, number, SimplifiedTrackPoint[]]>;
private _fileStateCollectionOberver: GPXFileStateCollectionObserver; private _currentPoints = $state(0);
private _maxPoints = $state(0);
private _fileStateCollectionObserver: GPXFileStateCollectionObserver;
private _updateSimplified = this.updateSimplified.bind(this); private _updateSimplified = this.updateSimplified.bind(this);
private _unsubscribes: (() => void)[] = []; private _unsubscribes: (() => void)[] = [];
constructor() { constructor() {
this._layers = new Map(); this._layers = new Map();
this._simplified = new Map(); this._simplified = new Map();
this._fileStateCollectionOberver = new GPXFileStateCollectionObserver( this._fileStateCollectionObserver = new GPXFileStateCollectionObserver(
(newFiles) => { (newFiles) => {
newFiles.forEach((fileState, fileId) => { newFiles.forEach((fileState, fileId) => {
this._layers.set( this._layers.set(
@@ -96,8 +98,8 @@ export class ReducedGPXLayerCollection {
} }
update() { update() {
let maxPoints = 0; this._currentPoints = 0;
let currentPoints = 0; this._maxPoints = 0;
let data: GeoJSON.FeatureCollection = { let data: GeoJSON.FeatureCollection = {
type: 'FeatureCollection', type: 'FeatureCollection',
@@ -109,12 +111,12 @@ export class ReducedGPXLayerCollection {
return; return;
} }
maxPoints += maxPts; this._maxPoints += maxPts;
let current = points.filter( let current = points.filter(
(point) => point.distance === undefined || point.distance >= get(tolerance) (point) => point.distance === undefined || point.distance >= get(tolerance)
); );
currentPoints += current.length; this._currentPoints += current.length;
data.features.push({ data.features.push({
type: 'Feature', type: 'Feature',
@@ -173,8 +175,16 @@ export class ReducedGPXLayerCollection {
fileActions.reduce(itemsAndPoints); fileActions.reduce(itemsAndPoints);
} }
get currentPoints() {
return this._currentPoints;
}
get maxPoints() {
return this._maxPoints;
}
destroy() { destroy() {
this._fileStateCollectionOberver.destroy(); this._fileStateCollectionObserver.destroy();
this._unsubscribes.forEach((unsubscribe) => unsubscribe()); this._unsubscribes.forEach((unsubscribe) => unsubscribe());
const map_ = get(map); const map_ = get(map);

View File

@@ -15,7 +15,7 @@
Route, Route,
TriangleAlert, TriangleAlert,
ArrowRightLeft, ArrowRightLeft,
Home, House,
RouteOff, RouteOff,
Repeat, Repeat,
SquareArrowUpLeft, SquareArrowUpLeft,
@@ -231,7 +231,7 @@
} }
}} }}
> >
<Home size="12" />{i18n._('toolbar.routing.route_back_to_start.button')} <House size="12" />{i18n._('toolbar.routing.route_back_to_start.button')}
</ButtonWithTooltip> </ButtonWithTooltip>
<ButtonWithTooltip <ButtonWithTooltip
label={i18n._('toolbar.routing.round_trip.tooltip')} label={i18n._('toolbar.routing.round_trip.tooltip')}

View File

@@ -12,12 +12,14 @@
disabled = false, disabled = false,
locale, locale,
class: className = '', class: className = '',
onchange = () => {},
}: { }: {
value?: DateValue; value?: DateValue;
placeholder?: string; placeholder?: string;
disabled?: boolean; disabled?: boolean;
locale: string; locale: string;
class?: string; class?: string;
onchange?: (date: DateValue | undefined) => void;
} = $props(); } = $props();
const df = new DateFormatter(locale, { const df = new DateFormatter(locale, {
@@ -43,6 +45,6 @@
{value ? df.format(value.toDate(getLocalTimeZone())) : placeholder} {value ? df.format(value.toDate(getLocalTimeZone())) : placeholder}
</Popover.Trigger> </Popover.Trigger>
<Popover.Content bind:ref={contentRef} class="w-auto p-0"> <Popover.Content bind:ref={contentRef} class="w-auto p-0">
<Calendar type="single" captionLayout="dropdown" bind:value /> <Calendar type="single" captionLayout="dropdown" bind:value onValueChange={onchange} />
</Popover.Content> </Popover.Content>
</Popover.Root> </Popover.Root>

View File

@@ -1,26 +1,45 @@
<script lang="ts"> <script lang="ts">
import { Input } from '$lib/components/ui/input'; import { Input } from '$lib/components/ui/input';
export let value: string | number; let {
id,
value = $bindable(),
disabled,
oninput = () => {},
onchange = () => {},
onkeypress = () => {},
onfocusin = () => {},
class: className,
}: {
id: string;
value: string | number;
disabled?: boolean;
oninput?: (e: Event) => void;
onchange?: (e: Event) => void;
onkeypress?: (e: KeyboardEvent) => void;
onfocusin?: (e: FocusEvent) => void;
class?: string;
} = $props();
</script> </script>
<div> <div>
<Input <Input
{id}
type="text" type="text"
step={1} step={1}
bind:value bind:value
on:input {disabled}
on:change {oninput}
on:keypress {onchange}
on:focusin={() => { {onkeypress}
onfocusin={(e) => {
let input = document.activeElement; let input = document.activeElement;
if (input instanceof HTMLInputElement) { if (input instanceof HTMLInputElement) {
input.select(); input.select();
} }
onfocusin(e);
}} }}
on:focusin class="w-[22px] {className ?? ''}"
class="w-[22px] {$$props.class ?? ''}"
{...$$restProps}
/> />
</div> </div>
@@ -29,8 +48,11 @@
div :global(input) { div :global(input) {
@apply px-0.5; @apply px-0.5;
@apply py-0;
@apply bg-transparent;
@apply text-right; @apply text-right;
@apply border-none; @apply border-none;
@apply shadow-none;
@apply focus:ring-0; @apply focus:ring-0;
@apply focus:ring-offset-0; @apply focus:ring-offset-0;
@apply focus:outline-none; @apply focus:outline-none;

View File

@@ -1,14 +1,22 @@
<script lang="ts"> <script lang="ts">
import { untrack } from 'svelte';
import TimeComponentInput from './TimeComponentInput.svelte'; import TimeComponentInput from './TimeComponentInput.svelte';
export let showHours = true; let {
export let value: number | undefined = undefined; showHours = true,
export let disabled: boolean = false; value = $bindable(),
export let onChange = () => {}; disabled = false,
onChange = () => {},
}: {
showHours?: boolean;
value?: number;
disabled?: boolean;
onChange?: () => void;
} = $props();
let hours: string | number = '--'; let hours: string | number = $state('--');
let minutes: string | number = '--'; let minutes: string | number = $state('--');
let seconds: string | number = '--'; let seconds: string | number = $state('--');
function maybeParseInt(value: string | number): number { function maybeParseInt(value: string | number): number {
if (value === '--' || value === '') { if (value === '--' || value === '') {
@@ -17,24 +25,30 @@
return typeof value === 'string' ? parseInt(value) : value; return typeof value === 'string' ? parseInt(value) : value;
} }
function computeValue() { function computeValue(): number {
return Math.max( return Math.max(
maybeParseInt(hours) * 3600 + maybeParseInt(minutes) * 60 + maybeParseInt(seconds), maybeParseInt(hours) * 3600 + maybeParseInt(minutes) * 60 + maybeParseInt(seconds),
1 1
); );
} }
function updateValue() { $effect(() => {
value = computeValue(); const val = computeValue();
} untrack(() => {
value = val;
});
});
$: hours, minutes, seconds, updateValue(); $effect(() => {
if (value === undefined) {
$: if (value === undefined) { untrack(() => {
hours = '--'; hours = '--';
minutes = '--'; minutes = '--';
seconds = '--'; seconds = '--';
} else if (value !== computeValue()) { });
} else {
untrack(() => {
if (value != computeValue()) {
let rounded = Math.max(Math.round(value), 1); let rounded = Math.max(Math.round(value), 1);
if (showHours) { if (showHours) {
hours = Math.floor(rounded / 3600); hours = Math.floor(rounded / 3600);
@@ -46,6 +60,9 @@
} }
seconds = (rounded % 60).toString().padStart(2, '0'); seconds = (rounded % 60).toString().padStart(2, '0');
} }
});
}
});
let container: HTMLDivElement; let container: HTMLDivElement;
let countKeyPress = 0; let countKeyPress = 0;
@@ -65,7 +82,7 @@
<div <div
bind:this={container} bind:this={container}
class="flex flex-row items-center w-full min-w-fit border rounded-md px-3 focus-within:outline-none focus-within:ring-2 focus-within:ring-ring focus-within:ring-offset-2 {disabled class="h-9 flex flex-row items-center w-full min-w-fit border rounded-md px-3 focus-within:outline-none focus-within:ring-2 focus-within:ring-ring focus-within:ring-offset-2 {disabled
? 'opacity-50 cursor-not-allowed' ? 'opacity-50 cursor-not-allowed'
: ''}" : ''}"
> >
@@ -75,7 +92,7 @@
bind:value={hours} bind:value={hours}
{disabled} {disabled}
class="w-[30px]" class="w-[30px]"
on:input={() => { oninput={() => {
if (typeof hours === 'string') { if (typeof hours === 'string') {
hours = parseInt(hours); hours = parseInt(hours);
} }
@@ -87,8 +104,8 @@
} }
onChange(); onChange();
}} }}
on:keypress={onKeyPress} onkeypress={onKeyPress}
on:focusin={() => { onfocusin={() => {
countKeyPress = 0; countKeyPress = 0;
}} }}
/> />
@@ -98,7 +115,7 @@
id="minutes" id="minutes"
bind:value={minutes} bind:value={minutes}
{disabled} {disabled}
on:input={() => { oninput={() => {
if (typeof minutes === 'string') { if (typeof minutes === 'string') {
minutes = parseInt(minutes); minutes = parseInt(minutes);
} }
@@ -113,8 +130,8 @@
minutes = minutes.toString().padStart(showHours ? 2 : 1, '0'); minutes = minutes.toString().padStart(showHours ? 2 : 1, '0');
onChange(); onChange();
}} }}
on:keypress={onKeyPress} onkeypress={onKeyPress}
on:focusin={() => { onfocusin={() => {
countKeyPress = 0; countKeyPress = 0;
}} }}
/> />
@@ -123,7 +140,7 @@
id="seconds" id="seconds"
bind:value={seconds} bind:value={seconds}
{disabled} {disabled}
on:input={() => { oninput={() => {
if (typeof seconds === 'string') { if (typeof seconds === 'string') {
seconds = parseInt(seconds); seconds = parseInt(seconds);
} }
@@ -138,8 +155,8 @@
seconds = seconds.toString().padStart(2, '0'); seconds = seconds.toString().padStart(2, '0');
onChange(); onChange();
}} }}
on:keypress={onKeyPress} onkeypress={onKeyPress}
on:focusin={() => { onfocusin={() => {
countKeyPress = 0; countKeyPress = 0;
}} }}
/> />