Add border/casing/line attribute support to svg map keys

This commit is contained in:
Anton Khorev 2023-12-23 20:38:32 +03:00
parent f41bb4c325
commit b13ac4ec21
3 changed files with 133 additions and 6 deletions

View file

@ -1,7 +1,35 @@
module SvgHelper
def solid_svg_tag(width, height, fill, **options)
tag.svg :width => width, :height => height, **options do
tag.rect :width => "100%", :height => "100%", :fill => fill
def key_svg_tag(**options)
border_width = options["border"] ? (options["border-width"] || 1) : 0
rect_attrs = {
:width => "100%",
:height => "100%",
:fill => options["fill"] || "none"
}
if border_width.positive?
rect_attrs[:x] = rect_attrs[:y] = format("%g", 0.5 * border_width)
rect_attrs[:width] = options["width"] - border_width
rect_attrs[:height] = options["height"] - border_width
end
svg_attrs = options.slice("width", "height", "opacity", :class)
tag.svg(**svg_attrs) do
concat tag.rect(**rect_attrs, **stroke_attrs(options, "border")) if options["fill"] || options["border"]
concat tag.line(:x2 => "100%", :y1 => "50%", :y2 => "50%", **stroke_attrs(options, "line")) if options["line"]
if options["casing"]
casing_width = options["casing-width"] || 1
y_top = 0.5 * casing_width
y_bottom = options["height"] - (0.5 * casing_width)
concat tag.g(tag.line(:x2 => "100%", :y1 => y_top, :y2 => y_top) +
tag.line(:x2 => "100%", :y1 => y_bottom, :y2 => y_bottom),
**stroke_attrs(options, "casing"))
end
end
end
private
def stroke_attrs(attrs, prefix)
attrs.select { |key| key.start_with?(prefix) }.transform_keys { |key| key.delete_prefix(prefix).prepend("stroke") }
end
end

View file

@ -4,10 +4,10 @@
<% layer_data.each do |entry| %>
<%= tag.tr :class => "mapkey-table-entry", :data => { :layer => layer_name, :zoom_min => entry["min_zoom"], :zoom_max => entry["max_zoom"] } do %>
<td>
<% if entry["width"] && entry["height"] && entry["fill"] %>
<%= solid_svg_tag entry["width"], entry["height"], entry["fill"], :class => "d-block mx-auto" %>
<% else %>
<% if entry["image"] %>
<%= image_tag "key/#{layer_name}/#{entry['image']}", :class => "d-block mx-auto" %>
<% else %>
<%= key_svg_tag :class => "d-block mx-auto", **entry %>
<% end %>
</td>
<td>

View file

@ -0,0 +1,99 @@
require "test_helper"
class SvgHelperTest < ActionView::TestCase
def test_key_fill
svg = key_svg_tag("width" => 60, "height" => 40, "fill" => "green")
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="60" height="40">
<rect width="100%" height="100%" fill="green" />
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_border
svg = key_svg_tag("width" => 60, "height" => 40, "border" => "red")
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="60" height="40">
<rect x="0.5" y="0.5" width="59" height="39" fill="none" stroke="red" />
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_border_width
svg = key_svg_tag("width" => 60, "height" => 40, "border" => "red", "border-width" => 3)
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="60" height="40">
<rect x="1.5" y="1.5" width="57" height="37" fill="none" stroke="red" stroke-width="3" />
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_border_with_integer_coords
svg = key_svg_tag("width" => 60, "height" => 40, "border" => "red", "border-width" => 2)
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="60" height="40">
<rect x="1" y="1" width="58" height="38" fill="none" stroke="red" stroke-width="2" />
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_border_fractional_width
svg = key_svg_tag("width" => 60, "height" => 40, "border" => "red", "border-width" => 1.5)
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="60" height="40">
<rect x="0.75" y="0.75" width="58.5" height="38.5" fill="none" stroke="red" stroke-width="1.5" />
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_line
svg = key_svg_tag("width" => 80, "height" => 20, "line" => "blue")
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="80" height="20">
<line x2="100%" y1="50%" y2="50%" stroke="blue" />
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_line_width
svg = key_svg_tag("width" => 80, "height" => 20, "line" => "blue", "line-width" => 3)
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="80" height="20">
<line x2="100%" y1="50%" y2="50%" stroke="blue" stroke-width="3" />
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_casing
svg = key_svg_tag("width" => 80, "height" => 20, "casing" => "yellow")
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="80" height="20">
<g stroke="yellow">
<line x2="100%" y1="0.5" y2="0.5" />
<line x2="100%" y1="19.5" y2="19.5" />
</g>
</svg>
HTML
assert_dom_equal expected, svg
end
def test_key_casing_width
svg = key_svg_tag("width" => 80, "height" => 20, "casing" => "yellow", "casing-width" => 5)
expected = <<~HTML.gsub(/\n\s*/, "")
<svg width="80" height="20">
<g stroke="yellow" stroke-width="5">
<line x2="100%" y1="2.5" y2="2.5" />
<line x2="100%" y1="17.5" y2="17.5" />
</g>
</svg>
HTML
assert_dom_equal expected, svg
end
end