geo_pattern 1.3.2 → 1.4.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.rubocop.yml +58 -0
- data/.simplecov +25 -0
- data/.travis.yml +12 -0
- data/Gemfile +3 -1
- data/README.md +125 -23
- data/Rakefile +72 -9
- data/fixtures/generated_patterns/chevrons.svg +1 -0
- data/fixtures/generated_patterns/concentric_circles.svg +1 -0
- data/fixtures/generated_patterns/diamonds.svg +1 -0
- data/fixtures/generated_patterns/diamonds_with_base_color.svg +1 -0
- data/fixtures/generated_patterns/diamonds_with_color.svg +1 -0
- data/fixtures/generated_patterns/hexagons.svg +1 -0
- data/fixtures/generated_patterns/mosaic_squares.svg +1 -0
- data/fixtures/generated_patterns/nested_squares.svg +1 -0
- data/fixtures/generated_patterns/octagons.svg +1 -0
- data/fixtures/generated_patterns/overlapping_circles.svg +1 -0
- data/fixtures/generated_patterns/overlapping_rings.svg +1 -0
- data/fixtures/generated_patterns/plaid.svg +1 -0
- data/fixtures/generated_patterns/plus_signs.svg +1 -0
- data/fixtures/generated_patterns/sine_waves.svg +1 -0
- data/fixtures/generated_patterns/squares.svg +1 -0
- data/fixtures/generated_patterns/tessellation.svg +1 -0
- data/fixtures/generated_patterns/triangles.svg +1 -0
- data/fixtures/generated_patterns/xes.svg +1 -0
- data/geo_pattern.gemspec +8 -2
- data/lib/geo_pattern.rb +42 -20
- data/lib/geo_pattern/background.rb +25 -0
- data/lib/geo_pattern/background_generators/solid_generator.rb +52 -0
- data/lib/geo_pattern/color.rb +25 -0
- data/lib/geo_pattern/color_generators/base_color_generator.rb +55 -0
- data/lib/geo_pattern/color_generators/simple_generator.rb +27 -0
- data/lib/geo_pattern/color_preset.rb +26 -0
- data/lib/geo_pattern/errors.rb +7 -0
- data/lib/geo_pattern/geo_pattern_task.rb +59 -0
- data/lib/geo_pattern/helpers.rb +47 -0
- data/lib/geo_pattern/pattern.rb +84 -0
- data/lib/geo_pattern/pattern_generator.rb +33 -82
- data/lib/geo_pattern/pattern_helpers.rb +31 -2
- data/lib/geo_pattern/pattern_preset.rb +23 -0
- data/lib/geo_pattern/pattern_sieve.rb +36 -0
- data/lib/geo_pattern/pattern_store.rb +63 -0
- data/lib/geo_pattern/pattern_validator.rb +27 -0
- data/lib/geo_pattern/rake_task.rb +109 -0
- data/lib/geo_pattern/roles/comparable_metadata.rb +35 -0
- data/lib/geo_pattern/roles/named_generator.rb +13 -0
- data/lib/geo_pattern/seed.rb +21 -0
- data/lib/geo_pattern/structure.rb +25 -0
- data/lib/geo_pattern/structure_generators/base_generator.rb +85 -0
- data/lib/geo_pattern/structure_generators/chevrons_generator.rb +55 -0
- data/lib/geo_pattern/structure_generators/concentric_circles_generator.rb +56 -0
- data/lib/geo_pattern/structure_generators/diamonds_generator.rb +66 -0
- data/lib/geo_pattern/structure_generators/hexagons_generator.rb +67 -0
- data/lib/geo_pattern/structure_generators/mosaic_squares_generator.rb +85 -0
- data/lib/geo_pattern/structure_generators/nested_squares_generator.rb +60 -0
- data/lib/geo_pattern/structure_generators/octagons_generator.rb +44 -0
- data/lib/geo_pattern/structure_generators/overlapping_circles_generator.rb +55 -0
- data/lib/geo_pattern/structure_generators/overlapping_rings_generator.rb +55 -0
- data/lib/geo_pattern/structure_generators/plaid_generator.rb +55 -0
- data/lib/geo_pattern/structure_generators/plus_signs_generator.rb +62 -0
- data/lib/geo_pattern/structure_generators/sine_waves_generator.rb +43 -0
- data/lib/geo_pattern/structure_generators/squares_generator.rb +36 -0
- data/lib/geo_pattern/structure_generators/tessellation_generator.rb +103 -0
- data/lib/geo_pattern/structure_generators/triangles_generator.rb +61 -0
- data/lib/geo_pattern/structure_generators/xes_generator.rb +67 -0
- data/lib/geo_pattern/svg_image.rb +101 -0
- data/lib/geo_pattern/version.rb +1 -1
- data/script/bootstrap +30 -0
- data/script/console +8 -0
- data/script/test +21 -0
- data/spec/background_generators/solid_generator_spec.rb +50 -0
- data/spec/background_spec.rb +25 -0
- data/spec/color_generators/base_color_generator_spec.rb +31 -0
- data/spec/color_generators/simple_generator_spec.rb +12 -0
- data/spec/color_preset_spec.rb +53 -0
- data/spec/color_spec.rb +15 -0
- data/spec/geo_pattern_spec.rb +95 -24
- data/spec/helpers_spec.rb +65 -0
- data/spec/pattern_preset_spec.rb +41 -0
- data/spec/pattern_sieve_spec.rb +66 -0
- data/spec/pattern_spec.rb +72 -0
- data/spec/pattern_store_spec.rb +47 -0
- data/spec/pattern_validator_spec.rb +28 -0
- data/spec/seed_spec.rb +14 -0
- data/spec/spec_helper.rb +1 -6
- data/spec/structure_generators/chevrons_generator_spec.rb +5 -0
- data/spec/structure_generators/concentric_circles_generator_spec.rb +5 -0
- data/spec/structure_generators/diamonds_generator_spec.rb +5 -0
- data/spec/structure_generators/hexagons_generator_spec.rb +5 -0
- data/spec/structure_generators/mosaic_squares_generator_spec.rb +5 -0
- data/spec/structure_generators/nested_squares_generator_spec.rb +5 -0
- data/spec/structure_generators/octagons_generator_spec.rb +5 -0
- data/spec/structure_generators/overlapping_circles_generator_spec.rb +5 -0
- data/spec/structure_generators/overlapping_rings_generator_spec.rb +5 -0
- data/spec/structure_generators/plaid_generator_spec.rb +5 -0
- data/spec/structure_generators/plus_signs_generator_spec.rb +5 -0
- data/spec/structure_generators/sine_waves_generator_spec.rb +5 -0
- data/spec/structure_generators/squares_generator_spec.rb +5 -0
- data/spec/structure_generators/tessellation_generator_spec.rb +5 -0
- data/spec/structure_generators/triangles_generator_spec.rb +5 -0
- data/spec/structure_generators/xes_generator_spec.rb +5 -0
- data/spec/structure_spec.rb +38 -0
- data/spec/support/helpers/fixtures.rb +12 -0
- data/spec/support/kernel.rb +40 -0
- data/spec/support/matchers/image.rb +17 -0
- data/spec/support/matchers/name.rb +15 -0
- data/spec/support/rspec.rb +1 -1
- data/spec/support/shared_examples/generator.rb +46 -0
- data/spec/support/shared_examples/pattern.rb +31 -0
- data/spec/support/shared_examples/pattern_name.rb +7 -0
- data/spec/support/shared_examples/structure.rb +48 -0
- data/spec/svg_spec.rb +3 -3
- metadata +141 -25
- data/lib/geo_pattern/pattern/base_pattern.rb +0 -47
- data/lib/geo_pattern/pattern/chevron_pattern.rb +0 -45
- data/lib/geo_pattern/pattern/concentric_circles_pattern.rb +0 -47
- data/lib/geo_pattern/pattern/diamond_pattern.rb +0 -56
- data/lib/geo_pattern/pattern/hexagon_pattern.rb +0 -57
- data/lib/geo_pattern/pattern/mosaic_squares_pattern.rb +0 -76
- data/lib/geo_pattern/pattern/nested_squares_pattern.rb +0 -51
- data/lib/geo_pattern/pattern/octagon_pattern.rb +0 -35
- data/lib/geo_pattern/pattern/overlapping_circles_pattern.rb +0 -46
- data/lib/geo_pattern/pattern/overlapping_rings_pattern.rb +0 -46
- data/lib/geo_pattern/pattern/plaid_pattern.rb +0 -49
- data/lib/geo_pattern/pattern/plus_sign_pattern.rb +0 -53
- data/lib/geo_pattern/pattern/sine_wave_pattern.rb +0 -36
- data/lib/geo_pattern/pattern/square_pattern.rb +0 -27
- data/lib/geo_pattern/pattern/tessellation_pattern.rb +0 -93
- data/lib/geo_pattern/pattern/triangle_pattern.rb +0 -51
- data/lib/geo_pattern/pattern/xes_pattern.rb +0 -58
- data/lib/geo_pattern/svg.rb +0 -77
- data/spec/support/helpers.rb +0 -34
@@ -0,0 +1,60 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class NestedSquaresGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :block_size, :square_size
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
@block_size = map(hex_val(0, 1), 0, 15, 4, 12)
|
10
|
+
@square_size = block_size * 7
|
11
|
+
|
12
|
+
self.height = self.width = (square_size + block_size) * 6 + block_size * 6
|
13
|
+
end
|
14
|
+
|
15
|
+
def generate_structure
|
16
|
+
i = 0
|
17
|
+
for y in 0..5
|
18
|
+
for x in 0..5
|
19
|
+
val = hex_val(i, 1)
|
20
|
+
opacity = opacity(val)
|
21
|
+
fill = fill_color(val)
|
22
|
+
|
23
|
+
styles = {
|
24
|
+
'fill' => 'none',
|
25
|
+
'stroke' => fill,
|
26
|
+
'style' => {
|
27
|
+
'opacity' => opacity,
|
28
|
+
'stroke-width' => "#{block_size}px"
|
29
|
+
}
|
30
|
+
}
|
31
|
+
|
32
|
+
svg.rect(x * square_size + x * block_size * 2 + block_size / 2,
|
33
|
+
y * square_size + y * block_size * 2 + block_size / 2,
|
34
|
+
square_size, square_size, styles)
|
35
|
+
|
36
|
+
val = hex_val(39 - i, 1)
|
37
|
+
opacity = opacity(val)
|
38
|
+
fill = fill_color(val)
|
39
|
+
|
40
|
+
styles = {
|
41
|
+
'fill' => 'none',
|
42
|
+
'stroke' => fill,
|
43
|
+
'style' => {
|
44
|
+
'opacity' => opacity,
|
45
|
+
'stroke-width' => "#{block_size}px"
|
46
|
+
}
|
47
|
+
}
|
48
|
+
|
49
|
+
svg.rect(x * square_size + x * block_size * 2 + block_size / 2 + block_size * 2,
|
50
|
+
y * square_size + y * block_size * 2 + block_size / 2 + block_size * 2,
|
51
|
+
block_size * 3, block_size * 3, styles)
|
52
|
+
i += 1
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
svg
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
@@ -0,0 +1,44 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class OctagonsGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :square_size, :tile
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
@square_size = map(hex_val(0, 1), 0, 15, 10, 60)
|
10
|
+
@tile = build_octogon_shape(square_size)
|
11
|
+
|
12
|
+
self.height = self.width = square_size * 6
|
13
|
+
end
|
14
|
+
|
15
|
+
def generate_structure
|
16
|
+
i = 0
|
17
|
+
for y in 0..5
|
18
|
+
for x in 0..5
|
19
|
+
val = hex_val(i, 1)
|
20
|
+
opacity = opacity(val)
|
21
|
+
fill = fill_color(val)
|
22
|
+
|
23
|
+
svg.polyline(tile,
|
24
|
+
'fill' => fill,
|
25
|
+
'fill-opacity' => opacity,
|
26
|
+
'stroke' => stroke_color,
|
27
|
+
'stroke-opacity' => stroke_opacity,
|
28
|
+
'transform' => "translate(#{x * square_size}, #{y * square_size})"
|
29
|
+
)
|
30
|
+
i += 1
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
svg
|
35
|
+
end
|
36
|
+
|
37
|
+
def build_octogon_shape(square_size)
|
38
|
+
s = square_size
|
39
|
+
c = s * 0.33
|
40
|
+
"#{c},0,#{s - c},0,#{s},#{c},#{s},#{s - c},#{s - c},#{s},#{c},#{s},0,#{s - c},0,#{c},#{c},0"
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class OverlappingCirclesGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :scale, :diameter, :radius
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
@scale = hex_val(0, 1)
|
10
|
+
@diameter = map(scale, 0, 15, 25, 200)
|
11
|
+
@radius = diameter / 2
|
12
|
+
|
13
|
+
self.height = self.width = radius * 6
|
14
|
+
end
|
15
|
+
|
16
|
+
def generate_structure
|
17
|
+
i = 0
|
18
|
+
for y in 0..5
|
19
|
+
for x in 0..5
|
20
|
+
val = hex_val(i, 1)
|
21
|
+
opacity = opacity(val)
|
22
|
+
fill = fill_color(val)
|
23
|
+
|
24
|
+
styles = {
|
25
|
+
'fill' => fill,
|
26
|
+
'style' => {
|
27
|
+
'opacity' => opacity
|
28
|
+
}
|
29
|
+
}
|
30
|
+
|
31
|
+
svg.circle(x * radius, y * radius, radius, styles)
|
32
|
+
|
33
|
+
# Add an extra one at top-right, for tiling.
|
34
|
+
if (x == 0)
|
35
|
+
svg.circle(6 * radius, y * radius, radius, styles)
|
36
|
+
end
|
37
|
+
|
38
|
+
# Add an extra row at the end that matches the first row, for tiling.
|
39
|
+
if (y == 0)
|
40
|
+
svg.circle(x * radius, 6 * radius, radius, styles)
|
41
|
+
end
|
42
|
+
|
43
|
+
# Add an extra one at bottom-right, for tiling.
|
44
|
+
if x == 0 && y == 0
|
45
|
+
svg.circle(6 * radius, 6 * radius, radius, styles)
|
46
|
+
end
|
47
|
+
i += 1
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
svg
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class OverlappingRingsGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :scale, :ring_size, :stroke_width
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
@scale = hex_val(0, 1)
|
10
|
+
@ring_size = map(scale, 0, 15, 10, 60)
|
11
|
+
@stroke_width = ring_size / 4
|
12
|
+
|
13
|
+
self.height = self.width = ring_size * 6
|
14
|
+
end
|
15
|
+
|
16
|
+
def generate_structure
|
17
|
+
i = 0
|
18
|
+
for y in 0..5
|
19
|
+
for x in 0..5
|
20
|
+
val = hex_val(i, 1)
|
21
|
+
opacity = opacity(val)
|
22
|
+
fill = fill_color(val)
|
23
|
+
|
24
|
+
styles = {
|
25
|
+
'fill' => 'none',
|
26
|
+
'stroke' => fill,
|
27
|
+
'style' => {
|
28
|
+
'opacity' => opacity,
|
29
|
+
'stroke-width' => "#{stroke_width}px"
|
30
|
+
}
|
31
|
+
}
|
32
|
+
|
33
|
+
svg.circle(x * ring_size, y * ring_size, ring_size - stroke_width / 2, styles)
|
34
|
+
|
35
|
+
# Add an extra one at top-right, for tiling.
|
36
|
+
if (x == 0)
|
37
|
+
svg.circle(6 * ring_size, y * ring_size, ring_size - stroke_width / 2, styles)
|
38
|
+
end
|
39
|
+
|
40
|
+
if (y == 0)
|
41
|
+
svg.circle(x * ring_size, 6 * ring_size, ring_size - stroke_width / 2, styles)
|
42
|
+
end
|
43
|
+
|
44
|
+
if x == 0 && y == 0
|
45
|
+
svg.circle(6 * ring_size, 6 * ring_size, ring_size - stroke_width / 2, styles)
|
46
|
+
end
|
47
|
+
i += 1
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
svg
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class PlaidGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
def generate_structure
|
7
|
+
local_height = 0
|
8
|
+
local_width = 0
|
9
|
+
|
10
|
+
# horizontal stripes
|
11
|
+
i = 0
|
12
|
+
18.times do
|
13
|
+
space = hex_val(i, 1)
|
14
|
+
local_height += space + 5
|
15
|
+
|
16
|
+
val = hex_val(i + 1, 1)
|
17
|
+
opacity = opacity(val)
|
18
|
+
fill = fill_color(val)
|
19
|
+
stripe_height = val + 5
|
20
|
+
|
21
|
+
svg.rect(0, local_height, '100%', stripe_height,
|
22
|
+
'opacity' => opacity,
|
23
|
+
'fill' => fill
|
24
|
+
)
|
25
|
+
local_height += stripe_height
|
26
|
+
i += 2
|
27
|
+
end
|
28
|
+
|
29
|
+
# vertical stripes
|
30
|
+
i = 0
|
31
|
+
18.times do
|
32
|
+
space = hex_val(i, 1)
|
33
|
+
local_width += space + 5
|
34
|
+
|
35
|
+
val = hex_val(i + 1, 1)
|
36
|
+
opacity = opacity(val)
|
37
|
+
fill = fill_color(val)
|
38
|
+
stripe_width = val + 5
|
39
|
+
|
40
|
+
svg.rect(local_width, 0, stripe_width, '100%',
|
41
|
+
'opacity' => opacity,
|
42
|
+
'fill' => fill
|
43
|
+
)
|
44
|
+
local_width += stripe_width
|
45
|
+
i += 2
|
46
|
+
end
|
47
|
+
|
48
|
+
self.height = local_height
|
49
|
+
self.width = local_width
|
50
|
+
|
51
|
+
svg
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
@@ -0,0 +1,62 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class PlusSignsGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :square_size, :plus_size, :plus_shape
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
@square_size = map(hex_val(0, 1), 0, 15, 10, 25)
|
10
|
+
@plus_size = square_size * 3
|
11
|
+
@plus_shape = build_plus_shape(square_size)
|
12
|
+
|
13
|
+
self.height = self.width = square_size * 12
|
14
|
+
end
|
15
|
+
|
16
|
+
def generate_structure
|
17
|
+
i = 0
|
18
|
+
for y in 0..5
|
19
|
+
for x in 0..5
|
20
|
+
val = hex_val(i, 1)
|
21
|
+
opacity = opacity(val)
|
22
|
+
fill = fill_color(val)
|
23
|
+
dx = (y % 2 == 0) ? 0 : 1
|
24
|
+
|
25
|
+
styles = {
|
26
|
+
'fill' => fill,
|
27
|
+
'stroke' => stroke_color,
|
28
|
+
'stroke-opacity' => stroke_opacity,
|
29
|
+
'style' => {
|
30
|
+
'fill-opacity' => opacity
|
31
|
+
}
|
32
|
+
}
|
33
|
+
|
34
|
+
svg.group(plus_shape, styles.merge(
|
35
|
+
'transform' => "translate(#{x * plus_size - x * square_size + dx * square_size - square_size},#{y * plus_size - y * square_size - plus_size / 2})"))
|
36
|
+
|
37
|
+
# Add an extra column on the right for tiling.
|
38
|
+
if (x == 0)
|
39
|
+
svg.group(plus_shape, styles.merge(
|
40
|
+
'transform' => "translate(#{4 * plus_size - x * square_size + dx * square_size - square_size},#{y * plus_size - y * square_size - plus_size / 2})"))
|
41
|
+
end
|
42
|
+
|
43
|
+
# Add an extra row on the bottom that matches the first row, for tiling.
|
44
|
+
if (y == 0)
|
45
|
+
svg.group(plus_shape, styles.merge(
|
46
|
+
'transform' => "translate(#{x * plus_size - x * square_size + dx * square_size - square_size},#{4 * plus_size - y * square_size - plus_size / 2})"))
|
47
|
+
end
|
48
|
+
|
49
|
+
# Add an extra one at top-right and bottom-right, for tiling.
|
50
|
+
if x == 0 && y == 0
|
51
|
+
svg.group(plus_shape, styles.merge(
|
52
|
+
'transform' => "translate(#{4 * plus_size - x * square_size + dx * square_size - square_size},#{4 * plus_size - y * square_size - plus_size / 2})"))
|
53
|
+
end
|
54
|
+
i += 1
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
svg
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
62
|
+
end
|
@@ -0,0 +1,43 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class SineWavesGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :period, :amplitude, :wave_width
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
@period = map(hex_val(0, 1), 0, 15, 100, 400).floor
|
10
|
+
@amplitude = map(hex_val(1, 1), 0, 15, 30, 100).floor
|
11
|
+
@wave_width = map(hex_val(2, 1), 0, 15, 3, 30).floor
|
12
|
+
|
13
|
+
self.height = wave_width * 36
|
14
|
+
self.width = period
|
15
|
+
end
|
16
|
+
|
17
|
+
def generate_structure
|
18
|
+
for i in 0..35
|
19
|
+
val = hex_val(i, 1)
|
20
|
+
opacity = opacity(val)
|
21
|
+
fill = fill_color(val)
|
22
|
+
x_offset = period / 4 * 0.7
|
23
|
+
|
24
|
+
styles = {
|
25
|
+
'fill' => 'none',
|
26
|
+
'stroke' => fill,
|
27
|
+
'style' => {
|
28
|
+
'opacity' => opacity,
|
29
|
+
'stroke-width' => "#{wave_width}px"
|
30
|
+
}
|
31
|
+
}
|
32
|
+
|
33
|
+
str = 'M0 ' + amplitude.to_s + ' C ' + x_offset.to_s + ' 0, ' + (period / 2 - x_offset).to_s + ' 0, ' + (period / 2).to_s + ' ' + amplitude.to_s + ' S ' + (period - x_offset).to_s + ' ' + (amplitude * 2).to_s + ', ' + period.to_s + ' ' + amplitude.to_s + ' S ' + (period * 1.5 - x_offset).to_s + ' 0, ' + (period * 1.5).to_s + ', ' + amplitude.to_s
|
34
|
+
|
35
|
+
svg.path(str, styles.merge('transform' => "translate(-#{period / 4}, #{wave_width * i - amplitude * 1.5})"))
|
36
|
+
svg.path(str, styles.merge('transform' => "translate(-#{period / 4}, #{wave_width * i - amplitude * 1.5 + wave_width * 36})"))
|
37
|
+
end
|
38
|
+
|
39
|
+
svg
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
@@ -0,0 +1,36 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class SquaresGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :square_size
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
@square_size = map(hex_val(0, 1), 0, 15, 10, 60)
|
10
|
+
|
11
|
+
self.height = self.width = square_size * 6
|
12
|
+
end
|
13
|
+
|
14
|
+
def generate_structure
|
15
|
+
i = 0
|
16
|
+
for y in 0..5
|
17
|
+
for x in 0..5
|
18
|
+
val = hex_val(i, 1)
|
19
|
+
opacity = opacity(val)
|
20
|
+
fill = fill_color(val)
|
21
|
+
|
22
|
+
svg.rect(x * square_size, y * square_size, square_size, square_size,
|
23
|
+
'fill' => fill,
|
24
|
+
'fill-opacity' => opacity,
|
25
|
+
'stroke' => stroke_color,
|
26
|
+
'stroke-opacity' => stroke_opacity
|
27
|
+
)
|
28
|
+
i += 1
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
svg
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
@@ -0,0 +1,103 @@
|
|
1
|
+
module GeoPattern
|
2
|
+
module StructureGenerators
|
3
|
+
class TessellationGenerator < BaseGenerator
|
4
|
+
private
|
5
|
+
|
6
|
+
attr_reader :side_length, :hex_height, :hex_width, :triangle_height, :triangle, :tile_width, :tile_height
|
7
|
+
|
8
|
+
def after_initialize
|
9
|
+
# 3.4.6.4 semi-regular tessellation
|
10
|
+
@side_length = map(hex_val(0, 1), 0, 15, 5, 40)
|
11
|
+
@hex_height = side_length * Math.sqrt(3)
|
12
|
+
@hex_width = side_length * 2
|
13
|
+
@triangle_height = side_length / 2 * Math.sqrt(3)
|
14
|
+
@triangle = build_rotated_triangle_shape(side_length, triangle_height)
|
15
|
+
@tile_width = side_length * 3 + triangle_height * 2
|
16
|
+
@tile_height = (hex_height * 2) + (side_length * 2)
|
17
|
+
|
18
|
+
self.height = tile_height
|
19
|
+
self.width = tile_width
|
20
|
+
end
|
21
|
+
|
22
|
+
def generate_structure
|
23
|
+
for i in 0..19
|
24
|
+
val = hex_val(i, 1)
|
25
|
+
opacity = opacity(val)
|
26
|
+
fill = fill_color(val)
|
27
|
+
|
28
|
+
styles = {
|
29
|
+
'stroke' => stroke_color,
|
30
|
+
'stroke-opacity' => stroke_opacity,
|
31
|
+
'fill' => fill,
|
32
|
+
'fill-opacity' => opacity,
|
33
|
+
'stroke-width' => 1
|
34
|
+
}
|
35
|
+
|
36
|
+
case i
|
37
|
+
when 0 # all 4 corners
|
38
|
+
svg.rect(-side_length / 2, -side_length / 2, side_length, side_length, styles)
|
39
|
+
svg.rect(tile_width - side_length / 2, -side_length / 2, side_length, side_length, styles)
|
40
|
+
svg.rect(-side_length / 2, tile_height - side_length / 2, side_length, side_length, styles)
|
41
|
+
svg.rect(tile_width - side_length / 2, tile_height - side_length / 2, side_length, side_length, styles)
|
42
|
+
when 1 # center / top square
|
43
|
+
svg.rect(hex_width / 2 + triangle_height, hex_height / 2, side_length, side_length, styles)
|
44
|
+
when 2 # side squares
|
45
|
+
svg.rect(-side_length / 2, tile_height / 2 - side_length / 2, side_length, side_length, styles)
|
46
|
+
svg.rect(tile_width - side_length / 2, tile_height / 2 - side_length / 2, side_length, side_length, styles)
|
47
|
+
when 3 # center / bottom square
|
48
|
+
svg.rect(hex_width / 2 + triangle_height, hex_height * 1.5 + side_length, side_length, side_length, styles)
|
49
|
+
when 4 # left top / bottom triangle
|
50
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{side_length / 2}, #{-side_length / 2}) rotate(0, #{side_length / 2}, #{triangle_height / 2})"))
|
51
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{side_length / 2}, #{tile_height - -side_length / 2}) rotate(0, #{side_length / 2}, #{triangle_height / 2}) scale(1, -1)"))
|
52
|
+
when 5 # right top / bottom triangle
|
53
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{tile_width - side_length / 2}, #{-side_length / 2}) rotate(0, #{side_length / 2}, #{triangle_height / 2}) scale(-1, 1)"))
|
54
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{tile_width - side_length / 2}, #{tile_height + side_length / 2}) rotate(0, #{side_length / 2}, #{triangle_height / 2}) scale(-1, -1)"))
|
55
|
+
when 6 # center / top / right triangle
|
56
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{tile_width / 2 + side_length / 2}, #{hex_height / 2})"))
|
57
|
+
when 7 # center / top / left triangle
|
58
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{tile_width - tile_width / 2 - side_length / 2}, #{hex_height / 2}) scale(-1, 1)"))
|
59
|
+
when 8 # center / bottom / right triangle
|
60
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{tile_width / 2 + side_length / 2}, #{tile_height - hex_height / 2}) scale(1, -1)"))
|
61
|
+
when 9 # center / bottom / left triangle
|
62
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{tile_width - tile_width / 2 - side_length / 2}, #{tile_height - hex_height / 2}) scale(-1, -1)"))
|
63
|
+
when 10 # left / middle triangle
|
64
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{side_length / 2}, #{tile_height / 2 - side_length / 2})"))
|
65
|
+
when 11 # right / middle triangle
|
66
|
+
svg.polyline(triangle, styles.merge('transform' => "translate(#{tile_width - side_length / 2}, #{tile_height / 2 - side_length / 2}) scale(-1, 1)"))
|
67
|
+
when 12 # left / top square
|
68
|
+
svg.rect(0, 0, side_length, side_length,
|
69
|
+
styles.merge('transform' => "translate(#{side_length / 2}, #{side_length / 2}) rotate(-30, 0, 0)"))
|
70
|
+
when 13 # right / top square
|
71
|
+
svg.rect(0, 0, side_length, side_length,
|
72
|
+
styles.merge('transform' => "scale(-1, 1) translate(#{-tile_width + side_length / 2}, #{side_length / 2}) rotate(-30, 0, 0)"))
|
73
|
+
when 14 # left / center-top square
|
74
|
+
svg.rect(0, 0, side_length, side_length,
|
75
|
+
styles.merge('transform' => "translate(#{side_length / 2}, #{tile_height / 2 - side_length / 2 - side_length}) rotate(30, 0, #{side_length})"))
|
76
|
+
when 15 # right / center-top square
|
77
|
+
svg.rect(0, 0, side_length, side_length,
|
78
|
+
styles.merge('transform' => "scale(-1, 1) translate(#{-tile_width + side_length / 2}, #{tile_height / 2 - side_length / 2 - side_length}) rotate(30, 0, #{side_length})"))
|
79
|
+
when 16 # left / center-top square
|
80
|
+
svg.rect(0, 0, side_length, side_length,
|
81
|
+
styles.merge('transform' => "scale(1, -1) translate(#{side_length / 2}, #{-tile_height + tile_height / 2 - side_length / 2 - side_length}) rotate(30, 0, #{side_length})"))
|
82
|
+
when 17 # right / center-bottom square
|
83
|
+
svg.rect(0, 0, side_length, side_length,
|
84
|
+
styles.merge('transform' => "scale(-1, -1) translate(#{-tile_width + side_length / 2}, #{-tile_height + tile_height / 2 - side_length / 2 - side_length}) rotate(30, 0, #{side_length})"))
|
85
|
+
when 18 # left / bottom square
|
86
|
+
svg.rect(0, 0, side_length, side_length,
|
87
|
+
styles.merge('transform' => "scale(1, -1) translate(#{side_length / 2}, #{-tile_height + side_length / 2}) rotate(-30, 0, 0)"))
|
88
|
+
when 19 # right / bottom square
|
89
|
+
svg.rect(0, 0, side_length, side_length,
|
90
|
+
styles.merge('transform' => "scale(-1, -1) translate(#{-tile_width + side_length / 2}, #{-tile_height + side_length / 2}) rotate(-30, 0, 0)"))
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
svg
|
95
|
+
end
|
96
|
+
|
97
|
+
def build_rotated_triangle_shape(side_length, width)
|
98
|
+
half_height = side_length / 2
|
99
|
+
"0, 0, #{width}, #{half_height}, 0, #{side_length}, 0, 0"
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|