2024-05-23 14:12:23 -04:00
|
|
|
# frozen_string_literal: true
|
2024-05-30 05:50:12 -04:00
|
|
|
|
2025-12-06 14:34:49 -05:00
|
|
|
class Map::LeafletController < ApplicationController
|
2025-12-08 18:17:24 -05:00
|
|
|
include SafeTimestampParser
|
|
|
|
|
|
2024-05-23 14:12:23 -04:00
|
|
|
before_action :authenticate_user!
|
2025-08-19 17:32:06 -04:00
|
|
|
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
|
2025-09-03 17:57:38 -04:00
|
|
|
@features = DawarichSettings.features
|
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-07 13:48:14 -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
|
2025-12-08 18:17:24 -05:00
|
|
|
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
|
2025-12-08 18:17:24 -05:00
|
|
|
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
|
2024-09-08 11:25:35 -04:00
|
|
|
|
|
|
|
|
def points
|
|
|
|
|
params[:import_id] ? points_from_import : points_from_user
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def points_from_import
|
2024-10-16 09:25:22 -04:00
|
|
|
current_user.imports.find(params[:import_id]).points.without_raw_data.order(timestamp: :asc)
|
2024-09-08 11:25:35 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def points_from_user
|
2025-08-21 16:32:29 -04:00
|
|
|
current_user.points.without_raw_data.order(timestamp: :asc)
|
2024-09-08 11:25:35 -04:00
|
|
|
end
|
2024-05-23 14:12:23 -04:00
|
|
|
end
|