diff options
50 files changed, 1659 insertions, 296 deletions
diff --git a/CHANGELOG.md b/CHANGELOG.md index 8e99480..55675f7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,188 @@ # Changelog All notable changes to this project will be documented in this file. +## [0.1.12] - 2021-06-23 + +### General +- Equipment and infix breakage changes + - All threatened equipment and infixes are no longer checked individually for breakage. Instead, a single breakage + roll occurs, and may cause equipment and/or infix breakage depending on its result. This means, at most, only + one piece of equipment and one infix can break per timer. It also means that in many cases, fewer breakage rolls + are happening per timer. + - Equipment breakage + - A random slot is determined for breakage (or no slot, if nothing breaks). If a slot is + selected in which you have nothing equipped, then no item breaks. Combat is still the only case where all + equipped items are threatened. Everywhere else, only items providing you with a skill speed increase are + threatened. + - There is a 1/1500 chance for the left ring, right ring, or neck slot to be targeted. If the roll was too + low for this to happen, the same roll is checked against a 1/1000 chance. If the roll is high enough to meet + the 1/1000 chance, the back, waist, or curio slot is targeted. If the roll is also too low for this to happen, + the same roll is checked against a 1/500 chance, and if it is high enough, one of the other slots are targeted. + - Infix breakage (fading) + - If the break roll succeeds (AKA something breaks), a random infix from the relevant skill breaks. + - There is a 1/100 chance for the break roll to succeed. This means that infixes will be used much more rapidly + than they were before this release (except for people with 5 or more infix slots, which no one has at the time + of this writing). + +### Locations +- "Floret Region" has been renamed to "Floret", and its description has been changed. + +### Activities +- Skill level-based activity duration scaling has been cut in half. That means you should now see a 0.5 second timer + decrease every level, instead of 1 second. +- Varying and widspread nerfs to XP rewards. This is a much-needed flattening of the curve; it's clear after nearly + a month of playtesting the the XP rewards are so high they heavily mitigate the intended effect of the XP curve. + Sorry to the lower-leveled players, but this is part of the playtest, plus you have the opportunity to test out + the new progression. This might look like you've been screwed, but really, when you think about it, you've actually + been gifted with opportunity. (Also there will be a server wipe before too long so don't sweat it.) + +### Items +- Apprentice wand can now be equipped in mainhand or offhand. + +### Fixed +- Results and chat areas sometimes didn't scroll to bottom when they should. This was caused by large result outputs + being larger than the scroll tolerance. +- Activities without any requirements (such as opening an unlocked rusted lockbox or building a level 1 hearth + amenity) would fail. +- Havencast spells were not properly scaling with havencast speed. Now, the apprentice wand should actually have an + effect on cast time. + +## [0.1.11.3] - 2021-06-22 + +### UI +- Leaderboard: Skill leaderboards are now always displayed alphabetically. + +### Fixed +- Couldn't unlock rusted lockboxes due to equipment requirement being set incorrectly. +- Couldn't change combat styles. +- Crafting bluster powder was granting fluxseethe XP instead of spicework XP. + +## [0.1.11.2] - 2021-06-16 + +## UI +- XPTNL is now displayed on the Skill view, along with the new progress bars. +- Skill view grid breakpoints adjusted, so less squishing. + +## [0.1.11.1] - 2021-06-16 + +### Fixed +- Bug where having an item equipped which has no game data effects (such as trophies, the effects of which are + hardcoded) would cause activities to fail to complete. + +## [0.1.11] - 2021-06-16 + +### General +- Any items equipped which provide a speed increase to a skill now have a chance to break when performing that skill. + The break chance is based on the equip slot of the item, and uses the same numbers as the combat equipment break + chance. The combat equipment break check process is unchanged. +- Omenbind has been implemented. + - Omens can be bound once a character has constructed a binding array in their hearth. + - Omens are tradable items that can be infixed in a particular skill via the new skills UI (see the UI section of this + changelog). + - A single omen can be infixed in a skill at skill level 1 (the level of the skill you're infixing into, not the level + of your omenbind skill, unless of course you're infixing into your omenbind skill). At level 20 and every 20 levels + thereafter, another infix slot becomes available. + - Omens can be removed and re-infixed at will. + - All omens infixed in a skill have a chance to fade away and be permanently lost when performing that skill + (like equipment breakage). The chance of an infixed omen fading away is 1/500. You do not need to actively be + getting a bonus from an infixed omen for it to have a chance to fade away. When performing a skill, all omens + infixed in that skill have the same chance to fade away. + - Duplicate omens can be infixed, and their effects will stack if possible. +- Havencast has been implemented. + - Spells can be cast from a new "Spells" view, accessible with the navigation. + - Spells typically require that a certain amount of mana is available. Mana can be made available by infixing certain + omens. + - Cantrips are spells which a character can cast by simply having the appropriate requirements and paying any cost. + Other spells must be learned from a spellpage by casting a decipher magicscript cantrip on the spellpage, + consuming it. Each spellpage can contain a different spell, and the one it provides isn't determined until + deciphering and consuming the spellpage. If the spell on the spellpage is one a character doesn't know yet, they + will learn the spell upon deciphering the spellpage. + - Non-cantrip spells also require that a spell focus is equipped. +- Characters now each have a bestiary where monster kills are recorded. For leviathans, only the character who actually + slays the leviathan by dealing the final damage gets credit for the kill in their bestiary. The bestiary can be viewed + via a link on the Character view. (Hopefully) obviously, kills before this release were not counted. + +### Items +- New items: faint mana, minor mana, faint forging haste, faint seething haste, faint working haste, simple spellpage, + apprentice wand +- Iron lockpicks now increase synthsever speed by 10 +- Axes, pickaxes, and spades skill speed increase numbers all adjusted to compensate for base timer reductions + +### Activities +- Planequarry activities now require a pickaxe, wealdreap activities now require an axe, magiculture activities now + require a spade, and base timers and item speed bonuses have been adjusted appropriately +- Planequarry at the Brine Trench base duration changed from 60 to 50 (should have been like, 120 before but it was + mistakenly set to 60, and now it's being adjusted for the tool/timer changes in this release) +- Planequarry XP award changes + - Pure iron ore 25 -> 18 + - Gaian ore 50 -> 25 +- Otherforge + - Base duration changes + - Purify crude iron ingot 105 -> 60 + - Purify iron ingot 115 -> 70 + - Level requirements changes + - Arcanite longsword 15 -> 17 (setting it to 15 was a programming error) +- Havencast + - New cantrips: Light, Dazzle, Decipher Simple Magicscript, Enchant Apprentice Wand + - New spells: Stinging Rays, Flame Whirl +- Wildscour + - Crumbling ruins now has a chance to drop a simple spellpage + +### Monsters +- Grinpads now have a chance to drop a simple spellpage + +### Hearth +- New amenity: binding array (level 1, level 2) + +### Titles +- 7 new titles + - Slayer: Kill 1,000 monsters + - Butcher: Kill 10,000 monsters + - Slaughterer: Kill 100,000 monsters + - Massacrer: Kill 1,000,000 monsters + - Spiteful: Kill 1,000 of one particular monster + - Hateful: Kill 10,000 of one particular monster + - Vicious: Kill 100,000 of one particular monster + +### UI +- Character skills were moved from the Character view to their own view, which features a new interface with XP bars. +- Reduced space between nav menu items + +## [0.1.10.3] - 2021-06-14 + +### Fixed +- Bug in title award code. + +## [0.1.10.2] - 2021-06-10 + +### Activities +- Smelt ingot activities XP changes + - Crude iron ingot 4 -> 6 + - Iron ingot 8 -> 12 + - Pure iron ingot 12 -> 18 + - Arcanite ingot 16 -> 24 +- Purify ingot activities XP changes + - Purify crude iron ingot 8 -> 12 + - Purfiy iron ingot 12 -> 18 +- Smelting ingot cost reduced from 10 ore to 8 ore +- Purifying ingot cost reduced from 5 ingots to 4 ingots + +### Items +- Reduced skill speed increases of arcanite axe, pickaxe, and spade from 100 to 90. + +### Fixed +- Capitalized "Laboratory" where it ought to be capitalized. + +## [0.1.10.1] - 2021-06-10 + +### UI +- Results box now insta-scrolls to bottom on any page change. Previously it would be back at the top again if you + clicked a menu link or something. This meant that auto-scroll wouldn't be working on page change due to the + scroll changes in 0.1.10. + +### Fixed +- Fatal bug when trying to access the New Character screen, introduced by the recent major UI changes. + ## [0.1.10] - 2021-06-09 ### Leviathans diff --git a/app/controllers/characters/bestiary_controller.rb b/app/controllers/characters/bestiary_controller.rb new file mode 100644 index 0000000..7e37d9c --- /dev/null +++ b/app/controllers/characters/bestiary_controller.rb @@ -0,0 +1,12 @@ +class Characters::BestiaryController < ApplicationController + before_action :set_character + + def index + @monster_kills = @character.monster_kills.all + end + + private + def set_character + @character = Character.find(params[:character_id]) + end +end diff --git a/app/controllers/characters/hearth_controller.rb b/app/controllers/characters/hearth_controller.rb index 82f72d6..c525add 100644 --- a/app/controllers/characters/hearth_controller.rb +++ b/app/controllers/characters/hearth_controller.rb @@ -7,6 +7,7 @@ class Characters::HearthController < ApplicationController forge: [], laboratory: [], spicebench: [], + binding_array: [], } Activity.where("gid like ?", "craft_%").each do |activity| @@ -20,6 +21,8 @@ class Characters::HearthController < ApplicationController @amenity_activities[:laboratory].push(activity) && next when "spicebench" @amenity_activities[:spicebench].push(activity) && next + when "binding_array" + @amenity_activities[:binding_array].push(activity) && next else raise "Invalid amenity gid (#{requirement_data[:gid]}" end diff --git a/app/controllers/characters/item_infixes_controller.rb b/app/controllers/characters/item_infixes_controller.rb new file mode 100644 index 0000000..0b5f1c5 --- /dev/null +++ b/app/controllers/characters/item_infixes_controller.rb @@ -0,0 +1,27 @@ +class Characters::ItemInfixesController < ApplicationController + def create + # TODO: Can this find-by-id happen automagically? + @item_infix = current_char.infix(Item.find(params[:item_id]), Skill.find(params[:skill_id])) + if @item_infix + flash[:notice] = "Infixed #{@item_infix.item.name}." + else + flash[:alert] = "Failed to infix item." + end + redirect_to character_skills_path(current_char) + end + + def destroy + @item_infix = ItemInfix.find(params[:id]) + if current_char.remove_infix(@item_infix) + flash[:notice] = "Removed #{@item_infix.item.name}." + else + flash[:alert] = "Failed to remove #{@item_infix.item.name}." + end + redirect_to character_skills_path(current_char) + end + + private + def item_infix_params + params.require(:item_infix).permit(:item_id, :skill_id) + end +end diff --git a/app/controllers/characters/items_controller.rb b/app/controllers/characters/items_controller.rb index 470e21c..e38b69a 100644 --- a/app/controllers/characters/items_controller.rb +++ b/app/controllers/characters/items_controller.rb @@ -1,6 +1,7 @@ class Characters::ItemsController < ApplicationController + before_action :set_character, only: :index + def index - @character = Character.find(params[:character_id]) end def equip @@ -66,4 +67,13 @@ class Characters::ItemsController < ApplicationController redirect_to character_items_path(current_char) end end + + private + def set_character + @character = Character.find(params[:character_id]) + unless current_char == @character + flash[:alert] = "You can only look at your own items." + redirect_to character_path(@character) + end + end end diff --git a/app/controllers/characters/rankings_controller.rb b/app/controllers/characters/rankings_controller.rb index bbae9fc..429b487 100644 --- a/app/controllers/characters/rankings_controller.rb +++ b/app/controllers/characters/rankings_controller.rb @@ -6,4 +6,4 @@ class Characters::RankingsController < ApplicationController def index @character = Character.find(params[:character_id]) end -end
\ No newline at end of file +end diff --git a/app/controllers/characters/skills_controller.rb b/app/controllers/characters/skills_controller.rb new file mode 100644 index 0000000..6fcf417 --- /dev/null +++ b/app/controllers/characters/skills_controller.rb @@ -0,0 +1,15 @@ +class Characters::SkillsController < ApplicationController + before_action :set_character, only: :index + + def index + end + + private + def set_character + @character = Character.find(params[:character_id]) + unless current_char == @character + flash[:alert] = "You can only look at your own skills." + redirect_to character_path(@character) + end + end +end diff --git a/app/controllers/characters/spells_controller.rb b/app/controllers/characters/spells_controller.rb new file mode 100644 index 0000000..a0e6913 --- /dev/null +++ b/app/controllers/characters/spells_controller.rb @@ -0,0 +1,9 @@ +class Characters::SpellsController < ApplicationController + def index + @spell_activities = Activity.where("gid like ?", "havencast_%").where(innate: true).order(:name) + # TODO: Don't load into memory + @spell_activities = @spell_activities.to_a + current_char.learned_activities + .map { |la| la.activity } + .select { |a| a.gid.start_with?("havencast_") } + end +end diff --git a/app/controllers/characters_controller.rb b/app/controllers/characters_controller.rb index 77e1a94..2eb906b 100644 --- a/app/controllers/characters_controller.rb +++ b/app/controllers/characters_controller.rb @@ -1,8 +1,8 @@ class CharactersController < ApplicationController skip_before_action :redirect_if_no_active_character, only: [:new, :create] + before_action :set_character, only: [:show, :set_combat_styles] def show - @character = Character.find(params[:id]) end def new @@ -22,11 +22,6 @@ class CharactersController < ApplicationController 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}." @@ -40,4 +35,12 @@ class CharactersController < ApplicationController def character_params params.require(:character).permit(:name) end + + def set_character + @character = Character.find(params[:id]) + unless current_char == @character + flash[:alert] = "You can only manage your own character." + redirect_to character_path(@character) + end + end end diff --git a/app/controllers/leaderboard_controller.rb b/app/controllers/leaderboard_controller.rb index e6e8543..14b8acb 100644 --- a/app/controllers/leaderboard_controller.rb +++ b/app/controllers/leaderboard_controller.rb @@ -2,7 +2,7 @@ class LeaderboardController < ApplicationController def index - @top_per_skill = Hash[Skill.all.map { |s| [s.name.to_sym, CharacterSkill.top_xp_for(s)] }] + @top_per_skill = Hash[Skill.all.order(:name).map { |s| [s.name.to_sym, CharacterSkill.top_xp_for(s)] }] @top_total_xp = Character.top_total_xp @top_total_level = Character.top_total_level end diff --git a/app/errors/item_infix_error.rb b/app/errors/item_infix_error.rb new file mode 100644 index 0000000..bc6b251 --- /dev/null +++ b/app/errors/item_infix_error.rb @@ -0,0 +1,2 @@ +class ItemInfixError < StandardError +end diff --git a/app/javascript/channels/chat_room_channel.js b/app/javascript/channels/chat_room_channel.js index af29953..514742f 100644 --- a/app/javascript/channels/chat_room_channel.js +++ b/app/javascript/channels/chat_room_channel.js @@ -10,14 +10,19 @@ consumer.subscriptions.create("ChatRoomChannel", { }, received(data) { + // If scrolled to the bottom or near the bottom, then smooth scroll to the bottom. + var shouldScroll = false; + var chatOutputElement = document.getElementById("chat_output"); + if ((chatOutputElement.scrollTop + 100) >= (chatOutputElement.scrollHeight - chatOutputElement.offsetHeight)) { + shouldScroll = true; + } + // Called when there's incoming data on the websocket for this channel var node = document.createElement("P"); node.innerHTML = data.html; - var chatOutputElement = document.getElementById("chat_output"); chatOutputElement.appendChild(node); - // If scrolled to the bottom or near the bottom, then smooth scroll to the bottom. - if ((chatOutputElement.scrollTop + 100) >= (chatOutputElement.scrollHeight - chatOutputElement.offsetHeight)) { + if (shouldScroll) { chatOutputElement.scrollTo({ top: chatOutputElement.scrollHeight, left: 0, behavior: 'smooth' }); diff --git a/app/javascript/controllers/results_controller.js b/app/javascript/controllers/results_controller.js new file mode 100644 index 0000000..266b80c --- /dev/null +++ b/app/javascript/controllers/results_controller.js @@ -0,0 +1,13 @@ +import { Controller } from "stimulus"; + +export default class extends Controller { + static targets = [ "output" ]; + + connect() { + this.scrollToBottom(); + } + + scrollToBottom() { + this.outputTarget.scrollTop = this.outputTarget.scrollHeight; + } +} diff --git a/app/lib/activity_processor.rb b/app/lib/activity_processor.rb index b5dd11c..cca2185 100644 --- a/app/lib/activity_processor.rb +++ b/app/lib/activity_processor.rb @@ -90,6 +90,13 @@ class ActivityProcessor item = Item.find_by_gid(result[:gid]) hp = @character.hearth.hearth_plantings.create(item: item) @results.push({ type: type, hearth_planting: hp }) + when "condition" + Character.transaction do + condition = Condition.find_by_gid(result[:gid]) + @character.states.create!(condition: condition, expires_at: Time.now + result[:duration]) + @results.push({ type: "message", body: result[:message] }) + @results.push({ type: type, condition: condition }) + end when "activity" next if rand > (result[:chance] || 1) table_roll = rand @@ -97,6 +104,7 @@ class ActivityProcessor score = table_entry[:score] if table_roll >= score new_activity = Activity.find_by_gid(table_entry[:gid]) + raise "Invalid activity gid (#{table_entry[:gid]})" unless new_activity unless @character.learned_activities.exists?(activity: new_activity) @character.learned_activities.create(activity: new_activity) @results.push({ type: type, activity: new_activity }) @@ -108,6 +116,26 @@ class ActivityProcessor end end + # Note: This will result in equipment being checked for breakage twice (after combat, and now) if it provides the + # `beastslay_speed` stat. At the time of this writing, that stat doesn't exist. + # But just something to keep in mind. + # Note: This will result in equipment being checked twice if it provides two speed stats. + # Fine for now since no equipment gives two skill speed stats, but may want to refine in the future. + if @activity.whatnot[:requirements]&.any? + required_skill_gids = @activity.whatnot[:requirements].select { |r| r[:type] == "skill" }.map { |r| r[:gid] }.uniq + required_skill_gids.each do |required_skill_gid| + skill = Skill.find_by_gid(required_skill_gid) + broken_item = @character.do_equipment_break_check(skill: skill) + if broken_item + @results.push({ type: "warning", message: "Your #{broken_item.name} was damaged beyond repair!" }) + end + broken_infix_item = @character.do_item_infix_break_check(skill: skill) + if broken_infix_item + @results.push({ type: "warning", message: "Your #{broken_infix_item.name} omen faded away." }) + end + end + end + if @character.activity && @character.queued_actions if @character.queued_actions > 0 @character.queued_actions -= 1 @@ -166,7 +194,8 @@ class ActivityProcessor end def handle_title_result(data) - if @character.award_title(data[:gid]) + title = Title.find_by_gid(data[:gid]) + if @character.award_title(title) @results.push({ type: "title", title: title }) end end @@ -230,8 +259,13 @@ class ActivityProcessor turn_order = [[char, mon], [mon, char]].shuffle end turn_order.cycle do |actor, target| + base_accuracy_roll = roll(20) accuracy_roll = base_accuracy_roll + actor.accuracy(with_combat_style: true) + if actor == mon + accuracy_roll += char.total_enemy_stat_change("accuracy") + end + evasion_roll = roll(20) + target.evasion(with_combat_style: true) if accuracy_roll >= evasion_roll || base_accuracy_roll == 20 @@ -273,8 +307,22 @@ class ActivityProcessor combat_message.call("#{target.name} evaded #{actor.name}'s attack.") end + # HACK: DoT is char-only, and one-damage-type-only for now. + if actor == char + actor.dots.each do |data| + damage_roll = rand(data[:min]..data[:max]) + effective_resistance = [target.resistance(data[:gid]), damage_roll].min + resolved_damage = damage_roll - (effective_resistance * rand(0.5..1)).round + mon_hp -= resolved_damage + combat_message.call("#{data[:message]} (#{resolved_damage} #{data[:gid]})") + end + end + if char_hp < 1 || mon_hp < 1 - break_check + broken_item = @character.do_equipment_break_check + if broken_item + @results.push({ type: "warning", message: "Your #{broken_item.name} was damaged beyond repair!" }) + end if monster_spawn hp_lost = monster_spawn.remaining_hp - mon_hp @@ -295,6 +343,11 @@ class ActivityProcessor end else @results.push({ type: "message", body: "You slew the #{mon.name}." }) + + monster_kills = character.monster_kills.find_or_initialize_by(monster: mon) + monster_kills.quantity ? monster_kills.quantity += 1 : monster_kills.quantity = 1 + monster_kills.save + if monster_spawn char.stop_activity return @@ -317,10 +370,4 @@ class ActivityProcessor end end end - - def break_check - @character.do_equipment_break_checks.each do |broken_item| - @results.push({ type: "warning", message: "Your #{broken_item.name} was damaged beyond repair!" }) - end - end end diff --git a/app/models/character.rb b/app/models/character.rb index d386002..2669626 100644 --- a/app/models/character.rb +++ b/app/models/character.rb @@ -9,10 +9,12 @@ class Character < ApplicationRecord has_many :character_items has_many :learned_activities has_many :items, through: :character_items + has_many :item_infixes has_many :character_skills has_many :conditions, through: :states has_many :states has_many :chat_messages + has_many :monster_kills has_many :bazaar_orders validates :name, presence: true # TODO: Make defaults better. This has to allow nil so the `attribute` default works, and I don't like that. @@ -115,14 +117,30 @@ class Character < ApplicationRecord end end - def do_equipment_break_checks(exclude_slots: []) - broken_items = [] - equipment.where.not(slot: exclude_slots).each do |equipment| - if equipment.break_check - broken_items.push(equipment.item) - end + def do_equipment_break_check(skill: nil) + skill = Skill.find_by_gid(skill) if skill&.is_a? String + # TODO: HACK: Should check other stats besides speed stat in the future. + # TODO: HACK: May not want a chance to break if speed is _reduced_. Though no equipment does this yet. + if skill + threatened_equipment = equipment.all.select { |eq| eq.effects&.select { |ef| ef[:gid] == "#{skill.gid}_speed" }&.any? } + else + threatened_equipment = equipment.all end - broken_items + break_slot = Equipment.random_break_slot + return nil unless break_slot + broken_equipment = threatened_equipment.find { |eq| eq.slot == break_slot } + return nil unless broken_equipment + broken_equipment.destroy + broken_equipment.item + end + + def do_item_infix_break_check(skill:) + skill = Skill.find_by_gid(skill) if skill.is_a? String + return nil unless ItemInfix.break_check + broken_ii = item_infixes.where(skill: skill).sample + return nil unless broken_ii + broken_ii.destroy + broken_ii.item end def has_item?(item, quantity = 1) @@ -136,6 +154,10 @@ class Character < ApplicationRecord self.equipment.find_by(item: item) end + def equipment_with_tag(tag) + self.equipment.all.find { |e| e.item.has_tag?(tag) } + end + def open_slots_for(item) full_slots = self.equipment.map { |e| e.slot } item.equip_slots.reject { |slot| full_slots.include?(slot) } @@ -182,6 +204,30 @@ class Character < ApplicationRecord end end + def max_infixes(skill) + skill = Skill.find_by_gid(skill) if skill.is_a? String + 1 + (skill_level(skill) / 20).floor + end + + def available_infixes(skill) + skill = Skill.find_by_gid(skill) if skill.is_a? String + max_infixes(skill) - item_infixes.where(skill: skill).count + end + + def infix(item, skill) + Character.transaction do + shift_item(item, -1) + item_infixes.create(item: item, skill: skill) + end + end + + def remove_infix(item_infix) + Character.transaction do + shift_item(item_infix.item, 1) + item_infix.destroy + end + end + def add_skill_xp(skill, amount) skill = Skill.find_by_gid(skill) if skill.is_a? String Character.transaction do @@ -247,7 +293,14 @@ class Character < ApplicationRecord activity.whatnot[:requirements]&.each do |requirement| case requirement[:type] when "equipment" - return false unless self.equipment_with_gid(requirement[:gid]) + if requirement[:tag] + return false unless self.equipment_with_tag(requirement[:tag]) + else + return false unless self.equipment_with_gid(requirement[:gid]) + end + when "stat" + # TODO: HACK: This won't work with built-in stats! Need to change this to work with power and whatnot. + return false unless self.total_stat_change(requirement[:gid]) >= requirement[:value] when "skill" return false unless self.skill_level(requirement[:gid]) >= requirement[:level] when "hearth_amenity" @@ -278,6 +331,7 @@ class Character < ApplicationRecord def start_resting return false if self.started_resting_at + stop_activity self.update(started_resting_at: Time.now) end @@ -306,17 +360,26 @@ class Character < ApplicationRecord hearth_amenity_effects = self.hearth.built_hearth_amenities.filter_map { |a| a.effects if a.effects } equipment_effects = self.equipment.filter_map { |a| a.effects if a.effects } state_effects = self.states.filter_map { |a| a.effects if a.effects && !a.expired? } - (hearth_amenity_effects + equipment_effects + state_effects).flatten + item_infix_effects = self.item_infixes.filter_map { |a| a.effects if a.effects } + (hearth_amenity_effects + equipment_effects + state_effects + item_infix_effects).flatten end def total_stat_change(gid) effects.filter_map { |e| e[:modifier] if e[:type] == "stat_change" && e[:gid] == gid }.sum end + def total_enemy_stat_change(gid) + effects.filter_map { |e| e[:modifier] if e[:type] == "enemy_stat_change" && e[:gid] == gid }.sum + end + def damage_ranges effects.filter_map { |e| { gid: e[:gid], min: e[:min], max: e[:max] } if e[:type] == "damage" } end + def dots + effects.filter_map { |e| { gid: e[:gid], min: e[:min], max: e[:max], message: e[:message] } if e[:type] == "dot" } + end + def planting_spots [total_stat_change("planting_spots"), 0].max end diff --git a/app/models/character_skill.rb b/app/models/character_skill.rb index 368cc28..406cade 100644 --- a/app/models/character_skill.rb +++ b/app/models/character_skill.rb @@ -39,6 +39,10 @@ class CharacterSkill < ApplicationRecord end end + def self.xp_required_for_level(level) + level <= 120 ? XP_TOTALS_PER_LEVEL[level - 1] : nil + end + def level XP_TOTALS_PER_LEVEL.each_with_index do |total, index| return index if total > self.xp @@ -46,7 +50,7 @@ class CharacterSkill < ApplicationRecord end def total_xp_for_next_level - xp_required_for_level(level + 1) + CharacterSkill.xp_required_for_level(level + 1) end def xp_to_next_level @@ -57,11 +61,13 @@ class CharacterSkill < ApplicationRecord CharacterSkill.top_xp_for(self.skill, limit: nil).map(&:character).map(&:id).index(self.character.id) + 1 end - private - def xp_required_for_level(level) - level <= 120 ? XP_TOTALS_PER_LEVEL[level - 1] : nil - end + def percentage_of_skill_level_completed + xp_gained_this_level = xp - CharacterSkill.xp_required_for_level(level) + total_xp_gain_neeeded_for_entire_level = total_xp_for_next_level - CharacterSkill.xp_required_for_level(level) + (xp_gained_this_level.to_f / total_xp_gain_neeeded_for_entire_level) * 100 + end + private def send_chat_message_if_leveled_up if CharacterSkill.level_for_xp(self.xp_was) < CharacterSkill.level_for_xp(self.xp) chat_message = ChatMessage.new(body: "reached #{self.skill.name} level #{self.level}!", diff --git a/app/models/concerns/has_costs_and_requirements.rb b/app/models/concerns/has_costs_and_requirements.rb index 34ff0f3..9c4abf8 100644 --- a/app/models/concerns/has_costs_and_requirements.rb +++ b/app/models/concerns/has_costs_and_requirements.rb @@ -19,8 +19,14 @@ module HasCostsAndRequirements case req[:type] when "skill" requirements.push "level #{req[:level]} #{Skill.find_by_gid(req[:gid]).name}" + when "stat" + requirements.push "#{req[:value]} #{req[:gid]}" when "equipment" - requirements.push "equipped #{Item.find_by_gid(req[:gid]).name}" + if req[:tag] + requirements.push "equipped #{req[:tag]}" + else + requirements.push "equipped #{Item.find_by_gid(req[:gid]).name}" + end when "hearth_amenity" requirements.push "level #{req[:level]} #{HearthAmenity.find_by_gid(req[:gid]).name}" else diff --git a/app/models/equipment.rb b/app/models/equipment.rb index e1dc7a3..12e8ae7 100644 --- a/app/models/equipment.rb +++ b/app/models/equipment.rb @@ -5,6 +5,17 @@ class Equipment < ApplicationRecord :left_ring, :right_ring, :waist, :legs, :feet, :curio] validates :slot, presence: true, uniqueness: { scope: :character } + def self.random_break_slot + roll = rand + if roll >= 0.99933333 + [:neck, :left_ring, :right_ring].sample + elsif roll >= 0.999 + [:back, :waist, :curio].sample + elsif roll >= 0.998 + [:mainhand, :offhand, :head, :torso, :grip, :legs, :feet].sample + end + end + def slot self[:slot].to_sym end @@ -14,14 +25,6 @@ class Equipment < ApplicationRecord end def break_check - roll = rand - if [:neck, :left_ring, :right_ring].include?(slot) - destroy and return true if roll > 0.9998 - elsif [:back, :waist, :curio].include?(slot) - destroy and return true if roll > 0.9996 - else - destroy and return true if roll > 0.999 - end false end end diff --git a/app/models/item.rb b/app/models/item.rb index 8b80788..0e25b2f 100644 --- a/app/models/item.rb +++ b/app/models/item.rb @@ -13,6 +13,13 @@ class Item < ApplicationRecord self.whatnot && self.whatnot[:use_effects]&.any? end + def infixable?(skill = nil) + skill = Skill.find_by_gid(skill) if skill.is_a? String + return false unless self.whatnot && self.whatnot[:infix_skills]&.any? + return true unless skill + self.whatnot[:infix_skills].select { |data| data[:gid] == skill.gid }.any? + end + def equip_slots return [] unless self.equipment? self.whatnot[:equip_slots].map { |data| data.to_sym } diff --git a/app/models/item_infix.rb b/app/models/item_infix.rb new file mode 100644 index 0000000..c484242 --- /dev/null +++ b/app/models/item_infix.rb @@ -0,0 +1,21 @@ +class ItemInfix < ApplicationRecord + belongs_to :character + belongs_to :item + belongs_to :skill + + before_create :check_max_infixes + + def self.break_check + rand >= 0.99 + end + + def effects + self.item.whatnot[:infix_effects] + end + + private + def check_max_infixes + current_infixes = character.item_infixes.where(skill: skill) + raise :abort if current_infixes.count >= character.max_infixes(skill) + end +end diff --git a/app/models/monster_kill.rb b/app/models/monster_kill.rb new file mode 100644 index 0000000..24519b1 --- /dev/null +++ b/app/models/monster_kill.rb @@ -0,0 +1,22 @@ +class MonsterKill < ApplicationRecord + belongs_to :monster + belongs_to :character + + validates :quantity, numericality: { greater_than_or_equal_to: 0, only_integer: true } + scope :ordered_by_monster_name, -> { includes(:monster).order("monsters.name") } + + after_save :award_titles + + private + def award_titles + character.award_title("spiteful") if quantity >= 1000 + character.award_title("hateful") if quantity >= 10_000 + character.award_title("vicious") if quantity >= 100_000 + + all_kills_quantity = character.monster_kills.sum(:quantity) + character.award_title("slayer") if all_kills_quantity >= 1_000 + character.award_title("butcher") if all_kills_quantity >= 10_000 + character.award_title("slaughterer") if all_kills_quantity >= 100_000 + character.award_title("massacrer") if all_kills_quantity >= 1_000_000 + end +end diff --git a/app/views/application/_navbar.html.erb b/app/views/application/_navbar.html.erb index be2d9f2..e315b42 100644 --- a/app/views/application/_navbar.html.erb +++ b/app/views/application/_navbar.html.erb @@ -1,21 +1,27 @@ -<ul class="py-2 px-2 col-span-12 text-display"> +<ul class="py-2 px-2 col-span-12 text-display space-x-2.5"> <% if current_char %> - <li class="mr-6 inline"> + <li class="inline"> <%= link_to "Locations", locations_path %> </li> - <li class="mr-6 inline"> + <li class="inline"> <%= link_to "Character", character_path(current_char) %> </li> - <li class="mr-6 inline"> + <li class="inline"> + <%= link_to "Skills", character_skills_path(current_char) %> + </li> + <li class="inline"> <%= link_to "Inventory", character_items_path(current_char) %> </li> - <li class="mr-6 inline"> + <li class="inline"> + <%= link_to "Spells", character_spells_path(current_char) %> + </li> + <li class="inline"> <%= link_to "Hearth", character_hearth_path(current_char) %> </li> - <li class="mr-6 inline"> + <li class="inline"> <%= link_to "Bazaar", bazaar_path %> </li> - <li class="mr-6 inline"> + <li class="inline"> <%= link_to "Messages", messages_path %> </li> <% end %> diff --git a/app/views/application/_results.html.erb b/app/views/application/_results.html.erb index ec62991..a7dc002 100644 --- a/app/views/application/_results.html.erb +++ b/app/views/application/_results.html.erb @@ -13,7 +13,7 @@ <p>You planted <%= link_to result[:hearth_planting].item.name, item_path(result[:hearth_planting].item) %> in the loam.</p> <% when "activity" %> - <p>You realized how to <%= result[:activity].name %>!</p> + <p>You learned how to <%= result[:activity].name %>!</p> <% when "monster" %> <p>You encountered a <%= result[:monster].name %>.</p> <p class="text-xs italic"><%= result[:monster].description %></p> @@ -24,6 +24,8 @@ <p class="text-xs">You gained <%= result[:xp] %> <%= result[:skill].name %> XP.</p> <% when "title" %> <p>You earned the title <%= render "application/components/text/title", title: result[:title] %>!</p> + <% when "condition" %> + <p>You gained the <%= result[:condition].name %> condition.</p> <% when "message" %> <p><%= result[:body] %></p> <% when "warning" %> diff --git a/app/views/application/_timer.html.erb b/app/views/application/_timer.html.erb index d683c6a..51d9b81 100644 --- a/app/views/application/_timer.html.erb +++ b/app/views/application/_timer.html.erb @@ -1,61 +1,63 @@ -<% if current_char.activity %> - <h2 class="text-lg text-display text-center"><%= current_char.activity.name %></h2> - <div data-controller="timer" - data-timer-time-remaining-value="<%= current_char.activity_time_remaining %>" - data-timer-activity-duration-value="<%= current_char.activity_duration - current_char.rested_duration_to_spend_on_activity %>" - data-timer-post-url-value="<%= finish_activity_url %>"> - <div class="text-center"> - <span data-timer-target="timer" class="text-xl text-display"></span> - </div> - <div class="border border-gray-800 h-4 my-1"> - <div data-timer-target="progressBar" class="bg-gray-600 h-full" - style="width: <%= current_char.percentage_of_activity_completed %>%"> +<% if current_char %> + <% if current_char.activity %> + <h2 class="text-lg text-display text-center"><%= current_char.activity.name %></h2> + <div data-controller="timer" + data-timer-time-remaining-value="<%= current_char.activity_time_remaining %>" + data-timer-activity-duration-value="<%= current_char.activity_duration - current_char.rested_duration_to_spend_on_activity %>" + data-timer-post-url-value="<%= finish_activity_url %>"> + <div class="text-center"> + <span data-timer-target="timer" class="text-xl text-display"></span> + </div> + <div class="border border-gray-800 h-4 my-1"> + <div data-timer-target="progressBar" class="bg-gray-600 h-full" + style="width: <%= current_char.percentage_of_activity_completed %>%"> + </div> </div> </div> - </div> - - <% most_recent_cs = current_char.character_skills.order(:updated_at).last %> - <div class="text-center text-sm"> - <div class="text-xs"><%= most_recent_cs.skill.name %> level <%= most_recent_cs.level %></div> - <div><%= most_recent_cs.xp_to_next_level %> XP to next level</div> - </div> - <div class="text-center my-2"> - <%= button_to "Stop", stop_activity_path, class: "text-sm" %> - </div> - - <div class="text-center text-xs my-2"> - <% if current_char.activity.gid.include?("beastslay") %> - <%= current_char.wounds %> / <%= pluralize(current_char.max_wounds, "wound") %> - <% end %> - <% current_char.active_states.each do |state| %> - <div> - <%= state.condition.name %> - </div> - <div> - (expires in <%= distance_of_time_in_words_to_now(state.expires_at) %>) - </div> - <% end %> - </div> + <% most_recent_cs = current_char.character_skills.order(:updated_at).last %> + <div class="text-center text-sm"> + <div class="text-xs"><%= most_recent_cs.skill.name %> level <%= most_recent_cs.level %></div> + <div><%= most_recent_cs.xp_to_next_level %> XP to next level</div> + </div> -<% else %> - <div class="text-center"> - <% if current_char.resting? %> - <p>You're resting.</p> - <% else %> - <p>You're not doing anything.</p> - <% end %> + <div class="text-center my-2"> + <%= button_to "Stop", stop_activity_path, class: "text-sm" %> + </div> - <div class="my-2"> - <%= button_to current_char.resting? ? "Stop Resting" : "Start Resting", toggle_resting_path %> + <div class="text-center text-xs my-2"> + <% if current_char.activity.gid.include?("beastslay") %> + <%= current_char.wounds %> / <%= pluralize(current_char.max_wounds, "wound") %> + <% end %> + <% current_char.active_states.each do |state| %> + <div> + <%= state.condition.name %> + </div> + <div> + (expires in <%= distance_of_time_in_words_to_now(state.expires_at) %>) + </div> + <% end %> </div> - <div class="text-xs"> - You have <%= distance_of_time_in_words_to_now(current_char.rested_until) %> of rested time. + <% else %> + <div class="text-center"> <% if current_char.resting? %> - This does not include time from your current rest. That time will be added when you stop resting. + <p>You're resting.</p> + <% else %> + <p>You're not doing anything.</p> <% end %> + + <div class="my-2"> + <%= button_to current_char.resting? ? "Stop Resting" : "Start Resting", toggle_resting_path %> + </div> + + <div class="text-xs"> + You have <%= distance_of_time_in_words_to_now(current_char.rested_until) %> of rested time. + <% if current_char.resting? %> + This does not include time from your current rest. That time will be added when you stop resting. + <% end %> + </div> </div> - </div> + <% end %> <% end %> diff --git a/app/views/application/components/text/_title.html.erb b/app/views/application/components/text/_title.html.erb index a7c2a7e..5e1d5a2 100644 --- a/app/views/application/components/text/_title.html.erb +++ b/app/views/application/components/text/_title.html.erb @@ -10,6 +10,20 @@ <span class="text-purple-200">Aspirant</span> <% when "sentinel" %> <span class="text-gray-500">S</span><span class="text-gray-400">e</span><span class="text-gray-300">n</span><span class="text-gray-200">ti</span><span class="text-gray-300">n</span><span class="text-gray-400">e</span><span class="text-gray-500">l</span> + <% when "spiteful" %> + <span class="text-transparent bg-clip-text bg-gradient-to-b from-gray-500 to-red-900">Spiteful</span> + <% when "hateful" %> + <span class="text-transparent bg-clip-text bg-gradient-to-b from-gray-600 to-red-800">Hateful</span> + <% when "vicious" %> + <span class="text-transparent bg-clip-text bg-gradient-to-b from-gray-700 to-red-700">Vicious</span> + <% when "slayer" %> + <span class="text-red-400">Slayer</span> + <% when "butcher" %> + <span class="text-red-500">Butcher</span> + <% when "slaughterer" %> + <span class="text-red-600">Slaughterer</span> + <% when "massacrer" %> + <span class="text-red-700">Massacrer</span> <% else %> <%= title.name %> <% end %> diff --git a/app/views/characters/bestiary/index.html.erb b/app/views/characters/bestiary/index.html.erb new file mode 100644 index 0000000..f7376e9 --- /dev/null +++ b/app/views/characters/bestiary/index.html.erb @@ -0,0 +1,22 @@ +<h1 class="text-3xl mb-4">Bestiary</h1> + +<% if @monster_kills.any? %> + <table class="table-auto"> + <thead> + <tr> + <th class="table-header-padded">Monster</th> + <th class="table-header-padded">Kills</th> + </tr> + </thead> + <tbody> + <% @monster_kills.ordered_by_monster_name.each do |mk| %> + <tr> + <td class="table-cell-padded"><%= mk.monster.name %></td> + <td class="table-cell-padded"><%= mk.quantity %></td> + </tr> + <% end %> + </tbody> + </table> +<% else %> + <p>You haven't killed any monsters yet.</p> +<% end %> diff --git a/app/views/characters/items/index.html.erb b/app/views/characters/items/index.html.erb index 671e68f..57e8531 100644 --- a/app/views/characters/items/index.html.erb +++ b/app/views/characters/items/index.html.erb @@ -35,6 +35,10 @@ character_items: @character.character_items.ordered_by_item_name.select { |ci| ci.item.has_tag?("currency") } %> +<%= render "characters/items/inventory_section", heading: "Omens", + character_items: @character.character_items.ordered_by_item_name.select { |ci| + ci.item.has_tag?("omen") } %> + <%= render "characters/items/inventory_section", heading: "Seeds", character_items: @character.character_items.ordered_by_item_name.select { |ci| ci.item.has_tag?("seed") } %> diff --git a/app/views/characters/show.html.erb b/app/views/characters/show.html.erb index a7b7c0e..ce8c461 100644 --- a/app/views/characters/show.html.erb +++ b/app/views/characters/show.html.erb @@ -5,6 +5,7 @@ <div class="text-lg text-display mb-4"> <ul class="flex flex-row"> <li class="mr-2"><%= link_to "Titles", character_titles_path(@character) %></li> + <li class="mr-2"><%= link_to "Bestiary", character_bestiary_path(@character) %></li> <li class="mr-2"><%= link_to "Rankings", character_rankings_path(@character) %></li> </ul> </div> @@ -30,7 +31,7 @@ <div class="my-2"> <% if @character == current_char %> <h2 class="text-xl mb-2">Combat Styles</h2> - <%= form_with url: character_combat_styles_path(character_id: @character) do |f| %> + <%= form_with url: combat_styles_character_path(@character) do |f| %> <%= f.label :offensive_style, "Offensive" %> <%= f.select :offensive_style, Character.offensive_styles.keys.to_a, selected: @character.offensive_style %> @@ -149,31 +150,6 @@ </div> </div> -<div class="my-6"> - <h2 class="text-xl mb-4">Skills</h2> - - <table class="table-auto mb-8"> - <thead> - <tr> - <th class="table-header-padded">Skill</th> - <th class="table-header-padded">Level</th> - <th class="table-header-padded">XPTNL</th> - <th class="table-header-padded">Total XP</th> - </tr> - </thead> - <tbody> - <% @character.character_skills.ordered_by_skill_name.each do |cs| %> - <tr> - <td class="table-cell-padded"><%= cs.skill.name %></td> - <td class="table-cell-padded"><%= cs.level %></td> - <td class="table-cell-padded"><%= cs.xp_to_next_level %></td> - <td class="table-cell-padded"><%= cs.xp %></td> - </tr> - <% end %> - </tbody> - </table> -</div> - <% if @character == current_char %> <%= link_to "Manage account", edit_user_registration_path, class: "text-sm" %> <% end %> diff --git a/app/views/characters/skills/_infix_slot.html.erb b/app/views/characters/skills/_infix_slot.html.erb new file mode 100644 index 0000000..0113596 --- /dev/null +++ b/app/views/characters/skills/_infix_slot.html.erb @@ -0,0 +1,3 @@ +<div class="flex justify-between items-center border-t border-gray-700 p-2"> + <%= yield %> +</div> diff --git a/app/views/characters/skills/index.html.erb b/app/views/characters/skills/index.html.erb new file mode 100644 index 0000000..af40e38 --- /dev/null +++ b/app/views/characters/skills/index.html.erb @@ -0,0 +1,58 @@ +<h2 class="text-3xl mb-2">Skills</h2> +<div class="grid gap-4 grid-cols-1 lg:grid-cols-2 xl:grid-cols-3 2xl:grid-cols-4"> + <% @character.character_skills.ordered_by_skill_name.each do |cs| %> + <div> + <div class="rounded border border-gray-700"> + <div class="flex p-1"> + <div class="flex-grow"> + <div class="text-xl text-display mb-1"> + <%= cs.skill.name %> + </div> + <div class="flex items-center text-xs space-x-4"> + <div><span class="bg-gray-700 px-1 py-0.5 rounded mr-1">XP</span><%= cs.xp %></div> + <div><span class="bg-gray-700 px-1 py-0.5 rounded mr-1">TNL</span><%= cs.xp_to_next_level %></div> + </div> + </div> + <div class="text-xl m-2 text-display"> + <%= cs.level %> + </div> + </div> + <div class="border border-gray-700 h-2 mt-1 -mb-px -mx-px"> + <div class="bg-gray-600 h-full" style="width: <%= cs.percentage_of_skill_level_completed %>%"> + </div> + </div> + <% @character.item_infixes.where(skill: cs.skill).each do |ii| %> + <%= render "characters/skills/infix_slot" do %> + <div> + <%= ii.item.name %> + </div> + <div> + <%= button_to "Remove", character_item_infix_path(id: ii.id), method: :delete %> + </div> + <% end %> + <% end %> + <% @character.available_infixes(cs.skill).times do %> + <%= render "characters/skills/infix_slot" do %> + <%# TODO: Don't load all into memory %> + <% infixable_items = @character.items.select {|i| i.infixable?(cs.skill)} %> + <% if infixable_items.any? %> + <%= form_with url: character_item_infixes_path, class: "w-full" do |f| %> + <div class="flex space-x-1"> + <div class="flex-grow"> + <%= f.select :item_id, infixable_items.map { |i| [i.name, i.id]}, {}, class: "w-full" %> + <%= f.hidden_field :skill_id, value: cs.skill.id %> + </div> + <div> + <%= f.submit "Infix" %> + </div> + </div> + <% end %> + <% else %> + <div class="text-gray-500">No omens to infix.</div> + <% end %> + <% end %> + <% end %> + </div> + </div> + <% end %> +</div> diff --git a/app/views/characters/spells/index.html.erb b/app/views/characters/spells/index.html.erb new file mode 100644 index 0000000..5f415a3 --- /dev/null +++ b/app/views/characters/spells/index.html.erb @@ -0,0 +1,10 @@ +<h2 class="text-3xl mb-2">Spells</h2> +<div data-controller="activity-select"> + <%= form_with url: start_activity_path, method: :post do |f| %> + <%= f.select :id, @spell_activities.map { |a| [a.name, a.id] }, {}, + { data: { activity_select_target: "select", action: "activity-select#load" } } %> + <%= f.number_field :actions, value: 1, size: 5, min: 1, max: 2_000_000_000 %> + <%= f.submit "Cast" %> + <% end %> + <div data-activity-select-target="output" class="my-1"></div> +</div> diff --git a/app/views/game/finish_activity.js.erb b/app/views/game/finish_activity.js.erb index fa3d228..f281986 100644 --- a/app/views/game/finish_activity.js.erb +++ b/app/views/game/finish_activity.js.erb @@ -4,10 +4,16 @@ var resultControlsDiv = document.getElementById("activity_controls"); var outputHTML = "<%= j render(partial: "application/results", locals: { results: @results }) %>" if (resultOutputDiv) { - resultOutputDiv.innerHTML += outputHTML; - // If scrolled to the bottom or near the bottom, then smooth scroll to the bottom. + // (Check before adding output, because a large output could prevent the scroll when it shouldn't.) + var shouldScroll = false; if ((resultOutputDiv.scrollTop + 100) >= (resultOutputDiv.scrollHeight - resultOutputDiv.offsetHeight)) { + shouldScroll = true; + } + + resultOutputDiv.innerHTML += outputHTML; + + if (shouldScroll) { resultOutputDiv.scrollTo({ top: resultOutputDiv.scrollHeight, left: 0, behavior: 'smooth' }); diff --git a/app/views/layouts/application.html.erb b/app/views/layouts/application.html.erb index 309dc29..048a6ed 100644 --- a/app/views/layouts/application.html.erb +++ b/app/views/layouts/application.html.erb @@ -40,6 +40,8 @@ </div> <div class="game-container-box overflow-auto text-sm p-2 flex-grow" id="result_output" style="max-height: 60%" + data-controller="results" + data-results-target="output" data-turbolinks-permanent> </div> </div> diff --git a/config/routes.rb b/config/routes.rb index fa6d748..6e435f2 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -24,14 +24,18 @@ Rails.application.routes.draw do end resources :characters, only: [:show, :new, :create] do - post "/combat_styles", to: "characters#set_combat_styles" + post "/combat_styles", to: "characters#set_combat_styles", on: :member scope module: :characters do get "/rankings", to: "rankings#index" + get :bestiary, to: "bestiary#index" post "/items/unequip/:slot", to: "items#unequip", as: :item_unequip + resources :item_infixes, only: [:create, :destroy] resources :items, only: [:index] do post "/equip", to: "items#equip" post "/use", to: "items#use" end + resources :skills, only: [:index] + resources :spells, only: [:index] resources :titles, only: [:index] do post "/activate", to: "titles#activate" end diff --git a/data/activities.yml b/data/activities.yml index d48c9b0..a0a0072 100644 --- a/data/activities.yml +++ b/data/activities.yml @@ -61,7 +61,7 @@ construct_forge_level2: gid: "forge" level: 2 construct_laboratory_level1: - name: "Construct laboratory Level 1" + name: "Construct Laboratory Level 1" description: "Build a level 1 laboratory." whatnot: requirements: @@ -82,7 +82,7 @@ construct_laboratory_level1: gid: "laboratory" level: 1 construct_laboratory_level2: - name: "Upgrade laboratory to Level 2" + name: "Upgrade Laboratory to Level 2" description: "Upgrade your laboratory to level 2." whatnot: requirements: @@ -251,6 +251,69 @@ construct_spicebench_level2: - type: "hearth_amenity" gid: "spicebench" level: 2 +construct_binding_array_level1: + name: "Construct Binding Array Level 1" + description: "Build a level 1 binding array." + whatnot: + requirements: + - type: "hearth_amenity" + gid: "foundation" + level: 1 + duration: + base: 1000 + cost: + - type: "item" + gid: "stone" + quantity: 200 + - type: "item" + gid: "wood" + quantity: 400 + - type: "item" + gid: "iron_ingot" + quantity: 4 + - type: "item" + gid: "red_beryl" + quantity: 2 + - type: "item" + gid: "tourmaline" + quantity: 2 + results: + - type: "hearth_amenity" + gid: "binding_array" + level: 1 +construct_binding_array_level2: + name: "Construct Binding Array Level 2" + description: "Upgrade your binding array to level 2." + whatnot: + requirements: + - type: "hearth_amenity" + gid: "binding_array" + level: 1 + duration: + base: 3600 + cost: + - type: "item" + gid: "stone" + quantity: 200 + - type: "item" + gid: "wood" + quantity: 300 + - type: "item" + gid: "red_beryl" + quantity: 2 + - type: "item" + gid: "tourmaline" + quantity: 2 + - type: "item" + gid: "paraiba_tourmaline" + quantity: 1 + - type: "item" + gid: "yellow_beryl" + quantity: 1 + results: + - type: "hearth_amenity" + gid: "binding_array" + level: 2 plant_mudtub_seed: name: "Plant Mudtub Seed" description: "Plant a mudtub seed." @@ -259,12 +322,17 @@ plant_mudtub_seed: - type: "hearth_amenity" gid: "loamspire" level: 1 + - type: "skill" + gid: "magiculture" + level: 1 + - type: "equipment" + tag: "spade" duration: - base: 120 + base: 60 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -282,12 +350,17 @@ harvest_mudtub: - type: "hearth_amenity" gid: "loamspire" level: 1 + - type: "skill" + gid: "magiculture" + level: 1 + - type: "equipment" + tag: "spade" duration: - base: 120 + base: 60 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -299,7 +372,7 @@ harvest_mudtub: gid: "mudtub" xp: - gid: "magiculture" - value: 10 + value: 5 plant_midoras_seed: name: "Plant Midoras Seed" description: "Plant a midoras seed." @@ -311,12 +384,14 @@ plant_midoras_seed: - type: "skill" gid: "magiculture" level: 2 + - type: "equipment" + tag: "spade" duration: - base: 130 + base: 70 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -337,12 +412,14 @@ harvest_midoras: - type: "skill" gid: "magiculture" level: 2 + - type: "equipment" + tag: "spade" duration: - base: 130 + base: 70 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -354,7 +431,7 @@ harvest_midoras: gid: "midoras" xp: - gid: "magiculture" - value: 15 + value: 7 plant_templis_seed: name: "Plant Templis Seed" description: "Plant a templis seed." @@ -366,12 +443,14 @@ plant_templis_seed: - type: "skill" gid: "magiculture" level: 4 + - type: "equipment" + tag: "spade" duration: - base: 140 + base: 80 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -392,12 +471,14 @@ harvest_templis: - type: "skill" gid: "magiculture" level: 4 + - type: "equipment" + tag: "spade" duration: - base: 140 + base: 80 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -409,7 +490,7 @@ harvest_templis: gid: "templis" xp: - gid: "magiculture" - value: 30 + value: 9 plant_enzon_seed: name: "Plant Enzon Seed" description: "Plant an enzon seed." @@ -421,12 +502,14 @@ plant_enzon_seed: - type: "skill" gid: "magiculture" level: 7 + - type: "equipment" + tag: "spade" duration: - base: 150 + base: 90 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -447,12 +530,14 @@ harvest_enzon: - type: "skill" gid: "magiculture" level: 7 + - type: "equipment" + tag: "spade" duration: - base: 150 + base: 90 scaling: - type: "skill" gid: "magiculture" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "magiculture_speed" scale_value: 1 @@ -464,7 +549,7 @@ harvest_enzon: gid: "enzon" xp: - gid: "magiculture" - value: 45 + value: 11 craft_crude_iron_ingot: name: "Smelt Crude Iron Ingot" description: "Smelt a crude iron ingot." @@ -473,26 +558,29 @@ craft_crude_iron_ingot: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 50 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 cost: - type: "item" gid: "crude_iron_ore" - quantity: 10 + quantity: 8 results: - type: "item" gid: "crude_iron_ingot" xp: - gid: "otherforge" - value: 4 + value: 6 craft_iron_ingot: name: "Smelt Iron Ingot" description: "Smelt an iron ingot." @@ -501,20 +589,23 @@ craft_iron_ingot: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 60 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 cost: - type: "item" gid: "iron_ore" - quantity: 10 + quantity: 8 results: - type: "item" gid: "iron_ingot" @@ -538,20 +629,20 @@ craft_pure_iron_ingot: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 cost: - type: "item" gid: "pure_iron_ore" - quantity: 10 + quantity: 8 results: - type: "item" gid: "pure_iron_ingot" xp: - gid: "otherforge" - value: 12 + value: 11 craft_arcanite_ingot: name: "Smelt Arcanite Ingot" description: "Smelt an arcanite ingot." @@ -569,7 +660,7 @@ craft_arcanite_ingot: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -585,7 +676,7 @@ craft_arcanite_ingot: gid: "arcanite_ingot" xp: - gid: "otherforge" - value: 16 + value: 14 craft_purify_crude_iron_ingot: name: "Purify Crude Iron Ingot" description: "Purify crude iron ingots into an iron ingot." @@ -595,19 +686,19 @@ craft_purify_crude_iron_ingot: gid: "forge" level: 1 duration: - base: 105 + base: 60 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 cost: - type: "item" gid: "crude_iron_ingot" - quantity: 5 + quantity: 4 results: - type: "item" gid: "iron_ingot" @@ -626,25 +717,25 @@ craft_purify_iron_ingot: gid: "otherforge" level: 5 duration: - base: 115 + base: 70 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 cost: - type: "item" gid: "iron_ingot" - quantity: 5 + quantity: 4 results: - type: "item" gid: "pure_iron_ingot" xp: - gid: "otherforge" - value: 12 + value: 11 craft_iron_lockpicks: name: "Craft Iron Lockpicks" description: "Craft iron lockpicks." @@ -653,13 +744,16 @@ craft_iron_lockpicks: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 50 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -681,13 +775,16 @@ craft_iron_dagger: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 50 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -718,7 +815,7 @@ craft_iron_short_sword: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -749,7 +846,7 @@ craft_iron_longsword: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -771,13 +868,16 @@ craft_iron_buckler: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 100 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -808,7 +908,7 @@ craft_iron_shield: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -839,7 +939,7 @@ craft_iron_cap: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -870,7 +970,7 @@ craft_iron_sallet: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -901,7 +1001,7 @@ craft_iron_leggings: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -932,7 +1032,7 @@ craft_iron_chainmail: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -963,7 +1063,7 @@ craft_iron_platemail: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -994,7 +1094,7 @@ craft_arcanite_dagger: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1025,7 +1125,7 @@ craft_arcanite_short_sword: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1049,14 +1149,14 @@ craft_arcanite_longsword: level: 2 - type: "skill" gid: "otherforge" - level: 15 + level: 17 duration: base: 215 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1087,7 +1187,7 @@ craft_arcanite_buckler: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1118,7 +1218,7 @@ craft_arcanite_shield: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1149,7 +1249,7 @@ craft_arcanite_cap: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1180,7 +1280,7 @@ craft_arcanite_sallet: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1211,7 +1311,7 @@ craft_arcanite_leggings: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1242,7 +1342,7 @@ craft_arcanite_chainmail: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1273,7 +1373,7 @@ craft_arcanite_platemail: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1304,7 +1404,7 @@ craft_balgoloth_plate: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1329,13 +1429,16 @@ craft_iron_pickaxe: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 75 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1360,13 +1463,16 @@ craft_iron_axe: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 75 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1391,13 +1497,16 @@ craft_iron_spade: - type: "hearth_amenity" gid: "forge" level: 1 + - type: "skill" + gid: "otherforge" + level: 1 duration: base: 75 minimum: 35 scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1431,7 +1540,7 @@ craft_arcanite_pickaxe: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1465,7 +1574,7 @@ craft_arcanite_axe: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1499,7 +1608,7 @@ craft_arcanite_spade: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1533,7 +1642,7 @@ craft_onus_of_vision: scaling: - type: "skill" gid: "otherforge" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "otherforge_speed" scale_value: 1 @@ -1567,13 +1676,16 @@ craft_mending_salve: - type: "hearth_amenity" gid: "laboratory" level: 1 + - type: "skill" + gid: "fluxseethe" + level: 1 duration: base: 90 minimum: 35 scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1607,7 +1719,7 @@ craft_manadross_tincture: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1644,7 +1756,7 @@ craft_lusterlight_tincture: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1681,7 +1793,7 @@ craft_quarrying_draught: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1715,7 +1827,7 @@ craft_reaping_draught: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1749,7 +1861,7 @@ craft_trawling_draught: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1783,7 +1895,7 @@ craft_woodflesh_potion: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1817,7 +1929,7 @@ craft_lightblood_potion: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1854,7 +1966,7 @@ craft_mercuria_potion: scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -1879,13 +1991,19 @@ planequarry_floret_mines: description: "Planequarry at the Floret Mines." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "planequarry" + level: 1 + - type: "equipment" + tag: "pickaxe" duration: - base: 130 + base: 70 minimum: 35 scaling: - type: "skill" gid: "planequarry" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "planequarry_speed" scale_value: 1 @@ -1907,12 +2025,12 @@ planequarry_floret_mines: score: 0.95 xp: - gid: "planequarry" - value: 10 + value: 8 - gid: "pure_iron_ore" score: 0.996 xp: - gid: "planequarry" - value: 25 + value: 10 - type: "item" chance: 0.02 table: @@ -1930,14 +2048,14 @@ planequarry_floret_mines: score: 0.90 xp: - gid: "planequarry" - value: 25 + value: 20 titles: - gid: "beryly" - gid: "paraiba_tourmaline" score: 0.95 xp: - gid: "planequarry" - value: 25 + value: 20 planequarry_deepshaft: name: "Quarry Deepshaft" description: "Descend far below the the Floret Mines into the labyrinth of shafts left behind by the ancients." @@ -1947,13 +2065,15 @@ planequarry_deepshaft: - type: "skill" gid: "planequarry" level: 10 + - type: "equipment" + tag: "pickaxe" duration: - base: 160 + base: 80 minimum: 35 scaling: - type: "skill" gid: "planequarry" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "planequarry_speed" scale_value: 1 @@ -1975,17 +2095,17 @@ planequarry_deepshaft: score: 0.20 xp: - gid: "planequarry" - value: 10 + value: 8 - gid: "pure_iron_ore" score: 0.90 xp: - gid: "planequarry" - value: 25 + value: 10 - gid: "gaian_ore" score: 0.995 xp: - gid: "planequarry" - value: 50 + value: 12 - type: "item" chance: 0.03 table: @@ -2003,26 +2123,32 @@ planequarry_deepshaft: score: 0.90 xp: - gid: "planequarry" - value: 25 + value: 20 titles: - gid: "beryly" - gid: "paraiba_tourmaline" score: 0.95 xp: - gid: "planequarry" - value: 25 + value: 20 planequarry_brine_trench: name: "Quarry Brine Trench" description: "Planequarry in the south Floret brine trench." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "planequarry" + level: 1 + - type: "equipment" + tag: "pickaxe" duration: - base: 60 + base: 50 minimum: 25 scaling: - type: "skill" gid: "planequarry" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "planequarry_speed" scale_value: 1 @@ -2039,19 +2165,23 @@ planequarry_brine_trench: gid: "seas_tear" xp: - gid: "planequarry" - value: 22 + value: 15 beastslay_killing_fields: name: "Slay in the Killing Fields" description: "Hunt monsters in the Killing Fields." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "beastslay" + level: 1 duration: base: 60 minimum: 35 scaling: - type: "skill" gid: "beastslay" - scale_value: 1 + scale_value: 0.5 results: - type: "monster" chance: 1 @@ -2069,6 +2199,10 @@ beastslay_hopegraves: description: "Hunt monsters in the Hopegraves." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "beastslay" + level: 1 duration: base: 70 minimum: 35 @@ -2091,13 +2225,19 @@ wealdreap_twil_woods: description: "Wealdreap within Twil Woods." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "wealdreap" + level: 1 + - type: "equipment" + tag: "axe" duration: - base: 120 + base: 60 minimum: 35 scaling: - type: "skill" gid: "wealdreap" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "wealdreap_speed" scale_value: 1 @@ -2119,17 +2259,17 @@ wealdreap_twil_woods: score: 0.59 xp: - gid: "wealdreap" - value: 9 + value: 8 - gid: "shrine_hassock" score: 0.98 xp: - gid: "wealdreap" - value: 30 + value: 12 - gid: "discord_pome" score: 0.996 xp: - gid: "wealdreap" - value: 40 + value: 15 - type: "item" chance: 0.02 table: @@ -2150,7 +2290,7 @@ wealdreap_twil_woods: max_quantity: 3 xp: - gid: "wealdreap" - value: 15 + value: 10 wealdreap_twil_grove: name: "Reap Twil Grove" description: "Wealdreap within the hidden woodways of the Twil Woods Grove." @@ -2160,13 +2300,15 @@ wealdreap_twil_grove: - type: "skill" gid: "wealdreap" level: 10 + - type: "equipment" + tag: "axe" duration: - base: 150 + base: 90 minimum: 35 scaling: - type: "skill" gid: "wealdreap" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "wealdreap_speed" scale_value: 1 @@ -2178,22 +2320,22 @@ wealdreap_twil_grove: score: 0.10 xp: - gid: "wealdreap" - value: 9 + value: 8 - gid: "last_breath" score: 0.65 xp: - gid: "wealdreap" - value: 15 + value: 10 - gid: "silver_iris" score: 0.98 xp: - gid: "wealdreap" - value: 30 + value: 17 - gid: "claritas_flower" score: 0.996 xp: - gid: "wealdreap" - value: 45 + value: 19 - type: "item" chance: 0.02 table: @@ -2214,31 +2356,34 @@ wealdreap_twil_grove: max_quantity: 3 xp: - gid: "wealdreap" - value: 15 + value: 10 - gid: "enzon_seed" score: 0.90 max_quantity: 3 xp: - gid: "wealdreap" - value: 22 + value: 12 manatrawl_sor_well: name: "Trawl Sor Well" description: "Manatrawl within Sor Well." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "manatrawl" + level: 1 + - type: "equipment" + tag: "aethermesh" duration: base: 60 minimum: 35 scaling: - type: "skill" gid: "manatrawl" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "manatrawl_speed" scale_value: 1 - requirements: - - type: "equipment" - gid: "aethermesh" results: - type: "item" chance: 1 @@ -2256,33 +2401,33 @@ manatrawl_sor_well: score: 0.934 xp: - gid: "manatrawl" - value: 20 + value: 15 - gid: "wisp_of_the_current" score: 0.996 xp: - gid: "manatrawl" - value: 35 + value: 25 manatrawl_sor_well_depths: name: "Trawl Sor Well Depths" description: "Manatrawl deep within Sor Well." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "manatrawl" + level: 7 + - type: "equipment" + tag: "aethermesh" duration: base: 70 minimum: 35 scaling: - type: "skill" gid: "manatrawl" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "manatrawl_speed" scale_value: 1 - requirements: - - type: "skill" - gid: "manatrawl" - level: 7 - - type: "equipment" - gid: "aethermesh" results: - type: "item" chance: 1 @@ -2301,26 +2446,29 @@ manatrawl_sor_well_depths: max_quantity: 2 xp: - gid: "manatrawl" - value: 20 + value: 15 - gid: "wisp_of_the_current" score: 0.995 xp: - gid: "manatrawl" - value: 35 + value: 25 synthsever_rusted_lockbox: name: "Unlock Rusted Lockbox" description: "Unseal the lock on a rusted lockbox." whatnot: + requirements: + - type: "equipment" + tag: "lockpicks" + - type: "skill" + gid: "synthsever" + level: 1 duration: base: 180 minimum: 35 scaling: - type: "skill" gid: "synthsever" - scale_value: 1 - requirements: - - type: "equipment" - gid: "iron_lockpicks" + scale_value: 0.5 cost: - type: "item" gid: "rusted_lockbox" @@ -2329,7 +2477,7 @@ synthsever_rusted_lockbox: gid: "unlocked_rusted_lockbox" - type: "xp" gid: "synthsever" - base: 25 + base: 20 open_unlocked_rusted_lockbox: name: "Open Unlocked Rusted Lockbox" description: "Open an unlocked rusted lockbox." @@ -2370,6 +2518,10 @@ wildscour_crumbling_ruins: description: "Wildscour within the crumbling ruins." location: "floret_region" whatnot: + requirements: + - type: "skill" + gid: "wildscour" + level: 1 duration: base: 60 minimum: 35 @@ -2407,11 +2559,16 @@ wildscour_crumbling_ruins: xp: - gid: "wildscour" value: 4 + - gid: "simple_spellpage" + score: 0.99 + xp: + - gid: "wildscour" + value: 15 - gid: "disturbing_doodad" score: 0.998 xp: - gid: "wildscour" - value: 93 + value: 30 craft_gem_dust_from_red_beryl: name: "Crush red beryl" description: "Crush a red beryl into gem dust." @@ -2420,13 +2577,16 @@ craft_gem_dust_from_red_beryl: - type: "hearth_amenity" gid: "laboratory" level: 1 + - type: "skill" + gid: "fluxseethe" + level: 1 duration: base: 60 minimum: 15 scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -2448,13 +2608,16 @@ craft_gem_dust_from_tourmaline: - type: "hearth_amenity" gid: "laboratory" level: 1 + - type: "skill" + gid: "fluxseethe" + level: 1 duration: base: 60 minimum: 15 scaling: - type: "skill" gid: "fluxseethe" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "fluxseethe_speed" scale_value: 1 @@ -2485,7 +2648,7 @@ craft_bluster_powder: scaling: - type: "skill" gid: "spicework" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "spicework_speed" scale_value: 1 @@ -2500,8 +2663,8 @@ craft_bluster_powder: - type: "item" gid: "bluster_powder" xp: - - gid: "fluxseethe" - value: 17 + - gid: "spicework" + value: 15 craft_mudtub_mash: name: "Craft mudtub mash" description: "Cook up some mudtub mash." @@ -2510,13 +2673,16 @@ craft_mudtub_mash: - type: "hearth_amenity" gid: "spicebench" level: 1 + - type: "skill" + gid: "spicework" + level: 1 duration: base: 60 minimum: 35 scaling: - type: "skill" gid: "spicework" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "spicework_speed" scale_value: 1 @@ -2547,7 +2713,7 @@ craft_midoras_spice: scaling: - type: "skill" gid: "spicework" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "spicework_speed" scale_value: 1 @@ -2560,7 +2726,7 @@ craft_midoras_spice: gid: "midoras_spice" xp: - gid: "spicework" - value: 12 + value: 10 craft_midoras_mudtub_mash: name: "Craft midoras mudtub mash" description: "Cook up some midoras mudtub mash." @@ -2578,7 +2744,7 @@ craft_midoras_mudtub_mash: scaling: - type: "skill" gid: "spicework" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "spicework_speed" scale_value: 1 @@ -2612,7 +2778,7 @@ craft_dusted_templis: scaling: - type: "skill" gid: "spicework" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "spicework_speed" scale_value: 1 @@ -2628,18 +2794,22 @@ craft_dusted_templis: gid: "dusted_templis" xp: - gid: "spicework" - value: 16 + value: 14 beastslay_leviathan_floret_region: name: "Hunt a Leviathan in the Floret Region" description: "You are hunting down a leviathan ravaging the Floret Region." whatnot: + requirements: + - type: "skill" + gid: "beastslay" + level: 1 duration: base: 80 minimum: 40 scaling: - type: "skill" gid: "beastslay" - scale_value: 1 + scale_value: 0.5 - type: "stat" gid: "beastslay_speed" scale_value: 1 @@ -2647,3 +2817,387 @@ beastslay_leviathan_floret_region: - type: "monster_spawn" location: "floret_region" chance: 1 +craft_faint_mana: + name: "Bind faint mana" + description: "Bind an omen of faint mana." + whatnot: + requirements: + - type: "hearth_amenity" + gid: "binding_array" + level: 1 + - type: "skill" + gid: "omenbind" + level: 1 + duration: + base: 60 + minimum: 35 + scaling: + - type: "skill" + gid: "omenbind" + scale_value: 0.5 + - type: "stat" + gid: "omenbind_speed" + scale_value: 1 + cost: + - type: "item" + gid: "vestige" + quantity: 5 + - type: "item" + gid: "arcane_dust" + quantity: 10 + results: + - type: "item" + gid: "faint_mana" + xp: + - gid: "omenbind" + value: 8 +craft_faint_forging_haste: + name: "Bind faint forging haste" + description: "Bind an omen of faint forging haste." + whatnot: + requirements: + - type: "hearth_amenity" + gid: "binding_array" + level: 1 + - type: "skill" + gid: "omenbind" + level: 5 + duration: + base: 75 + minimum: 35 + scaling: + - type: "skill" + gid: "omenbind" + scale_value: 0.5 + - type: "stat" + gid: "omenbind_speed" + scale_value: 1 + cost: + - type: "item" + gid: "vestige" + quantity: 10 + - type: "item" + gid: "iron_ingot" + quantity: 1 + results: + - type: "item" + gid: "faint_forging_haste" + xp: + - gid: "omenbind" + value: 12 +craft_faint_seething_haste: + name: "Bind faint seething haste" + description: "Bind an omen of faint seething haste." + whatnot: + requirements: + - type: "hearth_amenity" + gid: "binding_array" + level: 1 + - type: "skill" + gid: "omenbind" + level: 5 + duration: + base: 75 + minimum: 35 + scaling: + - type: "skill" + gid: "omenbind" + scale_value: 0.5 + - type: "stat" + gid: "omenbind_speed" + scale_value: 1 + cost: + - type: "item" + gid: "vestige" + quantity: 10 + - type: "item" + gid: "burstshroom" + quantity: 10 + results: + - type: "item" + gid: "faint_seething_haste" + xp: + - gid: "omenbind" + value: 12 +craft_faint_working_haste: + name: "Bind faint working haste" + description: "Bind an omen of faint working haste." + whatnot: + requirements: + - type: "hearth_amenity" + gid: "binding_array" + level: 1 + - type: "skill" + gid: "omenbind" + level: 5 + duration: + base: 75 + minimum: 35 + scaling: + - type: "skill" + gid: "omenbind" + scale_value: 0.5 + - type: "stat" + gid: "omenbind_speed" + scale_value: 1 + cost: + - type: "item" + gid: "vestige" + quantity: 10 + - type: "item" + gid: "salt" + quantity: 20 + results: + - type: "item" + gid: "faint_working_haste" + xp: + - gid: "omenbind" + value: 12 +craft_minor_mana: + name: "Bind minor mana" + description: "Bind an omen of minor mana." + whatnot: + requirements: + - type: "hearth_amenity" + gid: "binding_array" + level: 2 + - type: "skill" + gid: "omenbind" + level: 10 + duration: + base: 90 + minimum: 35 + scaling: + - type: "skill" + gid: "omenbind" + scale_value: 0.5 + - type: "stat" + gid: "omenbind_speed" + scale_value: 1 + cost: + - type: "item" + gid: "faint_mana" + quantity: 2 + - type: "item" + gid: "vestige" + quantity: 10 + - type: "item" + gid: "shimmering_essence" + quantity: 1 + results: + - type: "item" + gid: "minor_mana" + xp: + - gid: "omenbind" + value: 16 +havencast_light: + name: "Cast Light" + description: "Risk a little light." + whatnot: + tags: + - "spell" + - "cantrip" + requirements: + - type: "skill" + gid: "havencast" + level: 1 + - type: "stat" + gid: "mana" + value: 1 + duration: + base: 62 + minimum: 35 + scaling: + - type: "skill" + gid: "havencast" + scale_value: 0.5 + - type: "stat" + gid: "havencast_speed" + scale_value: 1 + results: + - type: "condition" + gid: "light" + duration: 3600 # 1 Hour + message: "A ball of light glows before you." + - type: "xp" + gid: "havencast" + base: 3 +havencast_dazzle: + name: "Cast Dazzle" + description: "Cast the Dazzle cantrip." + whatnot: + tags: + - "spell" + - "cantrip" + duration: + base: 62 + minimum: 35 + scaling: + - type: "skill" + gid: "havencast" + scale_value: 0.5 + - type: "stat" + gid: "havencast_speed" + scale_value: 1 + requirements: + - type: "skill" + gid: "havencast" + level: 2 + - type: "stat" + gid: "mana" + value: 1 + results: + - type: "condition" + gid: "dazzle" + duration: 600 # 10 minutes + message: "Sparkling lights distract your enemies." + - type: "xp" + gid: "havencast" + base: 5 +havencast_decipher_simple_magicscript: + name: "Cast Decipher Simple Magicscript" + description: "Cast the Decipher Simple Magicscript cantrip." + whatnot: + tags: + - "spell" + - "cantrip" + duration: + base: 64 + minimum: 35 + scaling: + - type: "skill" + gid: "havencast" + scale_value: 0.5 + - type: "stat" + gid: "havencast_speed" + scale_value: 1 + requirements: + - type: "skill" + gid: "havencast" + level: 3 + - type: "stat" + gid: "mana" + value: 1 + cost: + - type: "item" + gid: "simple_spellpage" + quantity: 1 + results: + - type: "xp" + gid: "havencast" + base: 10 + - type: "activity" + chance: 1 + table: + - gid: "havencast_stinging_rays" + score: 0 + - gid: "havencast_flame_whirl" + score: .50 +havencast_enchant_apprentice_wand: + name: "Cast Enchant Apprentice Wand" + description: "Cast the Enchant Apprentice Wand spell." + whatnot: + tags: + - "spell" + - "cantrip" + duration: + base: 70 + minimum: 35 + scaling: + - type: "skill" + gid: "havencast" + scale_value: 0.5 + - type: "stat" + gid: "havencast_speed" + scale_value: 1 + requirements: + - type: "skill" + gid: "havencast" + level: 5 + - type: "stat" + gid: "mana" + value: 1 + cost: + - type: "item" + gid: "wood" + quantity: 1 + - type: "item" + gid: "arcane_dust" + quantity: 10 + - type: "item" + gid: "shimmering_essence" + quantity: 1 + results: + - type: "item" + gid: "apprentice_wand" + - type: "xp" + gid: "havencast" + base: 15 +havencast_stinging_rays: + name: "Cast Stinging Rays" + description: "Cast the Stinging Rays spell." + innate: false + whatnot: + tags: + - "spell" + duration: + base: 70 + minimum: 35 + scaling: + - type: "skill" + gid: "havencast" + scale_value: 0.5 + - type: "stat" + gid: "havencast_speed" + scale_value: 1 + requirements: + - type: "skill" + gid: "havencast" + level: 5 + - type: "stat" + gid: "mana" + value: 1 + - type: "equipment" + tag: "focus" + results: + - type: "condition" + gid: "stinging_rays" + duration: 600 # 10 minutes + message: "Beams of arcane energy appear in the air around you." + - type: "xp" + gid: "havencast" + base: 11 +havencast_flame_whirl: + name: "Cast Flame Whirl" + description: "Cast the Flame Whirl spell." + innate: false + whatnot: + tags: + - "spell" + duration: + base: 80 + minimum: 35 + scaling: + - type: "skill" + gid: "havencast" + scale_value: 0.5 + - type: "stat" + gid: "havencast_speed" + scale_value: 1 + requirements: + - type: "skill" + gid: "havencast" + level: 10 + - type: "stat" + gid: "mana" + value: 2 + - type: "equipment" + tag: "focus" + results: + - type: "condition" + gid: "flame_whirl" + duration: 600 # 10 minutes + message: "A thin disc of fire encircles you." + - type: "xp" + gid: "havencast" + base: 13 diff --git a/data/conditions.yml b/data/conditions.yml index b691254..4cb71b0 100644 --- a/data/conditions.yml +++ b/data/conditions.yml @@ -96,3 +96,41 @@ mercuria: - type: "stat_change" gid: "evasion" modifier: 1 +light: + name: "light" + description: "Allows seeing in dark areas." + whatnot: + effects: + - type: "special" + gid: "darkvision" + - type: "stat_change" + gid: "necrotic_resistance" + modifier: 1 +dazzle: + name: "dazzle" + description: "Reduces enemies' accuracy." + whatnot: + effects: + - type: "enemy_stat_change" + gid: "accuracy" + modifier: -1 +stinging_rays: + name: "stinging rays" + description: "Does arcane damage over time." + whatnot: + effects: + - type: "dot" + gid: "arcane" + min: 1 + max: 3 + message: "Stinging rays spring from the air around you and hit your enemy." +flame_whirl: + name: "flame whirl" + description: "Does fire damage over time." + whatnot: + effects: + - type: "dot" + gid: "fire" + min: 1 + max: 8 + message: "A ring of flame spins about you, licking at your enemy." diff --git a/data/hearth_amenities.yml b/data/hearth_amenities.yml index 16cb572..e58aa1c 100644 --- a/data/hearth_amenities.yml +++ b/data/hearth_amenities.yml @@ -20,7 +20,7 @@ forge: gid: "otherforge_speed" modifier: 4 laboratory: - name: "laboratory" + name: "Laboratory" description: "A table and some containers, sufficient for practicing the fluxseethe skill." whatnot: construct_activities: @@ -85,3 +85,14 @@ spicebench: level: 2 gid: "spicework_speed" modifier: 4 +binding_array: + name: "Binding Array" + description: >- + A apparatus for binding magical energy into a physical form, creating an omen. + whatnot: + construct_activities: + - level: 1 + gid: "construct_binding_array_level1" + - level: 2 + gid: "construct_binding_array_level2" + diff --git a/data/items.yml b/data/items.yml index e52ae39..59ce7ce 100644 --- a/data/items.yml +++ b/data/items.yml @@ -720,6 +720,11 @@ iron_lockpicks: whatnot: tags: - "tool" + - "lockpicks" + equip_effects: + - type: "stat_change" + gid: "synthsever_speed" + modifier: 10 equip_slots: - "mainhand" stone_pickaxe: @@ -728,12 +733,13 @@ stone_pickaxe: whatnot: tags: - "tool" + - "pickaxe" equip_slots: - "mainhand" equip_effects: - type: "stat_change" gid: "planequarry_speed" - modifier: 60 + modifier: 2 - type: "stat_change" gid: "accuracy" modifier: 0 @@ -746,12 +752,13 @@ stone_axe: whatnot: tags: - "tool" + - "axe" equip_slots: - "mainhand" equip_effects: - type: "stat_change" gid: "wealdreap_speed" - modifier: 60 + modifier: 2 - type: "stat_change" gid: "accuracy" modifier: 0 @@ -764,12 +771,13 @@ stone_spade: whatnot: tags: - "tool" + - "spade" equip_slots: - "mainhand" equip_effects: - type: "stat_change" gid: "magiculture_speed" - modifier: 60 + modifier: 2 - type: "stat_change" gid: "accuracy" modifier: 0 @@ -782,12 +790,13 @@ iron_pickaxe: whatnot: tags: - "tool" + - "pickaxe" equip_slots: - "mainhand" equip_effects: - type: "stat_change" gid: "planequarry_speed" - modifier: 80 + modifier: 10 - type: "stat_change" gid: "accuracy" modifier: 1 @@ -800,12 +809,13 @@ iron_axe: whatnot: tags: - "tool" + - "axe" equip_slots: - "mainhand" equip_effects: - type: "stat_change" gid: "wealdreap_speed" - modifier: 80 + modifier: 10 - type: "stat_change" gid: "accuracy" modifier: 1 @@ -818,12 +828,13 @@ iron_spade: whatnot: tags: - "tool" + - "spade" equip_slots: - "mainhand" equip_effects: - type: "stat_change" gid: "magiculture_speed" - modifier: 80 + modifier: 10 - type: "stat_change" gid: "accuracy" modifier: 1 @@ -836,6 +847,7 @@ arcanite_pickaxe: whatnot: tags: - "tool" + - "pickaxe" equip_slots: - "mainhand" equip_requirements: @@ -845,7 +857,7 @@ arcanite_pickaxe: equip_effects: - type: "stat_change" gid: "planequarry_speed" - modifier: 100 + modifier: 20 - type: "stat_change" gid: "accuracy" modifier: 2 @@ -858,6 +870,7 @@ arcanite_axe: whatnot: tags: - "tool" + - "axe" equip_slots: - "mainhand" equip_requirements: @@ -867,7 +880,7 @@ arcanite_axe: equip_effects: - type: "stat_change" gid: "wealdreap_speed" - modifier: 100 + modifier: 20 - type: "stat_change" gid: "accuracy" modifier: 2 @@ -880,6 +893,7 @@ arcanite_spade: whatnot: tags: - "tool" + - "spade" equip_slots: - "mainhand" equip_requirements: @@ -889,19 +903,34 @@ arcanite_spade: equip_effects: - type: "stat_change" gid: "magiculture_speed" - modifier: 100 + modifier: 20 - type: "stat_change" gid: "accuracy" modifier: 2 - type: "stat_change" gid: "power" modifier: 3 +apprentice_wand: + name: "apprentice wand" + description: "A simple magic wand." + whatnot: + tags: + - "tool" + - "focus" + equip_slots: + - "mainhand" + - "offhand" + equip_effects: + - type: "stat_change" + gid: "havencast_speed" + modifier: 2 aethermesh: name: "aethermesh" description: "A tool for manatrawl." whatnot: tags: - "tool" + - "aethermesh" equip_slots: - "mainhand" quarrying_draught: @@ -1094,3 +1123,69 @@ balgoloth_claw: gid: "pierce" min: 1 max: 6 +faint_mana: + name: "faint mana" + description: "A weak omen that provides a very small amount of magical energy." + whatnot: + tags: + - "omen" + infix_skills: + - gid: "havencast" + infix_effects: + - type: "stat_change" + gid: "mana" + modifier: 1 +minor_mana: + name: "minor mana" + description: "A weak omen that provides a small amount of magical energy." + whatnot: + tags: + - "omen" + infix_skills: + - gid: "havencast" + infix_effects: + - type: "stat_change" + gid: "mana" + modifier: 2 +faint_forging_haste: + name: "faint forging haste" + description: "A very weak omen that grants a very small increase to otherforge speed." + whatnot: + tags: + - "omen" + infix_skills: + - gid: "otherforge" + infix_effects: + - type: "stat_change" + gid: "otherforge_speed" + modifier: 2 +faint_seething_haste: + name: "faint seething haste" + description: "A very weak omen that grants a very small increase to fluxseethe speed." + whatnot: + tags: + - "omen" + infix_skills: + - gid: "fluxseethe" + infix_effects: + - type: "stat_change" + gid: "fluxseethe_speed" + modifier: 2 +faint_working_haste: + name: "faint working haste" + description: "A very weak omen that grants a very small increase to spicework speed." + whatnot: + tags: + - "omen" + infix_skills: + - gid: "spicework" + infix_effects: + - type: "stat_change" + gid: "spicework_speed" + modifier: 2 +simple_spellpage: + name: "simple spellpage" + description: "An old piece of parchment covered in magical script. Deciphering it can yield knowledge of a spell." + whatnot: + tags: + - "material" diff --git a/data/locations.yml b/data/locations.yml index 480fc30..10bc028 100644 --- a/data/locations.yml +++ b/data/locations.yml @@ -1,4 +1,4 @@ floret_region: - name: "Floret Region" - description: "A land not too far from the great city of Morning nestled against the Sor River." + name: "Floret" + description: "A relatively safe haven nestled against the great Sor River." whatnot: diff --git a/data/monsters.yml b/data/monsters.yml index 26a6f90..01596b0 100644 --- a/data/monsters.yml +++ b/data/monsters.yml @@ -68,7 +68,7 @@ stalk_beast: awards: - type: "xp" gid: "beastslay" - base: 9 + base: 7 - type: "item" chance: 1 table: @@ -109,7 +109,7 @@ grinpad: awards: - type: "xp" gid: "beastslay" - base: 13 + base: 11 - type: "item" chance: 1 table: @@ -119,6 +119,8 @@ grinpad: - gid: "vestige" score: 0.85 max_quantity: 10 + - gid: "simple_spellpage" + score: 0.99 - gid: "rusted_lockbox" score: 0.995 - gid: "warm_diadem" @@ -155,7 +157,7 @@ lesser_trodgeathomp: awards: - type: "xp" gid: "beastslay" - base: 20 + base: 16 - type: "title" gid: "retributor" - type: "item" @@ -211,7 +213,7 @@ bollyrot: awards: - type: "xp" gid: "beastslay" - base: 20 + base: 13 - type: "item" chance: 1 table: @@ -260,7 +262,7 @@ crypt_writhe: awards: - type: "xp" gid: "beastslay" - base: 28 + base: 20 - type: "item" chance: 0.05 gid: "black_gizzard" diff --git a/data/skills.yml b/data/skills.yml index 68de599..0bbd63a 100644 --- a/data/skills.yml +++ b/data/skills.yml @@ -52,7 +52,7 @@ omenbind: ago developed a technique for harnessing magical power and spells within physical objects like crystalline batteries and gem-laden crowns. These power-imbued objects are called omens. The practice of omenbinding flourishes to this day, and indeed the world would not be the same without its providing for saving and storing up power to be - used at a later time, or in greater mass. The most powerful omenbinders have been known to bind energy even within + used at a later time, or in greater mass. The most powerful omenbinding have been known to bind energy even within flesh, though this practice is as risky as it is dangerous. otherforge: name: "Otherforge" diff --git a/data/titles.yml b/data/titles.yml index a8c228a..23ff009 100644 --- a/data/titles.yml +++ b/data/titles.yml @@ -8,3 +8,17 @@ aspirant: name: "Aspirant" sentinel: name: "Sentinel" +spiteful: + name: "Spiteful" +hateful: + name: "Hateful" +vicious: + name: "Vicious" +slayer: + name: "Slayer" +butcher: + name: "Butcher" +slaughterer: + name: "Slaughterer" +massacrer: + name: "Massacrer" diff --git a/db/migrate/20210614004827_create_item_infixes.rb b/db/migrate/20210614004827_create_item_infixes.rb new file mode 100644 index 0000000..5b49892 --- /dev/null +++ b/db/migrate/20210614004827_create_item_infixes.rb @@ -0,0 +1,11 @@ +class CreateItemInfixes < ActiveRecord::Migration[6.1] + def change + create_table :item_infixes do |t| + t.references :character, null: false, foreign_key: true + t.references :item, null: false, foreign_key: true + t.references :skill, null: false, foreign_key: true + + t.timestamps + end + end +end diff --git a/db/migrate/20210616014044_create_monster_kills.rb b/db/migrate/20210616014044_create_monster_kills.rb new file mode 100644 index 0000000..b61908d --- /dev/null +++ b/db/migrate/20210616014044_create_monster_kills.rb @@ -0,0 +1,11 @@ +class CreateMonsterKills < ActiveRecord::Migration[6.1] + def change + create_table :monster_kills do |t| + t.references :monster, null: false, foreign_key: true + t.references :character, null: false, foreign_key: true + t.bigint :quantity + + t.timestamps + end + end +end diff --git a/db/schema.rb b/db/schema.rb index 75ebad0..6c503fd 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_06_10_010413) do +ActiveRecord::Schema.define(version: 2021_06_16_014044) do # These are extensions that must be enabled in order to support this database enable_extension "plpgsql" @@ -156,6 +156,17 @@ ActiveRecord::Schema.define(version: 2021_06_10_010413) do t.index ["character_id"], name: "index_hearths_on_character_id" end + create_table "item_infixes", force: :cascade do |t| + t.bigint "character_id", null: false + t.bigint "item_id", null: false + t.bigint "skill_id", null: false + t.datetime "created_at", precision: 6, null: false + t.datetime "updated_at", precision: 6, null: false + t.index ["character_id"], name: "index_item_infixes_on_character_id" + t.index ["item_id"], name: "index_item_infixes_on_item_id" + t.index ["skill_id"], name: "index_item_infixes_on_skill_id" + end + create_table "items", force: :cascade do |t| t.string "gid" t.string "name" @@ -196,6 +207,16 @@ ActiveRecord::Schema.define(version: 2021_06_10_010413) do t.index ["sender_id"], name: "index_messages_on_sender_id" end + create_table "monster_kills", force: :cascade do |t| + t.bigint "monster_id", null: false + t.bigint "character_id", null: false + t.bigint "quantity" + t.datetime "created_at", precision: 6, null: false + t.datetime "updated_at", precision: 6, null: false + t.index ["character_id"], name: "index_monster_kills_on_character_id" + t.index ["monster_id"], name: "index_monster_kills_on_monster_id" + end + create_table "monster_spawn_combats", force: :cascade do |t| t.bigint "monster_spawn_id", null: false t.bigint "character_id", null: false @@ -308,10 +329,15 @@ ActiveRecord::Schema.define(version: 2021_06_10_010413) do add_foreign_key "hearth_plantings", "hearths" add_foreign_key "hearth_plantings", "items" add_foreign_key "hearths", "characters" + add_foreign_key "item_infixes", "characters" + add_foreign_key "item_infixes", "items" + add_foreign_key "item_infixes", "skills" add_foreign_key "learned_activities", "activities" add_foreign_key "learned_activities", "characters" add_foreign_key "messages", "characters", column: "recipient_id" add_foreign_key "messages", "characters", column: "sender_id" + add_foreign_key "monster_kills", "characters" + add_foreign_key "monster_kills", "monsters" add_foreign_key "monster_spawn_combats", "characters" add_foreign_key "monster_spawn_combats", "monster_spawns" add_foreign_key "monster_spawns", "locations" diff --git a/test/fixtures/item_infixes.yml b/test/fixtures/item_infixes.yml new file mode 100644 index 0000000..535831c --- /dev/null +++ b/test/fixtures/item_infixes.yml @@ -0,0 +1,11 @@ +# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html + +one: + character: one + item: one + skill: one + +two: + character: two + item: two + skill: two diff --git a/test/fixtures/monster_kills.yml b/test/fixtures/monster_kills.yml new file mode 100644 index 0000000..776b079 --- /dev/null +++ b/test/fixtures/monster_kills.yml @@ -0,0 +1,11 @@ +# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html + +one: + monster: one + character: one + quantity: + +two: + monster: two + character: two + quantity: diff --git a/test/models/item_infix_test.rb b/test/models/item_infix_test.rb new file mode 100644 index 0000000..a6f63a3 --- /dev/null +++ b/test/models/item_infix_test.rb @@ -0,0 +1,7 @@ +require "test_helper" + +class ItemInfixTest < ActiveSupport::TestCase + # test "the truth" do + # assert true + # end +end diff --git a/test/models/monster_kill_test.rb b/test/models/monster_kill_test.rb new file mode 100644 index 0000000..aef80ac --- /dev/null +++ b/test/models/monster_kill_test.rb @@ -0,0 +1,7 @@ +require "test_helper" + +class MonsterKillTest < ActiveSupport::TestCase + # test "the truth" do + # assert true + # end +end @@ -2705,9 +2705,9 @@ dns-equal@^1.0.0: integrity sha1-s55/HabrCnW6nBcySzR1PEfgZU0= dns-packet@^1.3.1: - version "1.3.1" - resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-1.3.1.tgz#12aa426981075be500b910eedcd0b47dd7deda5a" - integrity sha512-0UxfQkMhYAUaZI+xrNZOz/as5KgDU0M/fQ9b6SpkyLbk3GEswDi6PADJVaYJradtRVsRIlF1zLyOodbcTCDzUg== + version "1.3.4" + resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-1.3.4.tgz#e3455065824a2507ba886c55a89963bb107dec6f" + integrity sha512-BQ6F4vycLXBvdrJZ6S3gZewt6rcrks9KBgM9vrhW+knGRqc8uEdT7fuCwloc7nny5xNoMJ17HGH0R/6fpo8ECA== dependencies: ip "^1.1.0" safe-buffer "^5.0.1" @@ -7984,9 +7984,9 @@ wrappy@1: integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= ws@^6.2.1: - version "6.2.1" - resolved "https://registry.yarnpkg.com/ws/-/ws-6.2.1.tgz#442fdf0a47ed64f59b6a5d8ff130f4748ed524fb" - integrity sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA== + version "6.2.2" + resolved "https://registry.yarnpkg.com/ws/-/ws-6.2.2.tgz#dd5cdbd57a9979916097652d78f1cc5faea0c32e" + integrity sha512-zmhltoSR8u1cnDsD43TX59mzoMZsLKqUweyYBAIvTngR3shc0W6aOZylZmq/7hqyVxPdi+5Ud2QInblgyE72fw== dependencies: async-limiter "~1.0.0" |