feat: add sprite spawns
This commit is contained in:
parent
0581c53454
commit
ca352fc531
13 changed files with 379 additions and 13 deletions
|
@ -336,3 +336,58 @@ a:has(button) {
|
|||
justify-content: space-between;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.sprite-list {
|
||||
display: flex;
|
||||
align-self: start;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.sprite {
|
||||
display: flex;
|
||||
gap: 5px;
|
||||
margin-bottom: 5px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.sprite-name {
|
||||
background-color: #3D9970;
|
||||
color: #fff;
|
||||
padding: 2px 4px;
|
||||
}
|
||||
|
||||
.sprite .spawning {
|
||||
color: #3D9970;
|
||||
}
|
||||
|
||||
.sprite .not-spawning {
|
||||
color: #b5443a;
|
||||
}
|
||||
|
||||
.sprite-levels {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr 1fr;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.sprite-map {
|
||||
margin-bottom: 10px;
|
||||
margin-top: 5px;
|
||||
}
|
||||
|
||||
.sprite-map summary {
|
||||
font-size: 12px;
|
||||
}
|
||||
|
||||
.sprite-levels > div {
|
||||
background-color: #111;
|
||||
color: #fff;
|
||||
font-family: monospace;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
text-align: center;
|
||||
text-transform: uppercase;
|
||||
font-weight: bold;
|
||||
padding: 3px 5px;
|
||||
}
|
||||
|
|
63
app/views/instance/_chlog.html.erb
Normal file
63
app/views/instance/_chlog.html.erb
Normal file
|
@ -0,0 +1,63 @@
|
|||
<h3>Challenge log</h3>
|
||||
|
||||
<div class="sprite-list">
|
||||
<% Bestiary.get_sprites_for_zone(instance.zone).each do |sprite| %>
|
||||
<% is_spawning = forecast[0][:curr_weather].in?(sprite[:weather]) %>
|
||||
<div class="sprite">
|
||||
<div class="sprite-name">
|
||||
<img src="/<%= sprite[:element] %>.png" width="15" style="margin-top: -3px;" />
|
||||
<%= sprite[:name] %>
|
||||
</div>
|
||||
<% if is_spawning %>
|
||||
<div class="spawning">is spawning!</div>
|
||||
<% else %>
|
||||
<% next_pattern = forecast.find { |f| f[:curr_weather].in?(sprite[:weather]) } %>
|
||||
<div class="not-spawning">spawns in
|
||||
<% if next_pattern.nil? %>
|
||||
>300m
|
||||
<% else %>
|
||||
<%= ((next_pattern[:time] - Time.now.utc) / 1.minutes).ceil %>m
|
||||
<% end %>
|
||||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
<% if is_spawning %>
|
||||
<div class="sprite-levels">
|
||||
<% sprite[:sprite_levels].each do |lvl| %>
|
||||
<div>
|
||||
<div style="display: flex; gap: 3px;">
|
||||
<div>lv<%= lvl[:lv] %></div>
|
||||
<div>
|
||||
<% if lvl[:mutates] %>
|
||||
<img src="/mutation.png" width="13" />
|
||||
<% elsif lvl[:adapts] %>
|
||||
<img src="/adaptation.png" width="13" />
|
||||
<% end %>
|
||||
</div>
|
||||
</div>
|
||||
~<%= lvl[:mx] %>/<%= lvl[:my] %>
|
||||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
<details class="sprite-map">
|
||||
<summary>
|
||||
show map
|
||||
</summary>
|
||||
|
||||
<div style="position: relative;">
|
||||
<img src="/maps/<%= sprite[:zone] %>_fw.jpg" style="width: 100%; position: relative;" />
|
||||
<% sprite[:sprite_levels].each do |lvl| %>
|
||||
<div
|
||||
title="LV<%= lvl[:lv] %>"
|
||||
style="line-height: 10px; position: absolute; left: <%= (lvl[:mx] / 42) * 100.0 - 2.0 %>%; top: <%= (lvl[:my] / 42) * 100.0 - 2.5 %>%"
|
||||
>
|
||||
•
|
||||
<br/>
|
||||
<small>LV<%= lvl[:lv] %></small>
|
||||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
</details>
|
||||
<% end %>
|
||||
<% end %>
|
||||
</div>
|
|
@ -34,7 +34,7 @@
|
|||
<% pop = instance.pops.find { |pop| pop.name == nm[:name].parameterize } %>
|
||||
<% mins = ActiveSupport::Duration.build(Time.current - pop.created_at) %>
|
||||
<div class="timer">
|
||||
<div>» <%= (120.minutes - mins).in_minutes.floor %>m</div>
|
||||
<div>» <%= (120.minutes - mins).in_minutes.ceil %>m</div>
|
||||
<div class="progress-container">
|
||||
<span class="progress-bar" style="width: <%= (mins.in_minutes / 120) * 100 %>%"></span>
|
||||
</div>
|
||||
|
@ -55,14 +55,14 @@
|
|||
<% end %>
|
||||
<% if nm[:weather] && forecast[0][:curr_weather] != nm[:weather] %>
|
||||
<% next_pattern = forecast.find { |f| f[:curr_weather] == nm[:weather] } %>
|
||||
<%= Weather.get_weather_name(nm[:weather]) %> in <%= ((next_pattern[:time] - Time.now.utc) / 1.minutes).floor %>m
|
||||
<%= Weather.get_weather_name(nm[:weather]) %> in <%= ((next_pattern[:time] - Time.now.utc) / 1.minutes).ceil %>m
|
||||
<% end %>
|
||||
<% if nm[:spawned_by][:weather] && forecast[0][:curr_weather] != nm[:spawned_by][:weather] %>
|
||||
<% next_pattern = forecast.find { |f| f[:curr_weather] == nm[:spawned_by][:weather] } %>
|
||||
<%= Weather.get_weather_name(nm[:spawned_by][:weather]) %> in <%= ((next_pattern[:time] - Time.now.utc) / 1.minutes).floor %>m
|
||||
<%= Weather.get_weather_name(nm[:spawned_by][:weather]) %> in <%= ((next_pattern[:time] - Time.now.utc) / 1.minutes).ceil %>m
|
||||
<% end %>
|
||||
<% if (nm[:night_only] || nm[:spawned_by][:night_only]) && is_day? %>
|
||||
<div>Night in <%= ((Clock.to_earth_time(Clock.get_current_eorzea_time.change(hour: 18)) - Time.now.utc) / 1.minutes).floor %>m</div>
|
||||
Night in <%= ((Clock.to_earth_time(Clock.get_current_eorzea_time.change(hour: 18)) - Time.now.utc) / 1.minutes).ceil %>m
|
||||
<% end %>
|
||||
</div>
|
||||
</div>
|
||||
|
@ -87,4 +87,12 @@
|
|||
</div>
|
||||
</section>
|
||||
<% end %>
|
||||
|
||||
|
||||
<a href="/maps/<%= @instance.zone %>_full.jpg" target="_blank">
|
||||
<button>full map (new tab)</button>
|
||||
</a>
|
||||
<%= form_with url: clone_instance_path(instance: @instance.public_id), html: { style: "display: inline-block;" } do |f| %>
|
||||
<%= f.submit "clone instance" %>
|
||||
<% end %>
|
||||
</div>
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
<div id="public_id" data-content="<%= @instance.public_id %>"></div>
|
||||
<div hx-get="" hx-trigger="every 5m" hx-swap="outerHTML" hx-select="#container" hx-target="#container">
|
||||
<div hx-get="" hx-trigger="<%= Rails.env == "development" ? "every 5m" : "every 5s" %>" hx-swap="outerHTML" hx-select="#container" hx-target="#container">
|
||||
<header>
|
||||
<div class="title">
|
||||
<%= link_to root_path do %><img src="/icon.png" width="50" alt="eureka.coffee logo" /><% end %>
|
||||
|
@ -36,13 +36,14 @@
|
|||
»
|
||||
<div class="weather">
|
||||
<img src="/weather/<%= @forecast[i + 1][:curr_weather] %>.png" width="25" title="<%= @forecast[i + 1][:weather_name] %>" />
|
||||
<div><%= ((@forecast[i + 1][:time] - Time.now.utc) / 1.minutes).floor %>m</div>
|
||||
<div><%= ((@forecast[i + 1][:time] - Time.now.utc) / 1.minutes).ceil %>m</div>
|
||||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<%= render partial: "fairies", locals: { instance: @instance } %>
|
||||
<%= render partial: "chlog", locals: { instance: @instance, forecast: @forecast } %>
|
||||
</div>
|
||||
</div>
|
||||
</main>
|
||||
|
@ -50,9 +51,3 @@
|
|||
<%= javascript_include_tag "list" %>
|
||||
</div>
|
||||
|
||||
<a href="/maps/<%= @instance.zone %>_full.jpg" target="_blank">
|
||||
<button>full map (new tab)</button>
|
||||
</a>
|
||||
<%= form_with url: clone_instance_path(instance: @instance.public_id), html: { style: "display: inline-block;" } do |f| %>
|
||||
<%= f.submit "clone instance" %>
|
||||
<% end %>
|
||||
|
|
|
@ -2,10 +2,12 @@ anemos_data = Tomlrb.load_file("./data/anemos.toml", symbolize_keys: true)
|
|||
pagos_data = Tomlrb.load_file("./data/pagos.toml", symbolize_keys: true)
|
||||
pyros_data = Tomlrb.load_file("./data/pyros.toml", symbolize_keys: true)
|
||||
hydatos_data = Tomlrb.load_file("./data/hydatos.toml", symbolize_keys: true)
|
||||
bestiary_data = Tomlrb.load_file("./data/bestiary.toml", symbolize_keys: true)
|
||||
|
||||
APP_DATA = {
|
||||
anemos: anemos_data,
|
||||
pagos: pagos_data,
|
||||
pyros: pyros_data,
|
||||
hydatos: hydatos_data
|
||||
hydatos: hydatos_data,
|
||||
bestiary: bestiary_data
|
||||
}
|
||||
|
|
236
data/bestiary.toml
Normal file
236
data/bestiary.toml
Normal file
|
@ -0,0 +1,236 @@
|
|||
[[bestiary]]
|
||||
name = "Blizzard Sprite"
|
||||
sprite = true
|
||||
weather = ["snow"]
|
||||
element = "ice"
|
||||
zone = "anemos"
|
||||
sprite_levels = [
|
||||
{ lv = 2, mx = 21.7, my = 31.7 },
|
||||
{ lv = 6, mx = 13.4, my = 23.9 },
|
||||
{ lv = 10, mx = 22.7, my = 24.6 },
|
||||
{ lv = 20, mx = 28.3, my = 14.4 },
|
||||
{ lv = 22, mx = 34.3, my = 16.0 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Typhoon Sprite"
|
||||
sprite = true
|
||||
weather = ["gales"]
|
||||
element = "wind"
|
||||
zone = "anemos"
|
||||
sprite_levels = [
|
||||
{ lv = 5, mx = 15.1, my = 24.9 },
|
||||
{ lv = 9, mx = 24.8, my = 23.7 },
|
||||
{ lv = 12, mx = 16.4, my = 19.0 },
|
||||
{ lv = 13, mx = 23.1, my = 20.4 },
|
||||
{ lv = 15, mx = 28.7, my = 20.5 },
|
||||
{ lv = 19, mx = 28.0, my = 16.6 },
|
||||
{ lv = 21, mx = 32.1, my = 17.3 },
|
||||
{ lv = 23, mx = 9.5, my = 17.6 },
|
||||
{ lv = 25, mx = 9.6, my = 21.4 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Rain Sprite"
|
||||
sprite = true
|
||||
weather = ["showers"]
|
||||
element = "water"
|
||||
zone = "anemos"
|
||||
sprite_levels = [
|
||||
{ lv = 7, mx = 13.9, my = 20.5 },
|
||||
{ lv = 9, mx = 22.2, my = 28.5 },
|
||||
{ lv = 11, mx = 19.0, my = 24.4 },
|
||||
{ lv = 14, mx = 26.1, my = 24.8 },
|
||||
{ lv = 17, mx = 18.3, my = 14.8 },
|
||||
{ lv = 18, mx = 20.8, my = 13.9 },
|
||||
{ lv = 24, mx = 9.1, my = 19.8 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Ember Sprite"
|
||||
sprite = true
|
||||
weather = ["heat"]
|
||||
element = "fire"
|
||||
zone = "pagos"
|
||||
sprite_levels = [
|
||||
{ lv = 23, mx = 15.6, my = 27.4 },
|
||||
{ lv = 27, mutates = true, mx = 28.6, my = 27.1 },
|
||||
{ lv = 31, adapts = true, mx = 15.6, my = 22.0 },
|
||||
{ lv = 35, adapts = true, mx = 8.4, my = 17.5 },
|
||||
{ lv = 39, adapts = true, mx = 24.0, my = 16.7 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Snowmelt Sprite"
|
||||
sprite = true
|
||||
weather = ["fog"]
|
||||
element = "water"
|
||||
zone = "pagos"
|
||||
sprite_levels = [
|
||||
{ lv = 22, mx = 13.4, my = 25.4 },
|
||||
{ lv = 26, mutates = true, mx = 27.3, my = 26.7 },
|
||||
{ lv = 29, mx = 32.8, my = 24.8 },
|
||||
{ lv = 34, mx = 6.9, my = 14.0 },
|
||||
{ lv = 38, adapts = true, mx = 22.2, my = 16.3 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Snows. Sprite"
|
||||
sprite = true
|
||||
weather = ["snow", "blizzards"]
|
||||
element = "ice"
|
||||
zone = "pagos"
|
||||
sprite_levels = [
|
||||
{ lv = 20, mx = 9.4, my = 23.6 },
|
||||
{ lv = 24, mx = 18.2, my = 28.1 },
|
||||
{ lv = 28, adapts = true, mx = 30.9, my = 27.9 },
|
||||
{ lv = 32, mutates = true, mx = 10.9, my = 12.9 },
|
||||
{ lv = 36, mx = 22.3, my = 18.1 },
|
||||
{ lv = 40, adapts = true, mx = 36.7, my = 15.5 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Thunders. Sprite"
|
||||
sprite = true
|
||||
weather = ["thunder"]
|
||||
element = "lightning"
|
||||
zone = "pagos"
|
||||
sprite_levels = [
|
||||
{ lv = 21, mx = 12.8, my = 27.9 },
|
||||
{ lv = 25, mutates = true, mx = 21.1, my = 28.7 },
|
||||
{ lv = 30, mx = 30.0, my = 22.8 },
|
||||
{ lv = 33, mutates = true, mx = 10.6, my = 16.3 },
|
||||
{ lv = 37, adapts = true, mx = 31.3, my = 17.3 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Typhoon Sprite"
|
||||
sprite = true
|
||||
weather = ["umbral_wind"]
|
||||
element = "wind"
|
||||
zone = "pyros"
|
||||
sprite_levels = [
|
||||
{ lv = 35, mx = 17.0, my = 25.8 },
|
||||
{ lv = 40, mx = 29.1, my = 27.5 },
|
||||
{ lv = 42, mx = 31.5, my = 32.5 },
|
||||
{ lv = 42, mx = 28.2, my = 32.9 },
|
||||
{ lv = 45, mx = 24.1, my = 18.7 },
|
||||
{ lv = 45, mx = 18.1, my = 15.5 },
|
||||
{ lv = 48, mx = 14.1, my = 9.1 },
|
||||
{ lv = 49, mx = 15.2, my = 7.3 },
|
||||
{ lv = 55, adapts = true, mx = 38.2, my = 16.4 },
|
||||
{ lv = 55, adapts = true, mx = 15.4, my = 36.9 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Ember Sprite"
|
||||
sprite = true
|
||||
weather = ["heat"]
|
||||
element = "fire"
|
||||
zone = "pyros"
|
||||
sprite_levels = [
|
||||
{ lv = 38, mx = 13.1, my = 29.0 },
|
||||
{ lv = 38, mx = 25.0, my = 24.4 },
|
||||
{ lv = 38, mx = 25.3, my = 27.0 },
|
||||
{ lv = 39, mx = 26.4, my = 23.9 },
|
||||
{ lv = 39, mx = 11.4, my = 29.7 },
|
||||
{ lv = 43, adapts = true, mx = 25.2, my = 37.7 },
|
||||
{ lv = 43, adapts = true, mx = 19.0, my = 34.4 },
|
||||
{ lv = 44, mx = 21.0, my = 32.2 },
|
||||
{ lv = 53, mx = 29.7 , my = 17.2 },
|
||||
{ lv = 53, mx = 35.8, my = 16.5 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Thunders. Sprite"
|
||||
sprite = true
|
||||
weather = ["thunder"]
|
||||
element = "lightning"
|
||||
zone = "pyros"
|
||||
sprite_levels = [
|
||||
{ lv = 36, mx = 17.5, my = 27.4 },
|
||||
{ lv = 37, mx = 23.7, my = 26.8 },
|
||||
{ lv = 41, mx = 24.0, my = 34.1 },
|
||||
{ lv = 46, adapts = true, mx = 12.0, my = 17.3 },
|
||||
{ lv = 51, mx = 25.7, my = 9.2 },
|
||||
{ lv = 54, adapts = true, mx = 36.5, my = 15.2 },
|
||||
{ lv = 54, adapts = true, mx = 12.1, my = 33.5 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Snows. Sprite"
|
||||
sprite = true
|
||||
weather = ["snow", "blizzards"]
|
||||
element = "ice"
|
||||
zone = "pyros"
|
||||
sprite_levels = [
|
||||
{ lv = 47, mx = 13.1, my = 14.5 },
|
||||
{ lv = 50, mx = 24.9, my = 6.5 },
|
||||
{ lv = 50, mx = 25.3, my = 12.2 },
|
||||
{ lv = 52, adapts = true, mx = 27.4, my = 17.5 },
|
||||
{ lv = 52, adapts = true, mx = 30.0, my = 12.3 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Thunders. Sprites"
|
||||
sprite = true
|
||||
weather = ["thunder"]
|
||||
element = "lightning"
|
||||
zone = "hydatos"
|
||||
sprite_levels = [
|
||||
{ lv = 53, adapts = true, mx = 11.6, my = 23.5 },
|
||||
{ lv = 53, adapts = true, mx = 14.7, my = 25.0 },
|
||||
{ lv = 57, mx = 7.7, my = 15.6 },
|
||||
{ lv = 57, mx = 5.7, my = 21.2 },
|
||||
{ lv = 61, mutates = true, mx = 25.6, my = 19.9 },
|
||||
{ lv = 65, mx = 34.6, my = 17.0 },
|
||||
{ lv = 65, mx = 30.9, my = 21.0 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Snows. Sprite"
|
||||
sprite = true
|
||||
weather = ["snow"]
|
||||
element = "ice"
|
||||
zone = "hydatos"
|
||||
sprite_levels = [
|
||||
{ lv = 52, mx = 16.4, my = 22.3 },
|
||||
{ lv = 52, mx = 13.0, my = 14.1 },
|
||||
{ lv = 56, adapts = true, mx = 10.4, my = 15.5 },
|
||||
{ lv = 56, adapts = true, mx = 8.4, my = 19.7 },
|
||||
{ lv = 56, adapts = true, mx = 10.8, my = 21.1 },
|
||||
{ lv = 60, adapts = true, mx = 24.4, my = 17.8 },
|
||||
{ lv = 64, adapts = true, mx = 34.9, my = 14.0 },
|
||||
{ lv = 65, mx = 32.4, my = 28.7 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Death Sprite"
|
||||
sprite = true
|
||||
weather = ["gloom"]
|
||||
element = "fire"
|
||||
zone = "hydatos"
|
||||
sprite_levels = [
|
||||
{ lv = 50, mx = 19.4, my = 16.1 },
|
||||
{ lv = 54, mx = 22.2, my = 23.2 },
|
||||
{ lv = 58, mx = 3.6, my = 14.1 },
|
||||
{ lv = 58, mx = 8.2, my = 12.7 },
|
||||
{ lv = 58, mx = 10.6, my = 19.4 },
|
||||
{ lv = 62, mx = 30.3, my = 27.2 },
|
||||
{ lv = 62, mx = 35.9, my = 27.1 },
|
||||
{ lv = 65, mx = 32.4, my = 28.7 }
|
||||
]
|
||||
|
||||
[[bestiary]]
|
||||
name = "Snowmelt Sprite"
|
||||
sprite = true
|
||||
weather = ["showers"]
|
||||
element = "water"
|
||||
zone = "hydatos"
|
||||
sprite_levels = [
|
||||
{ lv = 51, mx = 16.2, my = 20.6 },
|
||||
{ lv = 55, adapts = true, mx = 10.8, my = 26.9 },
|
||||
{ lv = 59, mx = 3.8, my = 27.3 },
|
||||
{ lv = 63, mutates = true, mx = 36.1, my = 18.5 },
|
||||
{ lv = 65, mx = 32.4, my = 28.7 }
|
||||
]
|
7
lib/bestiary.rb
Normal file
7
lib/bestiary.rb
Normal file
|
@ -0,0 +1,7 @@
|
|||
class Bestiary
|
||||
def self.get_sprites_for_zone(zone)
|
||||
b = APP_DATA[:bestiary][:bestiary]
|
||||
|
||||
b.filter { |m| m[:sprite] && m[:zone] == zone }
|
||||
end
|
||||
end
|
BIN
public/adaptation.png
Normal file
BIN
public/adaptation.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 744 B |
BIN
public/maps/anemos_fw.jpg
Normal file
BIN
public/maps/anemos_fw.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 556 KiB |
BIN
public/maps/hydatos_fw.jpg
Normal file
BIN
public/maps/hydatos_fw.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 532 KiB |
BIN
public/maps/pagos_fw.jpg
Normal file
BIN
public/maps/pagos_fw.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 548 KiB |
BIN
public/maps/pyros_fw.jpg
Normal file
BIN
public/maps/pyros_fw.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 601 KiB |
BIN
public/mutation.png
Normal file
BIN
public/mutation.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 793 B |
Loading…
Add table
Reference in a new issue