dawarich/app/controllers/map/leaflet_controller.rb

113 lines
2.8 KiB
Ruby
Raw Normal View History

2024-05-23 14:12:23 -04:00
# frozen_string_literal: true
class Map::LeafletController < ApplicationController
include SafeTimestampParser
2024-05-23 14:12:23 -04:00
before_action :authenticate_user!
layout 'map', only: :index
2024-05-23 14:12:23 -04:00
def index
2025-07-04 14:09:06 -04:00
@points = filtered_points
@coordinates = build_coordinates
@tracks = build_tracks
@distance = calculate_distance
@start_at = parsed_start_at
@end_at = parsed_end_at
@years = years_range
@points_number = points_count
@features = DawarichSettings.features
0.36.0 (#1952) * Implement OmniAuth GitHub authentication * Fix omniauth GitHub scope to include user email access * Remove margin-bottom * Implement Google OAuth2 authentication * Implement OIDC authentication for Dawarich using omniauth_openid_connect gem. * Add patreon account linking and patron checking service * Update docker-compose.yml to use boolean values instead of strings * Add support for KML files * Add tests * Update changelog * Remove patreon OAuth integration * Move omniauthable to a concern * Update an icon in integrations * Update changelog * Update app version * Fix family location sharing toggle * Move family location sharing to its own controller * Update changelog * Implement basic tagging functionality for places, allowing users to categorize and label places with custom tags. * Add places management API and tags feature * Add some changes related to places management feature * Fix some tests * Fix sometests * Add places layer * Update places layer to use Leaflet.Control.Layers.Tree for hierarchical layer control * Rework tag form * Add hashtag * Add privacy zones to tags * Add notes to places and manage place tags * Update changelog * Update e2e tests * Extract tag serializer to its own file * Fix some tests * Fix tags request specs * Fix some tests * Fix rest of the tests * Revert some changes * Add missing specs * Revert changes in place export/import code * Fix some specs * Fix PlaceFinder to only consider global places when finding existing places * Fix few more specs * Fix visits creator spec * Fix last tests * Update place creating modal * Add home location based on "Home" tagged place * Save enabled tag layers * Some fixes * Fix bug where enabling place tag layers would trigger saving enabled layers, overwriting with incomplete data * Update migration to use disable_ddl_transaction! and add up/down methods * Fix tag layers restoration and filtering logic * Update OIDC auto-registration and email/password registration settings * Fix potential xss
2025-11-24 13:45:09 -05:00
@home_coordinates = current_user.home_place_coordinates
2024-05-23 14:12:23 -04:00
end
private
2025-07-04 14:09:06 -04:00
def filtered_points
points.where('timestamp >= ? AND timestamp <= ?', start_at, end_at)
end
2024-05-23 14:12:23 -04:00
2025-07-04 14:09:06 -04:00
def build_coordinates
2025-08-01 12:42:53 -04:00
@points.pluck(:lonlat, :battery, :altitude, :timestamp, :velocity, :id, :country_name, :track_id)
2025-07-04 14:09:06 -04:00
.map { |lonlat, *rest| [lonlat.y, lonlat.x, *rest.map(&:to_s)] }
2024-05-23 14:12:23 -04:00
end
2025-07-04 14:09:06 -04:00
def extract_track_ids
@coordinates.map { |coord| coord[8]&.to_i }.compact.uniq.reject(&:zero?)
end
2024-05-23 14:12:23 -04:00
2025-07-04 14:09:06 -04:00
def build_tracks
track_ids = extract_track_ids
2025-07-20 12:57:53 -04:00
TracksSerializer.new(current_user, track_ids).call
2024-05-23 14:12:23 -04:00
end
2025-07-04 14:09:06 -04:00
def calculate_distance
2026-01-05 18:25:36 -05:00
return 0 if @points.count(:id) < 2
# Use PostGIS window function for efficient distance calculation
# This is O(1) database operation vs O(n) Ruby iteration
sql = <<~SQL.squish
SELECT COALESCE(SUM(distance_m) / 1000.0, 0) as total_km FROM (
SELECT ST_Distance(
lonlat::geography,
LAG(lonlat::geography) OVER (ORDER BY timestamp)
) as distance_m
FROM points
WHERE user_id = :user_id
AND timestamp >= :start_at
AND timestamp <= :end_at
) distances
SQL
result = Point.connection.select_value(
ActiveRecord::Base.sanitize_sql_array([
sql,
{ user_id: current_user.id, start_at: start_at, end_at: end_at }
])
)
result&.to_f&.round || 0
2025-07-04 14:09:06 -04:00
end
def parsed_start_at
Time.zone.at(start_at)
end
def parsed_end_at
Time.zone.at(end_at)
end
def years_range
(parsed_start_at.year..parsed_end_at.year).to_a
end
def points_count
@coordinates.count
end
def start_at
return safe_timestamp(params[:start_at]) if params[:start_at].present?
2025-07-04 14:09:06 -04:00
return Time.zone.at(points.last.timestamp).beginning_of_day.to_i if points.any?
Time.zone.today.beginning_of_day.to_i
end
def end_at
return safe_timestamp(params[:end_at]) if params[:end_at].present?
2025-07-04 14:09:06 -04:00
return Time.zone.at(points.last.timestamp).end_of_day.to_i if points.any?
Time.zone.today.end_of_day.to_i
2024-05-23 14:12:23 -04:00
end
def points
params[:import_id] ? points_from_import : points_from_user
end
def points_from_import
current_user.imports.find(params[:import_id]).points.without_raw_data.order(timestamp: :asc)
end
def points_from_user
current_user.points.without_raw_data.order(timestamp: :asc)
end
2024-05-23 14:12:23 -04:00
end