mirror of
https://github.com/Freika/dawarich.git
synced 2025-12-21 04:49:46 -06:00
* fix: move foreman to global gems to fix startup crash (#1971) * Update exporting code to stream points data to file in batches to red… (#1980) * Update exporting code to stream points data to file in batches to reduce memory usage * Update changelog * Update changelog * Feature/maplibre frontend (#1953) * Add a plan to use MapLibre GL JS for the frontend map rendering, replacing Leaflet * Implement phase 1 * Phases 1-3 + part of 4 * Fix e2e tests * Phase 6 * Implement fog of war * Phase 7 * Next step: fix specs, phase 7 done * Use our own map tiles * Extract v2 map logic to separate manager classes * Update settings panel on v2 map * Update v2 e2e tests structure * Reimplement location search in maps v2 * Update speed routes * Implement visits and places creation in v2 * Fix last failing test * Implement visits merging * Fix a routes e2e test and simplify the routes layer styling. * Extract js to modules from maps_v2_controller.js * Implement area creation * Fix spec problem * Fix some e2e tests * Implement live mode in v2 map * Update icons and panel * Extract some styles * Remove unused file * Start adding dark theme to popups on MapLibre maps * Make popups respect dark theme * Move v2 maps to maplibre namespace * Update v2 references to maplibre * Put place, area and visit info into side panel * Update API to use safe settings config method * Fix specs * Fix method name to config in SafeSettings and update usages accordingly * Add missing public files * Add handling for real time points * Fix remembering enabled/disabled layers of the v2 map * Fix lots of e2e tests * Add settings to select map version * Use maps/v2 as main path for MapLibre maps * Update routing * Update live mode * Update maplibre controller * Update changelog * Remove some console.log statements * Pull only necessary data for map v2 points * Feature/raw data archive (#2009) * 0.36.2 (#2007) * fix: move foreman to global gems to fix startup crash (#1971) * Update exporting code to stream points data to file in batches to red… (#1980) * Update exporting code to stream points data to file in batches to reduce memory usage * Update changelog * Update changelog * Feature/maplibre frontend (#1953) * Add a plan to use MapLibre GL JS for the frontend map rendering, replacing Leaflet * Implement phase 1 * Phases 1-3 + part of 4 * Fix e2e tests * Phase 6 * Implement fog of war * Phase 7 * Next step: fix specs, phase 7 done * Use our own map tiles * Extract v2 map logic to separate manager classes * Update settings panel on v2 map * Update v2 e2e tests structure * Reimplement location search in maps v2 * Update speed routes * Implement visits and places creation in v2 * Fix last failing test * Implement visits merging * Fix a routes e2e test and simplify the routes layer styling. * Extract js to modules from maps_v2_controller.js * Implement area creation * Fix spec problem * Fix some e2e tests * Implement live mode in v2 map * Update icons and panel * Extract some styles * Remove unused file * Start adding dark theme to popups on MapLibre maps * Make popups respect dark theme * Move v2 maps to maplibre namespace * Update v2 references to maplibre * Put place, area and visit info into side panel * Update API to use safe settings config method * Fix specs * Fix method name to config in SafeSettings and update usages accordingly * Add missing public files * Add handling for real time points * Fix remembering enabled/disabled layers of the v2 map * Fix lots of e2e tests * Add settings to select map version * Use maps/v2 as main path for MapLibre maps * Update routing * Update live mode * Update maplibre controller * Update changelog * Remove some console.log statements --------- Co-authored-by: Robin Tuszik <mail@robin.gg> * Remove esbuild scripts from package.json * Remove sideEffects field from package.json * Raw data archivation * Add tests * Fix tests * Fix tests * Update ExceptionReporter * Add schedule to run raw data archival job monthly * Change file structure for raw data archival feature * Update changelog and version for raw data archival feature --------- Co-authored-by: Robin Tuszik <mail@robin.gg> * Set raw_data to an empty hash instead of nil when archiving * Fix storage configuration and file extraction * Consider MIN_MINUTES_SPENT_IN_CITY during stats calculation (#2018) * Consider MIN_MINUTES_SPENT_IN_CITY during stats calculation * Remove raw data from visited cities api endpoint * Use user timezone to show dates on maps (#2020) * Fix/pre epoch time (#2019) * Use user timezone to show dates on maps * Limit timestamps to valid range to prevent database errors when users enter pre-epoch dates. * Limit timestamps to valid range to prevent database errors when users enter pre-epoch dates. * Fix tests failing due to new index on stats table * Fix failing specs * Update redis client configuration to support unix socket connection * Update changelog * Fix kml kmz import issues (#2023) * Fix kml kmz import issues * Refactor KML importer to improve readability and maintainability * Implement moving points in map v2 and fix route rendering logic to ma… (#2027) * Implement moving points in map v2 and fix route rendering logic to match map v1. * Fix route spec * fix(maplibre): update date format to ISO 8601 (#2029) * Add verification step to raw data archival process (#2028) * Add verification step to raw data archival process * Add actual verification of raw data archives after creation, and only clear raw_data for verified archives. * Fix failing specs * Eliminate zip-bomb risk * Fix potential memory leak in js * Return .keep files * Use Toast instead of alert for notifications * Add help section to navbar dropdown * Update changelog * Remove raw_data_archival_job * Ensure file is being closed properly after reading in Archivable concern --------- Co-authored-by: Robin Tuszik <mail@robin.gg>
152 lines
4.3 KiB
JavaScript
152 lines
4.3 KiB
JavaScript
import { BaseLayer } from './base_layer'
|
|
|
|
/**
|
|
* Routes layer showing travel paths
|
|
* Connects points chronologically with solid color
|
|
*/
|
|
export class RoutesLayer extends BaseLayer {
|
|
constructor(map, options = {}) {
|
|
super(map, { id: 'routes', ...options })
|
|
this.maxGapHours = options.maxGapHours || 5 // Max hours between points to connect
|
|
}
|
|
|
|
getSourceConfig() {
|
|
return {
|
|
type: 'geojson',
|
|
data: this.data || {
|
|
type: 'FeatureCollection',
|
|
features: []
|
|
}
|
|
}
|
|
}
|
|
|
|
getLayerConfigs() {
|
|
return [
|
|
{
|
|
id: this.id,
|
|
type: 'line',
|
|
source: this.sourceId,
|
|
layout: {
|
|
'line-join': 'round',
|
|
'line-cap': 'round'
|
|
},
|
|
paint: {
|
|
// Use color from feature properties if available, otherwise default blue
|
|
'line-color': [
|
|
'case',
|
|
['has', 'color'],
|
|
['get', 'color'],
|
|
'#0000ff' // Default blue color (matching v1)
|
|
],
|
|
'line-width': 3,
|
|
'line-opacity': 0.8
|
|
}
|
|
}
|
|
]
|
|
}
|
|
|
|
/**
|
|
* Calculate haversine distance between two points in kilometers
|
|
* @param {number} lat1 - First point latitude
|
|
* @param {number} lon1 - First point longitude
|
|
* @param {number} lat2 - Second point latitude
|
|
* @param {number} lon2 - Second point longitude
|
|
* @returns {number} Distance in kilometers
|
|
*/
|
|
static haversineDistance(lat1, lon1, lat2, lon2) {
|
|
const R = 6371 // Earth's radius in kilometers
|
|
const dLat = (lat2 - lat1) * Math.PI / 180
|
|
const dLon = (lon2 - lon1) * Math.PI / 180
|
|
const a = Math.sin(dLat / 2) * Math.sin(dLat / 2) +
|
|
Math.cos(lat1 * Math.PI / 180) * Math.cos(lat2 * Math.PI / 180) *
|
|
Math.sin(dLon / 2) * Math.sin(dLon / 2)
|
|
const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a))
|
|
return R * c
|
|
}
|
|
|
|
/**
|
|
* Convert points to route LineStrings with splitting
|
|
* Matches V1's route splitting logic for consistency
|
|
* @param {Array} points - Points from API
|
|
* @param {Object} options - Splitting options
|
|
* @returns {Object} GeoJSON FeatureCollection
|
|
*/
|
|
static pointsToRoutes(points, options = {}) {
|
|
if (points.length < 2) {
|
|
return { type: 'FeatureCollection', features: [] }
|
|
}
|
|
|
|
// Default thresholds (matching V1 defaults from polylines.js)
|
|
const distanceThresholdKm = (options.distanceThresholdMeters || 500) / 1000
|
|
const timeThresholdMinutes = options.timeThresholdMinutes || 60
|
|
|
|
// Sort by timestamp
|
|
const sorted = points.slice().sort((a, b) => a.timestamp - b.timestamp)
|
|
|
|
// Split into segments based on distance and time gaps (like V1)
|
|
const segments = []
|
|
let currentSegment = [sorted[0]]
|
|
|
|
for (let i = 1; i < sorted.length; i++) {
|
|
const prev = sorted[i - 1]
|
|
const curr = sorted[i]
|
|
|
|
// Calculate distance between consecutive points
|
|
const distance = this.haversineDistance(
|
|
prev.latitude, prev.longitude,
|
|
curr.latitude, curr.longitude
|
|
)
|
|
|
|
// Calculate time difference in minutes
|
|
const timeDiff = (curr.timestamp - prev.timestamp) / 60
|
|
|
|
// Split if either threshold is exceeded (matching V1 logic)
|
|
if (distance > distanceThresholdKm || timeDiff > timeThresholdMinutes) {
|
|
if (currentSegment.length > 1) {
|
|
segments.push(currentSegment)
|
|
}
|
|
currentSegment = [curr]
|
|
} else {
|
|
currentSegment.push(curr)
|
|
}
|
|
}
|
|
|
|
if (currentSegment.length > 1) {
|
|
segments.push(currentSegment)
|
|
}
|
|
|
|
// Convert segments to LineStrings
|
|
const features = segments.map(segment => {
|
|
const coordinates = segment.map(p => [p.longitude, p.latitude])
|
|
|
|
// Calculate total distance for the segment
|
|
let totalDistance = 0
|
|
for (let i = 0; i < segment.length - 1; i++) {
|
|
totalDistance += this.haversineDistance(
|
|
segment[i].latitude, segment[i].longitude,
|
|
segment[i + 1].latitude, segment[i + 1].longitude
|
|
)
|
|
}
|
|
|
|
return {
|
|
type: 'Feature',
|
|
geometry: {
|
|
type: 'LineString',
|
|
coordinates
|
|
},
|
|
properties: {
|
|
pointCount: segment.length,
|
|
startTime: segment[0].timestamp,
|
|
endTime: segment[segment.length - 1].timestamp,
|
|
distance: totalDistance
|
|
}
|
|
}
|
|
})
|
|
|
|
return {
|
|
type: 'FeatureCollection',
|
|
features
|
|
}
|
|
}
|
|
}
|