feat: implement basic fairy tracker
This commit is contained in:
parent
1e46ad269e
commit
8109bacb0d
27 changed files with 290 additions and 7 deletions
|
@ -1,6 +1,6 @@
|
|||
body {
|
||||
font-family: sans-serif;
|
||||
margin: 10px 40px;
|
||||
margin: 10px 25px;
|
||||
}
|
||||
|
||||
header {
|
||||
|
@ -31,6 +31,7 @@ header .muted {
|
|||
|
||||
main {
|
||||
display: grid;
|
||||
gap: 5px;
|
||||
grid-template-columns: 5fr 1fr;
|
||||
}
|
||||
|
||||
|
@ -214,3 +215,49 @@ span#password {
|
|||
color: #555;
|
||||
}
|
||||
|
||||
.fairies .map {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.fairies .fairy-dot {
|
||||
position: absolute;
|
||||
vertical-align: top;
|
||||
margin-top: -9px;
|
||||
margin-left: -1px;
|
||||
}
|
||||
|
||||
.fairy-marker {
|
||||
font-weight: bold;
|
||||
font-size: 12px;
|
||||
}
|
||||
|
||||
.fairy-a {
|
||||
color: #cf0e43;
|
||||
}
|
||||
|
||||
.fairy-b {
|
||||
color: gold;
|
||||
}
|
||||
|
||||
.fairy-c {
|
||||
color: aqua;
|
||||
}
|
||||
|
||||
.fairy-list {
|
||||
margin-top: 5px;
|
||||
font-weight: bold;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 5px 10px;
|
||||
}
|
||||
|
||||
.fairy-list > div {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
}
|
||||
|
||||
.fairy-list .note {
|
||||
font-weight: normal;
|
||||
color: #555;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
|
34
app/controllers/fairy_controller.rb
Normal file
34
app/controllers/fairy_controller.rb
Normal file
|
@ -0,0 +1,34 @@
|
|||
class FairyController < ApplicationController
|
||||
def mark
|
||||
instance_id, location = mark_params
|
||||
instance = Instance.find_by(public_id: instance_id)
|
||||
fairy = Fairy.new(location: location, instance_id: instance.id)
|
||||
unless params[:pwd]
|
||||
fairy.is_suggested = true
|
||||
end
|
||||
|
||||
if fairy.save
|
||||
new_instance = Instance.includes(:fairies).find(instance.id)
|
||||
render partial: "instance/fairies", locals: { instance: new_instance }
|
||||
end
|
||||
end
|
||||
|
||||
def unmark
|
||||
instance_id, location = mark_params
|
||||
if params[:pwd]
|
||||
instance = Instance.find_by(public_id: instance_id)
|
||||
Fairy.delete_by(instance_id: instance.id, location: location)
|
||||
new_instance = Instance.includes(:fairies).find(instance.id)
|
||||
render partial: "instance/fairies", locals: { instance: new_instance }
|
||||
end
|
||||
end
|
||||
|
||||
def despawn
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def mark_params
|
||||
params.expect(:instance, :location)
|
||||
end
|
||||
end
|
|
@ -13,7 +13,7 @@ class InstanceController < ApplicationController
|
|||
end
|
||||
|
||||
def show
|
||||
@instance = Instance.includes(:pops).find_by(public_id: show_instance_params)
|
||||
@instance = Instance.includes(:pops, :fairies).find_by(public_id: show_instance_params)
|
||||
@forecast = Weather.forecast(@instance.zone.to_sym)
|
||||
end
|
||||
|
||||
|
@ -23,7 +23,7 @@ class InstanceController < ApplicationController
|
|||
if parent_instance.password == pwd
|
||||
pop = Pop.new(instance_id: parent_instance.id, name: nm)
|
||||
if pop.save
|
||||
@instance = Instance.includes(:pops).find_by(public_id: instance_id)
|
||||
@instance = Instance.includes(:pops, :fairies).find_by(public_id: instance_id)
|
||||
@forecast = Weather.forecast(@instance.zone.to_sym)
|
||||
render partial: "list", locals: { instance: @instance, forecast: @forecast }
|
||||
end
|
||||
|
@ -35,7 +35,7 @@ class InstanceController < ApplicationController
|
|||
parent_instance = Instance.find_by(public_id: instance_id)
|
||||
if parent_instance.password == pwd
|
||||
Pop.delete_by(instance_id: parent_instance.id, name: nm)
|
||||
@instance = Instance.includes(:pops).find_by(public_id: instance_id)
|
||||
@instance = Instance.includes(:pops, :fairies).find_by(public_id: instance_id)
|
||||
@forecast = Weather.forecast(@instance.zone.to_sym)
|
||||
render partial: "list", locals: { instance: @instance, forecast: @forecast }
|
||||
end
|
||||
|
|
2
app/helpers/fairy_helper.rb
Normal file
2
app/helpers/fairy_helper.rb
Normal file
|
@ -0,0 +1,2 @@
|
|||
module FairyHelper
|
||||
end
|
|
@ -19,4 +19,20 @@ module InstanceHelper
|
|||
end
|
||||
false
|
||||
end
|
||||
|
||||
def get_map_x(zone)
|
||||
case zone
|
||||
when "anemos" then 1584
|
||||
end
|
||||
end
|
||||
|
||||
def get_map_y(zone)
|
||||
case zone
|
||||
when "anemos" then 1249
|
||||
end
|
||||
end
|
||||
|
||||
def has_fairy?(instance, fairy)
|
||||
instance.fairies.any? { |f| "#{fairy[:x]},#{fairy[:y]}" == f.location }
|
||||
end
|
||||
end
|
||||
|
|
|
@ -13,7 +13,7 @@ function checkPwd() {
|
|||
el.classList.add("hidden");
|
||||
});
|
||||
|
||||
const buttons = document.querySelectorAll(".action button");
|
||||
const buttons = document.querySelectorAll(".action");
|
||||
buttons.forEach(btn => {
|
||||
const oldUrl= btn.getAttribute("hx-post");
|
||||
btn.setAttribute("hx-post", `${oldUrl}&pwd=${pwd}`);
|
||||
|
|
3
app/models/fairy.rb
Normal file
3
app/models/fairy.rb
Normal file
|
@ -0,0 +1,3 @@
|
|||
class Fairy < ApplicationRecord
|
||||
belongs_to :instance
|
||||
end
|
|
@ -1,5 +1,6 @@
|
|||
class Instance < ApplicationRecord
|
||||
has_many :pops
|
||||
has_many :fairies
|
||||
|
||||
validates :zone, inclusion: { in: %w[anemos pagos pyros hydatos] }
|
||||
end
|
||||
|
|
2
app/views/fairy/despawn.html.erb
Normal file
2
app/views/fairy/despawn.html.erb
Normal file
|
@ -0,0 +1,2 @@
|
|||
<h1>Fairy#despawn</h1>
|
||||
<p>Find me in app/views/fairy/despawn.html.erb</p>
|
2
app/views/fairy/mark.html.erb
Normal file
2
app/views/fairy/mark.html.erb
Normal file
|
@ -0,0 +1,2 @@
|
|||
<h1>Fairy#mark</h1>
|
||||
<p>Find me in app/views/fairy/mark.html.erb</p>
|
2
app/views/fairy/suggest.html.erb
Normal file
2
app/views/fairy/suggest.html.erb
Normal file
|
@ -0,0 +1,2 @@
|
|||
<h1>Fairy#suggest</h1>
|
||||
<p>Find me in app/views/fairy/suggest.html.erb</p>
|
2
app/views/fairy/unmark.html.erb
Normal file
2
app/views/fairy/unmark.html.erb
Normal file
|
@ -0,0 +1,2 @@
|
|||
<h1>Fairy#unmark</h1>
|
||||
<p>Find me in app/views/fairy/unmark.html.erb</p>
|
83
app/views/instance/_fairies.html.erb
Normal file
83
app/views/instance/_fairies.html.erb
Normal file
|
@ -0,0 +1,83 @@
|
|||
<div class="fairies" id="fairies">
|
||||
<div class="map">
|
||||
<img src="/maps/<%= instance.zone %>.jpg" style="width: 100%;" />
|
||||
<% APP_DATA[instance.zone.to_sym][:fairies].each do |fairy| %>
|
||||
<div
|
||||
class="<%= class_names(unclaimed: !has_fairy?(instance, fairy)) %> fairy-dot"
|
||||
style="left: <%= fairy[:x].to_f / get_map_x(instance.zone) * 100 %>%; top: <%= fairy[:y].to_f / get_map_y(instance.zone) * 100 %>%; ">
|
||||
<% if has_fairy?(instance, fairy) %>
|
||||
<% idx = instance.fairies.index { |f| f.location == "#{fairy[:x]},#{fairy[:y]}" } %>
|
||||
<% if idx == 0 %>
|
||||
<span class="fairy-a fairy-marker">A</span>
|
||||
<% elsif idx == 1 %>
|
||||
<span class="fairy-b fairy-marker">B</span>
|
||||
<% elsif idx == 2 %>
|
||||
<span class="fairy-c fairy-marker">C</span>
|
||||
<% else %>
|
||||
<span class="other-fairy fairy-marker"><%= idx + 1 %></span>
|
||||
<% end %>
|
||||
<% else %>
|
||||
<div
|
||||
class="action button"
|
||||
hx-post="/fairy/mark?instance=<%= instance.public_id %>&location=<%= fairy[:x] %>,<%= fairy[:y] %>"
|
||||
hx-swap="outerHTML"
|
||||
hx-target="#fairies"
|
||||
hx-select="#fairies"
|
||||
>
|
||||
•
|
||||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
|
||||
<% if instance.fairies.empty? %>
|
||||
<div class="fairy-list">
|
||||
<div class="note">No elementals found yet! Click on any dot to mark that position.</div>
|
||||
</div>
|
||||
<% else %>
|
||||
<div class="fairy-list">
|
||||
<% instance.fairies.to_a.each_index do |idx| %>
|
||||
<div>
|
||||
<% if idx == 0 %>
|
||||
<div>
|
||||
Elemental <span class="fairy-a">A</span>
|
||||
</div>
|
||||
<% elsif idx == 1 %>
|
||||
<div>
|
||||
Elemental <span class="fairy-b">B</span>
|
||||
</div>
|
||||
<% elsif idx == 2 %>
|
||||
<div>
|
||||
Elemental <span class="fairy-c">C</span>
|
||||
</div>
|
||||
<% elsif idx > 2 %>
|
||||
<div>
|
||||
Elemental <span><%= idx + 1 %></span>
|
||||
</div>
|
||||
<% end %>
|
||||
|
||||
<div>
|
||||
<div
|
||||
class="copyable action needs_pwd"
|
||||
hx-post="/fairy/unmark?instance=<%= instance.public_id %>&location=<%= instance.fairies[idx].location %>"
|
||||
hx-swap="outerHTML"
|
||||
hx-target="#fairies"
|
||||
hx-select="#fairies"
|
||||
>
|
||||
remove</div>
|
||||
</div>
|
||||
</div>
|
||||
<% end %>
|
||||
<div class="note">Click on a black dot to mark another Elemental.</div>
|
||||
<div class="note">
|
||||
<div
|
||||
class="copyable"
|
||||
data-copy-content="Elementals: <%= instance.fairies[0..-1].map { |f| f.location }.join(", ") %>"
|
||||
>
|
||||
copy elemental positions for chat
|
||||
</div>
|
||||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
</div>
|
|
@ -56,12 +56,16 @@
|
|||
class="action reset"
|
||||
hx-post="/reset?instance=<%= @instance.public_id %>&nm=<%= nm[:name].parameterize %>"
|
||||
hx-target="#nm-list"
|
||||
hx-select="#nm-list"
|
||||
hx-swap="outerHTML"
|
||||
>Reset</button>
|
||||
<% else %>
|
||||
<button
|
||||
class="action"
|
||||
hx-post="/pop?instance=<%= @instance.public_id %>&nm=<%= nm[:name].parameterize %>"
|
||||
hx-target="#nm-list"
|
||||
hx-select="#nm-list"
|
||||
hx-swap="outerHTML"
|
||||
>Pop</button>
|
||||
<% end %>
|
||||
</div>
|
||||
|
|
|
@ -40,6 +40,10 @@
|
|||
</div>
|
||||
<% end %>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<%= render partial: "fairies", locals: { instance: @instance } %>
|
||||
</div>
|
||||
</div>
|
||||
</main>
|
||||
|
||||
|
|
|
@ -5,6 +5,8 @@ Rails.application.routes.draw do
|
|||
post "/pop", to: "instance#pop", as: :pop_in_instance
|
||||
post "/reset", to: "instance#reset", as: :reset_in_instance
|
||||
post "/auth", to: "instance#authenticate", as: :authenticate_to_instance
|
||||
post "/fairy/mark", to: "fairy#mark", as: :mark_fairy
|
||||
post "/fairy/unmark", to: "fairy#unmark", as: :unmark_fairy
|
||||
get "/:public_id", to: "instance#show", as: :show_instance
|
||||
|
||||
get "up" => "rails/health#show", as: :rails_health_check
|
||||
|
|
|
@ -5,6 +5,41 @@ weather = [
|
|||
["snow", 10]
|
||||
]
|
||||
|
||||
fairies = [
|
||||
{ x = 322, y = 125 },
|
||||
{ x = 444, y = 169 },
|
||||
{ x = 453, y = 309 },
|
||||
{ x = 134, y = 455 },
|
||||
{ x = 204, y = 566 },
|
||||
{ x = 180, y = 815 },
|
||||
{ x = 373, y = 1001 },
|
||||
{ x = 384, y = 731 },
|
||||
{ x = 349, y = 678 },
|
||||
{ x = 424, y = 634 },
|
||||
{ x = 567, y = 467 },
|
||||
{ x = 656, y = 513 },
|
||||
{ x = 675, y = 600 },
|
||||
{ x = 577, y = 699 },
|
||||
{ x = 808, y = 1107 },
|
||||
{ x = 914, y = 1047 },
|
||||
{ x = 829, y = 873 },
|
||||
{ x = 905, y = 738 },
|
||||
{ x = 834, y = 540 },
|
||||
{ x = 798, y = 304 },
|
||||
{ x = 698, y = 150 },
|
||||
{ x = 921, y = 401 },
|
||||
{ x = 1056, y = 466 },
|
||||
{ x = 1200, y = 543 },
|
||||
{ x = 1034, y = 616 },
|
||||
{ x = 1140, y = 756 },
|
||||
{ x = 1216, y = 862 },
|
||||
{ x = 1481, y = 591 },
|
||||
{ x = 1438, y = 242 },
|
||||
{ x = 1170, y = 161 },
|
||||
{ x = 1058, y = 240 },
|
||||
{ x = 1204, y = 308 }
|
||||
]
|
||||
|
||||
[[nms]]
|
||||
name = "Sabotender Corrido"
|
||||
level = 1
|
||||
|
|
|
@ -8,5 +8,7 @@ class CreateInstances < ActiveRecord::Migration[8.0]
|
|||
|
||||
t.timestamps
|
||||
end
|
||||
|
||||
add_index :instances, :public_id
|
||||
end
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
class CreateFairies < ActiveRecord::Migration[8.0]
|
||||
def change
|
||||
create_table :fairies do |t|
|
||||
t.integer :location
|
||||
t.string :location
|
||||
t.boolean :is_despawned
|
||||
t.boolean :is_suggested
|
||||
t.references :instance, null: false, foreign_key: true
|
||||
|
|
3
db/schema.rb
generated
3
db/schema.rb
generated
|
@ -15,7 +15,7 @@ ActiveRecord::Schema[8.0].define(version: 2025_03_10_183912) do
|
|||
enable_extension "pg_catalog.plpgsql"
|
||||
|
||||
create_table "fairies", force: :cascade do |t|
|
||||
t.integer "location"
|
||||
t.string "location"
|
||||
t.boolean "is_despawned"
|
||||
t.boolean "is_suggested"
|
||||
t.bigint "instance_id", null: false
|
||||
|
@ -31,6 +31,7 @@ ActiveRecord::Schema[8.0].define(version: 2025_03_10_183912) do
|
|||
t.string "zone"
|
||||
t.datetime "created_at", null: false
|
||||
t.datetime "updated_at", null: false
|
||||
t.index ["public_id"], name: "index_instances_on_public_id"
|
||||
end
|
||||
|
||||
create_table "pops", force: :cascade do |t|
|
||||
|
|
BIN
public/maps/anemos.jpg
Normal file
BIN
public/maps/anemos.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 499 KiB |
BIN
public/maps/hydatos.jpg
Normal file
BIN
public/maps/hydatos.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 532 KiB |
BIN
public/maps/pagos.jpg
Normal file
BIN
public/maps/pagos.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 548 KiB |
BIN
public/maps/pyros.jpg
Normal file
BIN
public/maps/pyros.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 601 KiB |
23
test/controllers/fairy_controller_test.rb
Normal file
23
test/controllers/fairy_controller_test.rb
Normal file
|
@ -0,0 +1,23 @@
|
|||
require "test_helper"
|
||||
|
||||
class FairyControllerTest < ActionDispatch::IntegrationTest
|
||||
test "should get mark" do
|
||||
get fairy_mark_url
|
||||
assert_response :success
|
||||
end
|
||||
|
||||
test "should get suggest" do
|
||||
get fairy_suggest_url
|
||||
assert_response :success
|
||||
end
|
||||
|
||||
test "should get unmark" do
|
||||
get fairy_unmark_url
|
||||
assert_response :success
|
||||
end
|
||||
|
||||
test "should get despawn" do
|
||||
get fairy_despawn_url
|
||||
assert_response :success
|
||||
end
|
||||
end
|
11
test/fixtures/fairies.yml
vendored
Normal file
11
test/fixtures/fairies.yml
vendored
Normal file
|
@ -0,0 +1,11 @@
|
|||
# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
|
||||
|
||||
# This model initially had no columns defined. If you add columns to the
|
||||
# model remove the "{}" from the fixture names and add the columns immediately
|
||||
# below each fixture, per the syntax in the comments below
|
||||
#
|
||||
one: {}
|
||||
# column: value
|
||||
#
|
||||
two: {}
|
||||
# column: value
|
7
test/models/fairy_test.rb
Normal file
7
test/models/fairy_test.rb
Normal file
|
@ -0,0 +1,7 @@
|
|||
require "test_helper"
|
||||
|
||||
class FairyTest < ActiveSupport::TestCase
|
||||
# test "the truth" do
|
||||
# assert true
|
||||
# end
|
||||
end
|
Loading…
Add table
Reference in a new issue