162 lines
3.6 KiB
Ruby
162 lines
3.6 KiB
Ruby
module GPX
|
|
class File
|
|
require "libxml"
|
|
|
|
include LibXML
|
|
|
|
attr_reader :possible_points
|
|
attr_reader :actual_points
|
|
attr_reader :tracksegs
|
|
|
|
def initialize(file)
|
|
@file = file
|
|
end
|
|
|
|
def points
|
|
@possible_points = 0
|
|
@actual_points = 0
|
|
@tracksegs = 0
|
|
|
|
@file.rewind
|
|
|
|
reader = XML::Reader.io(@file)
|
|
|
|
point = nil
|
|
|
|
while reader.read
|
|
if reader.node_type == XML::Reader::TYPE_ELEMENT
|
|
if reader.name == "trkpt"
|
|
point = TrkPt.new(@tracksegs, reader["lat"].to_f, reader["lon"].to_f)
|
|
@possible_points += 1
|
|
elsif reader.name == "ele" && point
|
|
point.altitude = reader.read_string.to_f
|
|
elsif reader.name == "time" && point
|
|
point.timestamp = DateTime.parse(reader.read_string)
|
|
end
|
|
elsif reader.node_type == XML::Reader::TYPE_END_ELEMENT
|
|
if reader.name == "trkpt" && point && point.valid?
|
|
point.altitude ||= 0
|
|
yield point
|
|
@actual_points += 1
|
|
elsif reader.name == "trkseg"
|
|
@tracksegs += 1
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def picture(min_lat, min_lon, max_lat, max_lon, num_points)
|
|
frames = 10
|
|
width = 250
|
|
height = 250
|
|
proj = OSM::Mercator.new(min_lat, min_lon, max_lat, max_lon, width, height)
|
|
|
|
linegc = Magick::Draw.new
|
|
linegc.stroke_linejoin('miter')
|
|
linegc.stroke_width(1)
|
|
linegc.stroke('#BBBBBB')
|
|
linegc.fill('#BBBBBB')
|
|
|
|
highlightgc = Magick::Draw.new
|
|
highlightgc.stroke_linejoin('miter')
|
|
highlightgc.stroke_width(3)
|
|
highlightgc.stroke('#000000')
|
|
highlightgc.fill('#000000')
|
|
|
|
images = frames.times.collect do
|
|
Magick::Image.new(width, height) do |image|
|
|
image.background_color = 'white'
|
|
image.format = 'GIF'
|
|
end
|
|
end
|
|
|
|
oldpx = 0.0
|
|
oldpy = 0.0
|
|
|
|
m = 0
|
|
mm = 0
|
|
points do |p|
|
|
px = proj.x(p.longitude)
|
|
py = proj.y(p.latitude)
|
|
|
|
if m > 0
|
|
frames.times do |n|
|
|
if n == mm
|
|
gc = highlightgc.dup
|
|
else
|
|
gc = linegc.dup
|
|
end
|
|
|
|
gc.line(px, py, oldpx, oldpy)
|
|
|
|
gc.draw(images[n])
|
|
end
|
|
end
|
|
|
|
m += 1
|
|
if m > num_points.to_f / frames.to_f * (mm + 1)
|
|
mm += 1
|
|
end
|
|
|
|
oldpy = py
|
|
oldpx = px
|
|
end
|
|
|
|
il = Magick::ImageList.new
|
|
|
|
images.each do |f|
|
|
il << f
|
|
end
|
|
|
|
il.delay = 50
|
|
il.format = 'GIF'
|
|
|
|
il.to_blob
|
|
end
|
|
|
|
def icon(min_lat, min_lon, max_lat, max_lon)
|
|
width = 50
|
|
height = 50
|
|
proj = OSM::Mercator.new(min_lat, min_lon, max_lat, max_lon, width, height)
|
|
|
|
gc = Magick::Draw.new
|
|
gc.stroke_linejoin('miter')
|
|
gc.stroke_width(1)
|
|
gc.stroke('#000000')
|
|
gc.fill('#000000')
|
|
|
|
image = Magick::Image.new(width, height) do |i|
|
|
i.background_color = 'white'
|
|
i.format = 'GIF'
|
|
end
|
|
|
|
oldpx = 0.0
|
|
oldpy = 0.0
|
|
|
|
first = true
|
|
|
|
points do |p|
|
|
px = proj.x(p.longitude)
|
|
py = proj.y(p.latitude)
|
|
|
|
gc.dup.line(px, py, oldpx, oldpy).draw(image) unless first
|
|
|
|
first = false
|
|
oldpy = py
|
|
oldpx = px
|
|
end
|
|
|
|
image.to_blob
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
class TrkPt < Struct.new(:segment, :latitude, :longitude, :altitude, :timestamp)
|
|
def valid?
|
|
latitude && longitude && timestamp &&
|
|
latitude >= -90 && latitude <= 90 &&
|
|
longitude >= -180 && longitude <= 180
|
|
end
|
|
end
|
|
end
|