diff options
-rw-r--r-- | app/controllers/characters_controller.rb | 15 | ||||
-rw-r--r-- | app/controllers/game_controller.rb | 8 | ||||
-rw-r--r-- | app/models/character.rb | 49 | ||||
-rw-r--r-- | app/models/monster.rb | 8 | ||||
-rw-r--r-- | app/views/characters/show.html.erb | 13 | ||||
-rw-r--r-- | config/routes.rb | 1 | ||||
-rw-r--r-- | db/migrate/20210526225100_add_combat_styles_to_characters.rb | 6 | ||||
-rw-r--r-- | db/schema.rb | 4 |
8 files changed, 86 insertions, 18 deletions
diff --git a/app/controllers/characters_controller.rb b/app/controllers/characters_controller.rb index d851ad5..77e1a94 100644 --- a/app/controllers/characters_controller.rb +++ b/app/controllers/characters_controller.rb @@ -21,6 +21,21 @@ class CharactersController < ApplicationController end end + def set_combat_styles + @character = Character.find(params[:character_id]) + unless @character == current_char + flash[:alert] = "You can't set the combat styles of another character." + redirect_to character_path(@character) and return + end + if @character.update(offensive_style: params[:offensive_style], + defensive_style: params[:defensive_style]) + flash[:notice] = "Changed combat styles to #{@character.offensive_style} and #{@character.defensive_style}." + else + flash[:alert] = "Failed to set combat styles." + end + redirect_to character_path(@character) + end + private def character_params params.require(:character).permit(:name) diff --git a/app/controllers/game_controller.rb b/app/controllers/game_controller.rb index 32ddd12..922301a 100644 --- a/app/controllers/game_controller.rb +++ b/app/controllers/game_controller.rb @@ -149,15 +149,15 @@ class GameController < ApplicationController end turn_order.cycle do |actor, target| base_accuracy_roll = roll(20) - accuracy_roll = base_accuracy_roll + actor.accuracy - evasion_roll = roll(20) + target.evasion + accuracy_roll = base_accuracy_roll + actor.accuracy(with_combat_style: true) + evasion_roll = roll(20) + target.evasion(with_combat_style: true) if accuracy_roll >= evasion_roll - dealt_damage = roll(4) + actor.power # TODO: Replace d4 with weapon damage + dealt_damage = roll(4) + actor.power(with_combat_style: true) # TODO: Replace d4 with weapon damage if base_accuracy_roll == 20 combat_message.call("#{actor.name} landed a critical hit!") dealt_damage = dealt_damage * 2 end - blocked_damage = (accuracy_roll >= (roll(20) + target.block)) ? 0 : target.block_value + blocked_damage = (accuracy_roll >= (roll(20) + target.block(with_combat_style: true))) ? 0 : target.block_value resolved_damage = dealt_damage - blocked_damage actor == char ? mon_hp -= resolved_damage : char_hp -= resolved_damage damage_text = "#{resolved_damage} damage." diff --git a/app/models/character.rb b/app/models/character.rb index 52c0959..d1faeb0 100644 --- a/app/models/character.rb +++ b/app/models/character.rb @@ -19,7 +19,10 @@ class Character < ApplicationRecord attribute :wounds, :integer, default: 0 - after_create :create_skills + enum offensive_style: [:balanced, :precise, :brutal] + enum defensive_style: [:centered, :elusive, :protective] + + after_create :create_skills, :set_combat_styles after_create { Hearth.create(character: self) } def beastslay_level; skill_level("beastslay"); end @@ -184,20 +187,44 @@ class Character < ApplicationRecord self.beastslay_level + (equipment_stats["speed"] || 0) end - def accuracy - self.beastslay_level + (equipment_stats["accuracy"] || 0) + def accuracy(with_combat_style: false) + base = self.beastslay_level + (equipment_stats["accuracy"] || 0) + if with_combat_style && self.precise? + base = (base * 1.25).floor + elsif with_combat_style && self.brutal? + base = (base * 0.75).ceil + end + base end - def power - self.beastslay_level + (equipment_stats["power"] || 0) + def power(with_combat_style: false) + base = self.beastslay_level + (equipment_stats["power"] || 0) + if with_combat_style && self.precise? + base = (base * 0.75).ceil + elsif with_combat_style && self.brutal? + base = (base * 1.25).floor + end + base end - def evasion - self.beastslay_level + (equipment_stats["evasion"] || 0) + def evasion(with_combat_style: false) + base = self.beastslay_level + (equipment_stats["evasion"] || 0) + if with_combat_style && self.elusive? + base = (base * 1.25).floor + elsif with_combat_style && self.protective? + base = (base * 0.75).ceil + end + base end - def block - self.beastslay_level + (equipment_stats["block"] || 0) + def block(with_combat_style: false) + base = self.beastslay_level + (equipment_stats["block"] || 0) + if with_combat_style && self.elusive? + base = (base * 0.75).ceil + elsif with_combat_style && self.protective? + base = (base * 1.25).floor + end + base end def block_value @@ -208,4 +235,8 @@ class Character < ApplicationRecord def create_skills Skill.all.each { |skill| self.character_skills.create(skill: skill, xp: 0) } end + + def set_combat_styles + self.update(offensive_style: :balanced, defensive_style: :centered) + end end diff --git a/app/models/monster.rb b/app/models/monster.rb index 1b521aa..d1e1b99 100644 --- a/app/models/monster.rb +++ b/app/models/monster.rb @@ -9,19 +9,19 @@ class Monster < ApplicationRecord self.whatnot[:speed][:base] end - def accuracy + def accuracy(with_combat_style: false) self.whatnot[:accuracy][:base] end - def power + def power(with_combat_style: false) self.whatnot[:power][:base] end - def evasion + def evasion(with_combat_style: false) self.whatnot[:evasion][:base] end - def block + def block(with_combat_style: false) self.whatnot[:block][:base] end diff --git a/app/views/characters/show.html.erb b/app/views/characters/show.html.erb index e7f9696..84e1ff5 100644 --- a/app/views/characters/show.html.erb +++ b/app/views/characters/show.html.erb @@ -11,6 +11,19 @@ <p class="mb-4">Learned <%= @character.learned_activities.count %> recipe(s) or technique(s).</p> +<% if @character == current_char %> + <h2 class="text-2xl mb-4">Combat Styles</h2> + <%= form_with url: character_combat_styles_path(character_id: @character) do |f| %> + <%= f.label :offensive_style, "Offensive" %> + <%= f.select :offensive_style, Character.offensive_styles.keys.to_a, selected: @character.offensive_style %> + + <%= f.label :defensive_style, "Defensive" %> + <%= f.select :defensive_style, Character.defensive_styles.keys.to_a, selected: @character.defensive_style %> + + <%= f.submit "Set" %> + <% end %> +<% end %> + <div class="my-6"> <h2 class="text-2xl mb-4">Combat Statistics</h2> <table class="table-auto"> diff --git a/config/routes.rb b/config/routes.rb index 514ab2e..95071d7 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -16,6 +16,7 @@ Rails.application.routes.draw do resources :activities, only: [:show] resources :characters, only: [:show, :new, :create] do + post "/combat_styles", to: "characters#set_combat_styles" scope module: :characters do post "/items/unequip/:slot", to: "items#unequip", as: :item_unequip resources :items, only: [:index] do diff --git a/db/migrate/20210526225100_add_combat_styles_to_characters.rb b/db/migrate/20210526225100_add_combat_styles_to_characters.rb new file mode 100644 index 0000000..8ccbfc4 --- /dev/null +++ b/db/migrate/20210526225100_add_combat_styles_to_characters.rb @@ -0,0 +1,6 @@ +class AddCombatStylesToCharacters < ActiveRecord::Migration[6.1] + def change + add_column :characters, :offensive_style, :integer + add_column :characters, :defensive_style, :integer + end +end diff --git a/db/schema.rb b/db/schema.rb index bd2b95b..74bb4e6 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -10,7 +10,7 @@ # # It's strongly recommended that you check this file into your version control system. -ActiveRecord::Schema.define(version: 2021_05_26_000349) do +ActiveRecord::Schema.define(version: 2021_05_26_225100) do # These are extensions that must be enabled in order to support this database enable_extension "plpgsql" @@ -81,6 +81,8 @@ ActiveRecord::Schema.define(version: 2021_05_26_000349) do t.bigint "active_title_id" t.integer "wounds" t.integer "queued_actions" + t.integer "offensive_style" + t.integer "defensive_style" t.index ["active_title_id"], name: "index_characters_on_active_title_id" t.index ["activity_id"], name: "index_characters_on_activity_id" t.index ["user_id"], name: "index_characters_on_user_id" |