veloren/voxygen/src/hud/diary.rs
2022-01-15 21:50:27 -05:00

3516 lines
131 KiB
Rust

use super::{
img_ids::{Imgs, ImgsRot},
item_imgs::{animate_by_pulse, ItemImgs, ItemKey::Tool},
Position, PositionSpecifier, Show, BLACK, CRITICAL_HP_COLOR, HP_COLOR, TEXT_COLOR,
UI_HIGHLIGHT_0, UI_MAIN, XP_COLOR,
};
use crate::{
game_input::GameInput,
hud::{
self,
slots::{AbilitySlot, SlotManager},
util,
},
ui::{
fonts::Fonts,
slot::{ContentSize, SlotMaker},
ImageFrame, Tooltip, TooltipManager, Tooltipable,
},
GlobalState,
};
use conrod_core::{
color, image,
widget::{self, Button, Image, Rectangle, State, Text},
widget_ids, Color, Colorable, Labelable, Positionable, Sizeable, UiCell, Widget, WidgetCommon,
};
use i18n::Localization;
use vek::*;
use client::{self, Client};
use common::{
combat,
comp::{
self,
ability::{Ability, ActiveAbilities, AuxiliaryAbility, MAX_ABILITIES},
inventory::{
item::{
tool::{MaterialStatManifest, ToolKind},
ItemKind,
},
slot::EquipSlot,
},
skills::{
self, AxeSkill, BowSkill, ClimbSkill, GeneralSkill, HammerSkill, MiningSkill,
RollSkill, SceptreSkill, Skill, StaffSkill, SwimSkill, SwordSkill, SKILL_MODIFIERS,
},
skillset::{SkillGroupKind, SkillSet},
Body, Energy, Health, Inventory, Poise,
},
consts::{ENERGY_PER_LEVEL, HP_PER_LEVEL},
};
use std::borrow::Cow;
const ART_SIZE: [f64; 2] = [320.0, 320.0];
widget_ids! {
pub struct Ids {
frame,
bg,
icon,
close,
title,
content_align,
section_imgs[],
section_btns[],
// Skill tree stuffs
exp_bar_bg,
exp_bar_frame,
exp_bar_content_align,
exp_bar_content,
exp_bar_rank,
exp_bar_txt,
tree_title_txt,
lock_imgs[],
available_pts_txt,
weapon_imgs[],
weapon_btns[],
skills_top_l_align,
skills_top_r_align,
skills_bot_l_align,
skills_bot_r_align,
skills_top_l[],
skills_top_r[],
skills_bot_l[],
skills_bot_r[],
sword_render,
skill_sword_combo_0,
skill_sword_combo_1,
skill_sword_combo_2,
skill_sword_combo_3,
skill_sword_combo_4,
skill_sword_dash_0,
skill_sword_dash_1,
skill_sword_dash_2,
skill_sword_dash_3,
skill_sword_dash_4,
skill_sword_dash_5,
skill_sword_dash_6,
skill_sword_spin_0,
skill_sword_spin_1,
skill_sword_spin_2,
skill_sword_spin_3,
skill_sword_spin_4,
skill_sword_passive_0,
axe_render,
skill_axe_combo_0,
skill_axe_combo_1,
skill_axe_combo_2,
skill_axe_combo_3,
skill_axe_combo_4,
skill_axe_spin_0,
skill_axe_spin_1,
skill_axe_spin_2,
skill_axe_spin_3,
skill_axe_spin_4,
skill_axe_spin_5,
skill_axe_leap_0,
skill_axe_leap_1,
skill_axe_leap_2,
skill_axe_leap_3,
skill_axe_leap_4,
hammer_render,
skill_hammer_combo_0,
skill_hammer_combo_1,
skill_hammer_combo_2,
skill_hammer_combo_3,
skill_hammer_combo_4,
skill_hammer_charged_0,
skill_hammer_charged_1,
skill_hammer_charged_2,
skill_hammer_charged_3,
skill_hammer_charged_4,
skill_hammer_leap_0,
skill_hammer_leap_1,
skill_hammer_leap_2,
skill_hammer_leap_3,
skill_hammer_leap_4,
skill_hammer_leap_5,
bow_render,
skill_bow_charged_0,
skill_bow_charged_1,
skill_bow_charged_2,
skill_bow_charged_3,
skill_bow_charged_4,
skill_bow_charged_5,
skill_bow_repeater_0,
skill_bow_repeater_1,
skill_bow_repeater_2,
skill_bow_repeater_3,
skill_bow_shotgun_0,
skill_bow_shotgun_1,
skill_bow_shotgun_2,
skill_bow_shotgun_3,
skill_bow_shotgun_4,
skill_bow_passive_0,
staff_render,
skill_staff_basic_0,
skill_staff_basic_1,
skill_staff_basic_2,
skill_staff_basic_3,
skill_staff_basic_4,
skill_staff_beam_0,
skill_staff_beam_1,
skill_staff_beam_2,
skill_staff_beam_3,
skill_staff_beam_4,
skill_staff_shockwave_0,
skill_staff_shockwave_1,
skill_staff_shockwave_2,
skill_staff_shockwave_3,
skill_staff_shockwave_4,
sceptre_render,
skill_sceptre_lifesteal_0,
skill_sceptre_lifesteal_1,
skill_sceptre_lifesteal_2,
skill_sceptre_lifesteal_3,
skill_sceptre_lifesteal_4,
skill_sceptre_heal_0,
skill_sceptre_heal_1,
skill_sceptre_heal_2,
skill_sceptre_heal_3,
skill_sceptre_heal_4,
skill_sceptre_aura_0,
skill_sceptre_aura_1,
skill_sceptre_aura_2,
skill_sceptre_aura_3,
skill_sceptre_aura_4,
pick_render,
skill_pick_m1,
skill_pick_m1_0,
skill_pick_m1_1,
skill_pick_m1_2,
general_combat_render_0,
general_combat_render_1,
skill_general_stat_0,
skill_general_stat_1,
skill_general_tree_0,
skill_general_tree_1,
skill_general_tree_2,
skill_general_tree_3,
skill_general_tree_4,
skill_general_tree_5,
skill_general_roll_0,
skill_general_roll_1,
skill_general_roll_2,
skill_general_roll_3,
skill_general_climb_0,
skill_general_climb_1,
skill_general_climb_2,
skill_general_swim_0,
skill_general_swim_1,
// Ability selection
spellbook_art,
sb_page_left_align,
sb_page_right_align,
spellbook_skills_bg,
spellbook_btn,
spellbook_btn_bg,
ability_select_title,
ability_page_left,
ability_page_right,
active_abilities[],
active_abilities_keys[],
main_weap_select,
off_weap_select,
abilities[],
ability_frames[],
abilities_dual[],
ability_titles[],
ability_descs[],
dragged_ability,
// Stats
stat_names[],
stat_values[],
}
}
#[derive(WidgetCommon)]
pub struct Diary<'a> {
show: &'a Show,
_client: &'a Client,
global_state: &'a GlobalState,
skill_set: &'a SkillSet,
active_abilities: &'a ActiveAbilities,
inventory: &'a Inventory,
health: &'a Health,
energy: &'a Energy,
poise: &'a Poise,
body: &'a Body,
msm: &'a MaterialStatManifest,
imgs: &'a Imgs,
item_imgs: &'a ItemImgs,
fonts: &'a Fonts,
localized_strings: &'a Localization,
rot_imgs: &'a ImgsRot,
tooltip_manager: &'a mut TooltipManager,
slot_manager: &'a mut SlotManager,
pulse: f32,
#[conrod(common_builder)]
common: widget::CommonBuilder,
created_btns_top_l: usize,
created_btns_top_r: usize,
created_btns_bot_l: usize,
created_btns_bot_r: usize,
}
pub struct DiaryShow {
pub skilltreetab: SelectedSkillTree,
pub section: DiarySection,
}
impl Default for DiaryShow {
fn default() -> Self {
Self {
skilltreetab: SelectedSkillTree::General,
section: DiarySection::SkillTrees,
}
}
}
#[allow(clippy::too_many_arguments)]
impl<'a> Diary<'a> {
pub fn new(
show: &'a Show,
_client: &'a Client,
global_state: &'a GlobalState,
skill_set: &'a SkillSet,
active_abilities: &'a ActiveAbilities,
inventory: &'a Inventory,
health: &'a Health,
energy: &'a Energy,
poise: &'a Poise,
body: &'a Body,
msm: &'a MaterialStatManifest,
imgs: &'a Imgs,
item_imgs: &'a ItemImgs,
fonts: &'a Fonts,
localized_strings: &'a Localization,
rot_imgs: &'a ImgsRot,
tooltip_manager: &'a mut TooltipManager,
slot_manager: &'a mut SlotManager,
pulse: f32,
) -> Self {
Self {
show,
_client,
global_state,
skill_set,
active_abilities,
inventory,
health,
energy,
poise,
body,
msm,
imgs,
item_imgs,
fonts,
localized_strings,
rot_imgs,
tooltip_manager,
slot_manager,
pulse,
common: widget::CommonBuilder::default(),
created_btns_top_l: 0,
created_btns_top_r: 0,
created_btns_bot_l: 0,
created_btns_bot_r: 0,
}
}
}
pub type SelectedSkillTree = SkillGroupKind;
// TODO: make it enum?
const TREES: [&str; 8] = [
"General Combat",
"Sword",
"Hammer",
"Axe",
"Sceptre",
"Bow",
"Fire Staff",
"Mining",
];
// Possible future sections: Bestiary ("Pokedex" of fought enemies), Weapon and
// armour catalogue, Achievements...
const SECTIONS: [&str; 3] = ["Skill-Trees", "Abilities", "Stats"];
pub enum Event {
Close,
ChangeSkillTree(SelectedSkillTree),
UnlockSkill(Skill),
ChangeSection(DiarySection),
}
#[derive(PartialEq)]
pub enum DiarySection {
SkillTrees,
AbilitySelection,
Stats,
}
pub struct DiaryState {
ids: Ids,
ability_page: usize,
}
impl<'a> Widget for Diary<'a> {
type Event = Vec<Event>;
type State = DiaryState;
type Style = ();
fn init_state(&self, id_gen: widget::id::Generator) -> Self::State {
DiaryState {
ids: Ids::new(id_gen),
ability_page: 0,
}
}
fn style(&self) -> Self::Style {}
fn update(mut self, args: widget::UpdateArgs<Self>) -> Self::Event {
common_base::prof_span!("Diary::update");
let widget::UpdateArgs { state, ui, .. } = args;
let mut events = Vec::new();
// Tooltips
let diary_tooltip = Tooltip::new({
// Edge images [t, b, r, l]
// Corner images [tr, tl, br, bl]
let edge = &self.rot_imgs.tt_side;
let corner = &self.rot_imgs.tt_corner;
ImageFrame::new(
[edge.cw180, edge.none, edge.cw270, edge.cw90],
[corner.none, corner.cw270, corner.cw90, corner.cw180],
Color::Rgba(0.08, 0.07, 0.04, 1.0),
5.0,
)
})
.title_font_size(self.fonts.cyri.scale(15))
.parent(ui.window)
.desc_font_size(self.fonts.cyri.scale(12))
.font_id(self.fonts.cyri.conrod_id)
.desc_text_color(TEXT_COLOR);
//Animation timer Frame
let frame_ani = (self.pulse * 4.0/* speed factor */).cos() * 0.5 + 0.8;
Image::new(self.imgs.diary_bg)
.w_h(1202.0, 886.0)
.mid_top_with_margin_on(ui.window, 5.0)
.color(Some(UI_MAIN))
.set(state.ids.bg, ui);
Image::new(self.imgs.diary_frame)
.w_h(1202.0, 886.0)
.middle_of(state.ids.bg)
.color(Some(UI_HIGHLIGHT_0))
.set(state.ids.frame, ui);
// Icon
Image::new(self.imgs.spellbook_button)
.w_h(30.0, 27.0)
.top_left_with_margins_on(state.ids.frame, 8.0, 8.0)
.set(state.ids.icon, ui);
// X-Button
if Button::image(self.imgs.close_button)
.w_h(24.0, 25.0)
.hover_image(self.imgs.close_btn_hover)
.press_image(self.imgs.close_btn_press)
.top_right_with_margins_on(state.ids.frame, 0.0, 0.0)
.set(state.ids.close, ui)
.was_clicked()
{
events.push(Event::Close);
}
// Title
Text::new(self.localized_strings.get("hud.diary"))
.mid_top_with_margin_on(state.ids.frame, 3.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(29))
.color(TEXT_COLOR)
.set(state.ids.title, ui);
// Content Alignment
Rectangle::fill_with([599.0 * 2.0, 419.0 * 2.0], color::TRANSPARENT)
.mid_top_with_margin_on(state.ids.frame, 46.0)
.set(state.ids.content_align, ui);
// Contents
// Section buttons
let sel_section = &self.show.diary_fields.section;
// Update len
state.update(|s| {
s.ids
.section_imgs
.resize(SECTIONS.len(), &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.section_btns
.resize(SECTIONS.len(), &mut ui.widget_id_generator())
});
for (i, section_name) in SECTIONS.iter().copied().enumerate() {
let section = match section_from_str(section_name) {
Some(st) => st,
None => {
tracing::warn!("unexpected section name: {}", section_name);
continue;
},
};
// Section Icons
let section_desc = match section_name {
"Abilities" => "List of your currently available abilities.",
"Skill-Trees" => "",
"Stats" => "",
_ => "",
};
let btn_img = {
let img = match section_name {
"Abilities" => self.imgs.spellbook_ico,
"Skill-Trees" => self.imgs.skilltree_ico,
"Stats" => self.imgs.stats_ico,
_ => self.imgs.nothing,
};
if i == 0 {
Image::new(img).top_left_with_margins_on(state.ids.content_align, 0.0, -50.0)
} else {
Image::new(img).down_from(state.ids.section_btns[i - 1], 5.0)
}
};
btn_img.w_h(50.0, 50.0).set(state.ids.section_imgs[i], ui);
// Section Buttons
let border_image = if section == *sel_section {
self.imgs.wpn_icon_border_pressed
} else {
self.imgs.wpn_icon_border
};
let hover_image = if section == *sel_section {
self.imgs.wpn_icon_border_pressed
} else {
self.imgs.wpn_icon_border_mo
};
let press_image = if section == *sel_section {
self.imgs.wpn_icon_border_pressed
} else {
self.imgs.wpn_icon_border_press
};
let section_buttons = Button::image(border_image)
.w_h(50.0, 50.0)
.hover_image(hover_image)
.press_image(press_image)
.middle_of(state.ids.section_imgs[i])
.with_tooltip(
self.tooltip_manager,
section_name,
section_desc,
&diary_tooltip,
TEXT_COLOR,
)
.set(state.ids.section_btns[i], ui);
if section_buttons.was_clicked() {
events.push(Event::ChangeSection(section))
}
}
match self.show.diary_fields.section {
DiarySection::SkillTrees => {
// Skill Trees
let sel_tab = &self.show.diary_fields.skilltreetab;
// Skill Tree Selection
state.update(|s| {
s.ids
.weapon_btns
.resize(TREES.len(), &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.weapon_imgs
.resize(TREES.len(), &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.lock_imgs
.resize(TREES.len(), &mut ui.widget_id_generator())
});
// Draw skillgroup tab's icons
for (i, skilltree_name) in TREES.iter().copied().enumerate() {
let skill_group = match skill_tree_from_str(skilltree_name) {
Some(st) => st,
None => {
tracing::warn!("unexpected tree name: {}", skilltree_name);
continue;
},
};
// Check if we have this skill tree unlocked
let locked = !self.skill_set.skill_group_accessible(skill_group);
// Weapon button image
let btn_img = {
let img = match skilltree_name {
"General Combat" => self.imgs.swords_crossed,
"Sword" => self.imgs.sword,
"Hammer" => self.imgs.hammer,
"Axe" => self.imgs.axe,
"Sceptre" => self.imgs.sceptre,
"Bow" => self.imgs.bow,
"Fire Staff" => self.imgs.staff,
"Mining" => self.imgs.mining,
_ => self.imgs.nothing,
};
if i == 0 {
Image::new(img).top_left_with_margins_on(
state.ids.content_align,
10.0,
5.0,
)
} else {
Image::new(img).down_from(state.ids.weapon_btns[i - 1], 5.0)
}
};
btn_img.w_h(50.0, 50.0).set(state.ids.weapon_imgs[i], ui);
// Lock Image
if locked {
Image::new(self.imgs.lock)
.w_h(50.0, 50.0)
.middle_of(state.ids.weapon_imgs[i])
.graphics_for(state.ids.weapon_imgs[i])
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 0.8)))
.set(state.ids.lock_imgs[i], ui);
}
// Weapon icons
let have_points = {
let available = self.skill_set.available_sp(skill_group);
let earned = self.skill_set.earned_sp(skill_group);
let total_cost = skill_group.total_skill_point_cost();
available > 0 && (earned - available) < total_cost
};
let border_image = if skill_group == *sel_tab || have_points {
self.imgs.wpn_icon_border_pressed
} else {
self.imgs.wpn_icon_border
};
let hover_image = if skill_group == *sel_tab {
self.imgs.wpn_icon_border_pressed
} else {
self.imgs.wpn_icon_border_mo
};
let press_image = if skill_group == *sel_tab {
self.imgs.wpn_icon_border_pressed
} else {
self.imgs.wpn_icon_border_press
};
let color = if skill_group != *sel_tab && have_points {
Color::Rgba(0.92, 0.76, 0.0, frame_ani)
} else {
TEXT_COLOR
};
let tooltip_txt = if locked {
self.localized_strings.get("hud.skill.not_unlocked")
} else {
""
};
let wpn_button = Button::image(border_image)
.w_h(50.0, 50.0)
.hover_image(hover_image)
.press_image(press_image)
.middle_of(state.ids.weapon_imgs[i])
.image_color(color)
.with_tooltip(
self.tooltip_manager,
skilltree_name,
tooltip_txt,
&diary_tooltip,
TEXT_COLOR,
)
.set(state.ids.weapon_btns[i], ui);
if wpn_button.was_clicked() {
events.push(Event::ChangeSkillTree(skill_group))
}
}
// Exp Bars and Rank Display
let current_exp = self.skill_set.available_experience(*sel_tab) as f64;
let max_exp = self.skill_set.skill_point_cost(*sel_tab) as f64;
let exp_percentage = current_exp / max_exp;
let rank = self.skill_set.earned_sp(*sel_tab);
let rank_txt = format!("{}", rank);
let exp_txt = format!("{}/{}", current_exp, max_exp);
let available_pts = self.skill_set.available_sp(*sel_tab);
let available_pts_txt = format!("{}", available_pts);
Image::new(self.imgs.diary_exp_bg)
.w_h(480.0, 76.0)
.mid_bottom_with_margin_on(state.ids.content_align, 10.0)
.set(state.ids.exp_bar_bg, ui);
Rectangle::fill_with([400.0, 40.0], color::TRANSPARENT)
.top_left_with_margins_on(state.ids.exp_bar_bg, 32.0, 40.0)
.set(state.ids.exp_bar_content_align, ui);
Image::new(self.imgs.bar_content)
.w_h(400.0 * exp_percentage, 40.0)
.top_left_with_margins_on(state.ids.exp_bar_content_align, 0.0, 0.0)
.color(Some(XP_COLOR))
.set(state.ids.exp_bar_content, ui);
Image::new(self.imgs.diary_exp_frame)
.w_h(480.0, 76.0)
.color(Some(UI_HIGHLIGHT_0))
.middle_of(state.ids.exp_bar_bg)
.set(state.ids.exp_bar_frame, ui);
// Show EXP bar text on hover
if ui
.widget_input(state.ids.exp_bar_frame)
.mouse()
.map_or(false, |m| m.is_over())
{
Text::new(&exp_txt)
.mid_top_with_margin_on(state.ids.exp_bar_frame, 47.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(14))
.color(TEXT_COLOR)
.graphics_for(state.ids.exp_bar_frame)
.set(state.ids.exp_bar_txt, ui);
}
Text::new(&rank_txt)
.mid_top_with_margin_on(state.ids.exp_bar_frame, match rank {
0..=99 => 5.0,
100..=999 => 8.0,
_ => 10.0,
})
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(match rank {
0..=99 => 28,
100..=999 => 21,
_ => 15,
}))
.color(TEXT_COLOR)
.set(state.ids.exp_bar_rank, ui);
Text::new(
&self
.localized_strings
.get("hud.skill.sp_available")
.replace("{number}", &available_pts_txt),
)
.mid_top_with_margin_on(state.ids.content_align, 700.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(28))
.color(if available_pts > 0 {
Color::Rgba(0.92, 0.76, 0.0, frame_ani)
} else {
TEXT_COLOR
})
.set(state.ids.available_pts_txt, ui);
// Skill Trees
// Alignment Placing
let x = 200.0;
let y = 100.0;
// Alignment rectangles for skills
Rectangle::fill_with([124.0 * 2.0, 124.0 * 2.0], color::TRANSPARENT)
.top_left_with_margins_on(state.ids.content_align, y, x)
.set(state.ids.skills_top_l_align, ui);
Rectangle::fill_with([124.0 * 2.0, 124.0 * 2.0], color::TRANSPARENT)
.top_right_with_margins_on(state.ids.content_align, y, x)
.set(state.ids.skills_top_r_align, ui);
Rectangle::fill_with([124.0 * 2.0, 124.0 * 2.0], color::TRANSPARENT)
.bottom_left_with_margins_on(state.ids.content_align, y, x)
.set(state.ids.skills_bot_l_align, ui);
Rectangle::fill_with([124.0 * 2.0, 124.0 * 2.0], color::TRANSPARENT)
.bottom_right_with_margins_on(state.ids.content_align, y, x)
.set(state.ids.skills_bot_r_align, ui);
match sel_tab {
SelectedSkillTree::General => {
self.handle_general_skills_window(&diary_tooltip, state, ui, events)
},
SelectedSkillTree::Weapon(ToolKind::Sword) => {
self.handle_sword_skills_window(&diary_tooltip, state, ui, events)
},
SelectedSkillTree::Weapon(ToolKind::Hammer) => {
self.handle_hammer_skills_window(&diary_tooltip, state, ui, events)
},
SelectedSkillTree::Weapon(ToolKind::Axe) => {
self.handle_axe_skills_window(&diary_tooltip, state, ui, events)
},
SelectedSkillTree::Weapon(ToolKind::Sceptre) => {
self.handle_sceptre_skills_window(&diary_tooltip, state, ui, events)
},
SelectedSkillTree::Weapon(ToolKind::Bow) => {
self.handle_bow_skills_window(&diary_tooltip, state, ui, events)
},
SelectedSkillTree::Weapon(ToolKind::Staff) => {
self.handle_staff_skills_window(&diary_tooltip, state, ui, events)
},
SelectedSkillTree::Weapon(ToolKind::Pick) => {
self.handle_mining_skills_window(&diary_tooltip, state, ui, events)
},
_ => events,
}
},
DiarySection::AbilitySelection => {
use comp::ability::AbilityInput;
// Background Art
Image::new(self.imgs.book_bg)
.w_h(299.0 * 4.0, 184.0 * 4.0)
.mid_top_with_margin_on(state.ids.content_align, 4.0)
//.graphics_for(state.ids.content_align)
.set(state.ids.spellbook_art, ui);
Image::new(self.imgs.skills_bg)
.w_h(240.0 * 2.0, 40.0 * 2.0)
.mid_bottom_with_margin_on(state.ids.content_align, 8.0)
.set(state.ids.spellbook_skills_bg, ui);
Rectangle::fill_with([299.0 * 2.0, 184.0 * 4.0], color::TRANSPARENT)
.top_left_with_margins_on(state.ids.spellbook_art, 0.0, 0.0)
.set(state.ids.sb_page_left_align, ui);
Rectangle::fill_with([299.0 * 2.0, 184.0 * 4.0], color::TRANSPARENT)
.top_right_with_margins_on(state.ids.spellbook_art, 0.0, 0.0)
.set(state.ids.sb_page_right_align, ui);
// Display all active abilities on bottom of window
state.update(|s| {
s.ids
.active_abilities
.resize(MAX_ABILITIES, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.active_abilities_keys
.resize(MAX_ABILITIES, &mut ui.widget_id_generator())
});
let mut slot_maker = SlotMaker {
empty_slot: self.imgs.inv_slot,
filled_slot: self.imgs.inv_slot,
selected_slot: self.imgs.inv_slot_sel,
background_color: Some(UI_MAIN),
content_size: ContentSize {
width_height_ratio: 1.0,
max_fraction: 0.9,
},
selected_content_scale: 1.067,
amount_font: self.fonts.cyri.conrod_id,
amount_margins: Vec2::new(-4.0, 0.0),
amount_font_size: self.fonts.cyri.scale(12),
amount_text_color: TEXT_COLOR,
content_source: &(self.active_abilities, self.inventory, self.skill_set),
image_source: self.imgs,
slot_manager: Some(self.slot_manager),
pulse: 0.0,
};
for i in 0..MAX_ABILITIES {
let ability_id = self
.active_abilities
.get_ability(
AbilityInput::Auxiliary(i),
Some(self.inventory),
Some(self.skill_set),
)
.ability_id(Some(self.inventory));
let (ability_title, ability_desc) = if let Some(ability_id) = ability_id {
util::ability_description(ability_id)
} else {
("Drag an ability here to use it.", "")
};
let image_size = 80.0;
let image_offsets = 92.0 * i as f64;
let slot = AbilitySlot::Slot(i);
let mut ability_slot = slot_maker.fabricate(slot, [image_size; 2]);
if i == 0 {
ability_slot = ability_slot.top_left_with_margins_on(
state.ids.spellbook_skills_bg,
0.0,
32.0 + image_offsets,
);
} else {
ability_slot =
ability_slot.right_from(state.ids.active_abilities[i - 1], 4.0)
}
ability_slot
.with_tooltip(
self.tooltip_manager,
ability_title,
ability_desc,
&diary_tooltip,
TEXT_COLOR,
)
.set(state.ids.active_abilities[i], ui);
// Display Slot Keybinding
let keys = &self.global_state.settings.controls;
let key_layout = &self.global_state.window.key_layout;
let ability_key = [
GameInput::Slot1,
GameInput::Slot2,
GameInput::Slot3,
GameInput::Slot4,
GameInput::Slot5,
]
.get(i)
.and_then(|input| keys.get_binding(*input))
.map(|key| key.display_string(key_layout))
.unwrap_or_default();
Text::new(&ability_key)
.top_left_with_margins_on(state.ids.active_abilities[i], 0.0, 4.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(20))
.color(TEXT_COLOR)
.graphics_for(state.ids.active_abilities[i])
.set(state.ids.active_abilities_keys[i], ui);
}
let same_weap_kinds = self
.inventory
.equipped(EquipSlot::ActiveMainhand)
.zip(self.inventory.equipped(EquipSlot::ActiveOffhand))
.map_or(false, |(a, b)| {
if let (ItemKind::Tool(tool_a), ItemKind::Tool(tool_b)) = (&a.kind, &b.kind)
{
(a.ability_spec(), tool_a.kind) == (b.ability_spec(), tool_b.kind)
} else {
false
}
});
let main_weap_abilities = ActiveAbilities::iter_unlocked_abilities(
Some(self.inventory),
Some(self.skill_set),
EquipSlot::ActiveMainhand,
)
.map(AuxiliaryAbility::MainWeapon)
.map(|a| (Ability::from(a).ability_id(Some(self.inventory)), a));
let off_weap_abilities = ActiveAbilities::iter_unlocked_abilities(
Some(self.inventory),
Some(self.skill_set),
EquipSlot::ActiveOffhand,
)
.map(AuxiliaryAbility::OffWeapon)
.map(|a| (Ability::from(a).ability_id(Some(self.inventory)), a));
let abilities: Vec<_> = if same_weap_kinds {
// When the weapons have the same ability kind take only the main weapons,
main_weap_abilities.collect()
} else {
main_weap_abilities.chain(off_weap_abilities).collect()
};
const ABILITIES_PER_PAGE: usize = 12;
let page_indices = (abilities.len().saturating_sub(1)) / ABILITIES_PER_PAGE;
if state.ability_page > page_indices {
state.update(|s| s.ability_page = 0);
}
state.update(|s| {
s.ids
.abilities
.resize(ABILITIES_PER_PAGE, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.abilities_dual
.resize(ABILITIES_PER_PAGE, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.ability_titles
.resize(ABILITIES_PER_PAGE, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.ability_frames
.resize(ABILITIES_PER_PAGE, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.ability_descs
.resize(ABILITIES_PER_PAGE, &mut ui.widget_id_generator())
});
// Page button
// Left Arrow
let left_arrow = Button::image(if state.ability_page > 0 {
self.imgs.arrow_l
} else {
self.imgs.arrow_l_inactive
})
.bottom_left_with_margins_on(state.ids.spellbook_art, -83.0, 10.0)
.w_h(48.0, 55.0);
// Grey out arrows when inactive
if state.ability_page > 0 {
if left_arrow
.hover_image(self.imgs.arrow_l_click)
.press_image(self.imgs.arrow_l)
.set(state.ids.ability_page_left, ui)
.was_clicked()
{
state.update(|s| s.ability_page -= 1);
}
} else {
left_arrow.set(state.ids.ability_page_left, ui);
}
// Right Arrow
let right_arrow = Button::image(if state.ability_page < page_indices {
self.imgs.arrow_r
} else {
self.imgs.arrow_r_inactive
})
.bottom_right_with_margins_on(state.ids.spellbook_art, -83.0, 10.0)
.w_h(48.0, 55.0);
if state.ability_page < page_indices {
// Only show right button if not on last page
if right_arrow
.hover_image(self.imgs.arrow_r_click)
.press_image(self.imgs.arrow_r)
.set(state.ids.ability_page_right, ui)
.was_clicked()
{
state.update(|s| s.ability_page += 1);
};
} else {
right_arrow.set(state.ids.ability_page_right, ui);
}
let ability_start = state.ability_page * ABILITIES_PER_PAGE;
let mut slot_maker = SlotMaker {
empty_slot: self.imgs.inv_slot,
filled_slot: self.imgs.inv_slot,
selected_slot: self.imgs.inv_slot_sel,
background_color: Some(UI_MAIN),
content_size: ContentSize {
width_height_ratio: 1.0,
max_fraction: 1.0,
},
selected_content_scale: 1.067,
amount_font: self.fonts.cyri.conrod_id,
amount_margins: Vec2::new(-4.0, 0.0),
amount_font_size: self.fonts.cyri.scale(12),
amount_text_color: TEXT_COLOR,
content_source: &(self.active_abilities, self.inventory, self.skill_set),
image_source: self.imgs,
slot_manager: Some(self.slot_manager),
pulse: 0.0,
};
for (id_index, (ability_id, ability)) in abilities
.iter()
.skip(ability_start)
.take(ABILITIES_PER_PAGE)
.enumerate()
{
let (ability_title, ability_desc) =
util::ability_description(ability_id.unwrap_or(""));
let (align_state, image_offsets) = if id_index < 6 {
(state.ids.sb_page_left_align, 120.0 * id_index as f64)
} else {
(state.ids.sb_page_right_align, 120.0 * (id_index - 6) as f64)
};
Image::new(if same_weap_kinds {
self.imgs.ability_frame_dual
} else {
self.imgs.ability_frame
})
.w_h(566.0, 108.0)
.top_left_with_margins_on(align_state, 16.0 + image_offsets, 16.0)
.color(Some(UI_HIGHLIGHT_0))
.set(state.ids.ability_frames[id_index], ui);
let slot = AbilitySlot::Ability(*ability);
slot_maker
.fabricate(slot, [100.0; 2])
.top_left_with_margins_on(align_state, 20.0 + image_offsets, 20.0)
.set(state.ids.abilities[id_index], ui);
if same_weap_kinds {
if let AuxiliaryAbility::MainWeapon(slot) = ability {
let ability = AuxiliaryAbility::OffWeapon(*slot);
let slot = AbilitySlot::Ability(ability);
slot_maker
.fabricate(slot, [100.0; 2])
.top_right_with_margins_on(align_state, 20.0 + image_offsets, 20.0)
.set(state.ids.abilities_dual[id_index], ui);
}
}
// The page width...
let text_width = 299.0 * 2.0
- if same_weap_kinds && matches!(ability, AuxiliaryAbility::MainWeapon(_)) {
// with double the width of an ability image and some padding subtracted
// if dual wielding two of the same weapon kind
(20.0 + 100.0 + 10.0) * 2.0
} else {
// or the width of an ability image and some padding subtracted
// otherwise
20.0 * 2.0 + 100.0
};
Text::new(ability_title)
.top_left_with_margins_on(state.ids.abilities[id_index], 5.0, 110.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(28))
.color(TEXT_COLOR)
.w(text_width)
.graphics_for(state.ids.abilities[id_index])
.set(state.ids.ability_titles[id_index], ui);
Text::new(ability_desc)
.top_left_with_margins_on(state.ids.abilities[id_index], 40.0, 110.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(18))
.color(TEXT_COLOR)
.w(text_width)
.graphics_for(state.ids.abilities[id_index])
.set(state.ids.ability_descs[id_index], ui);
}
events
},
DiarySection::Stats => {
const STATS: [&str; 13] = [
"Hitpoints",
"Energy",
"Poise",
"Combat-Rating",
"Protection",
"Stun-Resistance",
"Crit-Power",
"Energy Reward",
"Stealth",
"Weapon Power",
"Weapon Speed",
"Weapon Poise",
"Weapon Crit-Chance",
];
// Background Art
Image::new(self.imgs.book_bg)
.w_h(299.0 * 4.0, 184.0 * 4.0)
.mid_top_with_margin_on(state.ids.content_align, 4.0)
.set(state.ids.spellbook_art, ui);
state.update(|s| {
s.ids
.stat_names
.resize(STATS.len(), &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.stat_values
.resize(STATS.len(), &mut ui.widget_id_generator())
});
for (i, stat) in STATS.iter().copied().enumerate() {
// Stat names
let mut txt = Text::new(stat)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(29))
.color(BLACK);
if i == 0 {
txt = txt.top_left_with_margins_on(state.ids.spellbook_art, 20.0, 20.0);
} else {
txt = txt.down_from(state.ids.stat_names[i - 1], 10.0);
};
txt.set(state.ids.stat_names[i], ui);
let main_weap_stats = self
.inventory
.equipped(EquipSlot::ActiveMainhand)
.and_then(|item| match &item.kind {
ItemKind::Tool(tool) => {
Some(tool.stats.resolve_stats(self.msm, item.components()))
},
_ => None,
});
let off_weap_stats = self
.inventory
.equipped(EquipSlot::ActiveOffhand)
.and_then(|item| match &item.kind {
ItemKind::Tool(tool) => {
Some(tool.stats.resolve_stats(self.msm, item.components()))
},
_ => None,
});
// Stat values
let value = match stat {
"Hitpoints" => format!("{}", self.health.base_max() as u32),
"Energy" => format!("{}", self.energy.base_max() as u32),
"Poise" => format!("{}", self.poise.base_max() as u32),
"Combat-Rating" => {
let cr = combat::combat_rating(
self.inventory,
self.health,
self.energy,
self.poise,
self.skill_set,
*self.body,
self.msm,
);
format!("{:.2}", cr * 10.0)
},
"Protection" => {
let protection = combat::compute_protection(Some(self.inventory));
match protection {
Some(prot) => format!("{}", prot),
None => String::from("Invincible"),
}
},
"Stun-Resistance" => {
let stun_res = Poise::compute_poise_damage_reduction(self.inventory);
format!("{:.2}%", stun_res * 100.0)
},
"Crit-Power" => {
let critpwr = combat::compute_crit_mult(Some(self.inventory));
format!("x{:.2}", critpwr)
},
"Energy Reward" => {
let energy_rew =
combat::compute_energy_reward_mod(Some(self.inventory));
format!("{:+.0}%", (energy_rew - 1.0) * 100.0)
},
"Stealth" => {
let stealth = combat::compute_stealth_coefficient(Some(self.inventory));
format!("{:.2}", stealth)
},
"Weapon Power" => match (main_weap_stats, off_weap_stats) {
(Some(m_stats), Some(o_stats)) => {
format!("{} {}", m_stats.power * 10.0, o_stats.power * 10.0)
},
(Some(stats), None) | (None, Some(stats)) => {
format!("{}", stats.power * 10.0)
},
(None, None) => String::new(),
},
"Weapon Speed" => {
let spd_fmt = |sp| (sp - 1.0) * 100.0;
match (main_weap_stats, off_weap_stats) {
(Some(m_stats), Some(o_stats)) => format!(
"{:+.0}% {:+.0}%",
spd_fmt(m_stats.speed),
spd_fmt(o_stats.speed)
),
(Some(stats), None) | (None, Some(stats)) => {
format!("{:+.0}%", spd_fmt(stats.speed))
},
_ => String::new(),
}
},
"Weapon Poise" => match (main_weap_stats, off_weap_stats) {
(Some(m_stats), Some(o_stats)) => {
format!(
"{} {}",
m_stats.effect_power * 10.0,
o_stats.effect_power * 10.0
)
},
(Some(stats), None) | (None, Some(stats)) => {
format!("{}", stats.effect_power * 10.0)
},
(None, None) => String::new(),
},
"Weapon Crit-Chance" => {
let crit_fmt = |cc| cc * 100.0;
match (main_weap_stats, off_weap_stats) {
(Some(m_stats), Some(o_stats)) => format!(
"{:.1}% {:.1}%",
crit_fmt(m_stats.crit_chance),
crit_fmt(o_stats.crit_chance)
),
(Some(stats), None) | (None, Some(stats)) => {
format!("{:.1}%", crit_fmt(stats.crit_chance))
},
(None, None) => String::new(),
}
},
unknown => unreachable!(unknown),
};
let mut number = Text::new(&value)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(29))
.color(BLACK);
if i == 0 {
number = number.right_from(state.ids.stat_names[i], 265.0);
} else {
number = number.down_from(state.ids.stat_values[i - 1], 10.0);
};
number.set(state.ids.stat_values[i], ui);
}
events
},
}
}
}
fn skill_tree_from_str(string: &str) -> Option<SelectedSkillTree> {
match string {
"General Combat" => Some(SelectedSkillTree::General),
"Sword" => Some(SelectedSkillTree::Weapon(ToolKind::Sword)),
"Hammer" => Some(SelectedSkillTree::Weapon(ToolKind::Hammer)),
"Axe" => Some(SelectedSkillTree::Weapon(ToolKind::Axe)),
"Sceptre" => Some(SelectedSkillTree::Weapon(ToolKind::Sceptre)),
"Bow" => Some(SelectedSkillTree::Weapon(ToolKind::Bow)),
"Fire Staff" => Some(SelectedSkillTree::Weapon(ToolKind::Staff)),
"Mining" => Some(SelectedSkillTree::Weapon(ToolKind::Pick)),
_ => None,
}
}
fn section_from_str(string: &str) -> Option<DiarySection> {
match string {
"Abilities" => Some(DiarySection::AbilitySelection),
"Skill-Trees" => Some(DiarySection::SkillTrees),
"Stats" => Some(DiarySection::Stats),
_ => None,
}
}
enum SkillIcon<'a> {
Unlockable {
skill: Skill,
image: image::Id,
position: PositionSpecifier,
id: widget::Id,
},
Descriptive {
title: &'a str,
desc: &'a str,
image: image::Id,
position: PositionSpecifier,
id: widget::Id,
},
}
impl<'a> Diary<'a> {
fn handle_general_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
let tree_title = self.localized_strings.get("common.weapons.general");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 2;
let skills_top_r = 6;
let skills_bot_l = 4;
let skills_bot_r = 5;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
use skills::{GeneralSkill::*, RollSkill::*};
use SkillGroupKind::*;
use ToolKind::*;
// General Combat
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_general_combat_left".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.general_combat_render_0, ui);
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_general_combat_right".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.general_combat_render_0)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.general_combat_render_1, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Unlockable {
skill: Skill::General(HealthIncrease),
image: self.imgs.health_plus_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_general_stat_0,
},
SkillIcon::Unlockable {
skill: Skill::General(EnergyIncrease),
image: self.imgs.energy_plus_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_general_stat_1,
},
// Top right skills
SkillIcon::Unlockable {
skill: Skill::UnlockGroup(Weapon(Sword)),
image: self.imgs.unlock_sword_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[0], 3.0),
id: state.ids.skill_general_tree_0,
},
SkillIcon::Unlockable {
skill: Skill::UnlockGroup(Weapon(Axe)),
image: self.imgs.unlock_axe_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[1], 3.0),
id: state.ids.skill_general_tree_1,
},
SkillIcon::Unlockable {
skill: Skill::UnlockGroup(Weapon(Hammer)),
image: self.imgs.unlock_hammer_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[2], 3.0),
id: state.ids.skill_general_tree_2,
},
SkillIcon::Unlockable {
skill: Skill::UnlockGroup(Weapon(Bow)),
image: self.imgs.unlock_bow_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[3], 3.0),
id: state.ids.skill_general_tree_3,
},
SkillIcon::Unlockable {
skill: Skill::UnlockGroup(Weapon(Staff)),
image: self.imgs.unlock_staff_skill0,
position: MidTopWithMarginOn(state.ids.skills_top_r[4], 3.0),
id: state.ids.skill_general_tree_4,
},
SkillIcon::Unlockable {
skill: Skill::UnlockGroup(Weapon(Sceptre)),
image: self.imgs.unlock_sceptre_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[5], 3.0),
id: state.ids.skill_general_tree_5,
},
// Bottom left skills
SkillIcon::Descriptive {
title: "hud.skill.dodge_title",
desc: "hud.skill.dodge",
image: self.imgs.skill_dodge_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[0], 3.0),
id: state.ids.skill_general_roll_0,
},
SkillIcon::Unlockable {
skill: Skill::Roll(RollSkill::Cost),
image: self.imgs.utility_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[1], 3.0),
id: state.ids.skill_general_roll_1,
},
SkillIcon::Unlockable {
skill: Skill::Roll(Strength),
image: self.imgs.utility_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[2], 3.0),
id: state.ids.skill_general_roll_2,
},
SkillIcon::Unlockable {
skill: Skill::Roll(Duration),
image: self.imgs.utility_duration_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[3], 3.0),
id: state.ids.skill_general_roll_3,
},
// Bottom right skills
SkillIcon::Descriptive {
title: "hud.skill.climbing_title",
desc: "hud.skill.climbing",
image: self.imgs.skill_climbing_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_r[0], 3.0),
id: state.ids.skill_general_climb_0,
},
SkillIcon::Unlockable {
skill: Skill::Climb(ClimbSkill::Cost),
image: self.imgs.utility_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_r[1], 3.0),
id: state.ids.skill_general_climb_1,
},
SkillIcon::Unlockable {
skill: Skill::Climb(ClimbSkill::Speed),
image: self.imgs.utility_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_r[2], 3.0),
id: state.ids.skill_general_climb_2,
},
SkillIcon::Descriptive {
title: "hud.skill.swim_title",
desc: "hud.skill.swim",
image: self.imgs.skill_swim_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_r[3], 3.0),
id: state.ids.skill_general_swim_0,
},
SkillIcon::Unlockable {
skill: Skill::Swim(SwimSkill::Speed),
image: self.imgs.utility_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_r[4], 3.0),
id: state.ids.skill_general_swim_1,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_sword_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
// Title text
let tree_title = self.localized_strings.get("common.weapons.sword");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 5;
let skills_top_r = 7;
let skills_bot_l = 5;
let skills_bot_r = 1;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
// Skill icons and buttons
use skills::SwordSkill::*;
// Sword
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_sword".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.sword_render, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Descriptive {
title: "hud.skill.sw_trip_str_title",
desc: "hud.skill.sw_trip_str",
image: self.imgs.twohsword_m1,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_sword_combo_0,
},
SkillIcon::Unlockable {
skill: Skill::Sword(TsCombo),
image: self.imgs.physical_combo_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_sword_combo_1,
},
SkillIcon::Unlockable {
skill: Skill::Sword(TsDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[2], 3.0),
id: state.ids.skill_sword_combo_2,
},
SkillIcon::Unlockable {
skill: Skill::Sword(TsSpeed),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[3], 3.0),
id: state.ids.skill_sword_combo_3,
},
SkillIcon::Unlockable {
skill: Skill::Sword(TsRegen),
image: self.imgs.physical_energy_regen_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[4], 3.0),
id: state.ids.skill_sword_combo_4,
},
// Top right skills
SkillIcon::Descriptive {
title: "hud.skill.sw_dash_title",
desc: "hud.skill.sw_dash",
image: self.imgs.twohsword_m2,
position: MidTopWithMarginOn(state.ids.skills_top_r[0], 3.0),
id: state.ids.skill_sword_dash_0,
},
SkillIcon::Unlockable {
skill: Skill::Sword(DDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[1], 3.0),
id: state.ids.skill_sword_dash_1,
},
SkillIcon::Unlockable {
skill: Skill::Sword(DDrain),
image: self.imgs.physical_energy_drain_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[2], 3.0),
id: state.ids.skill_sword_dash_2,
},
SkillIcon::Unlockable {
skill: Skill::Sword(DCost),
image: self.imgs.physical_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[3], 3.0),
id: state.ids.skill_sword_dash_3,
},
SkillIcon::Unlockable {
skill: Skill::Sword(DSpeed),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[4], 3.0),
id: state.ids.skill_sword_dash_4,
},
SkillIcon::Unlockable {
skill: Skill::Sword(DChargeThrough),
image: self.imgs.physical_distance_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[5], 3.0),
id: state.ids.skill_sword_dash_5,
},
SkillIcon::Unlockable {
skill: Skill::Sword(DScaling),
image: self.imgs.physical_amount_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[6], 3.0),
id: state.ids.skill_sword_dash_6,
},
// Bottom left skills
SkillIcon::Unlockable {
skill: Skill::Sword(UnlockSpin),
image: self.imgs.sword_whirlwind,
position: MidTopWithMarginOn(state.ids.skills_bot_l[0], 3.0),
id: state.ids.skill_sword_spin_0,
},
SkillIcon::Unlockable {
skill: Skill::Sword(SDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[1], 3.0),
id: state.ids.skill_sword_spin_1,
},
SkillIcon::Unlockable {
skill: Skill::Sword(SSpeed),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[2], 3.0),
id: state.ids.skill_sword_spin_2,
},
SkillIcon::Unlockable {
skill: Skill::Sword(SCost),
image: self.imgs.physical_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[3], 3.0),
id: state.ids.skill_sword_spin_3,
},
SkillIcon::Unlockable {
skill: Skill::Sword(SSpins),
image: self.imgs.physical_amount_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[4], 3.0),
id: state.ids.skill_sword_spin_4,
},
// Bottom right skills
SkillIcon::Unlockable {
skill: Skill::Sword(InterruptingAttacks),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_r[0], 3.0),
id: state.ids.skill_sword_passive_0,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_hammer_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
// Title text
let tree_title = self.localized_strings.get("common.weapons.hammer");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 5;
let skills_top_r = 5;
let skills_bot_l = 6;
let skills_bot_r = 0;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
// Skill icons and buttons
use skills::HammerSkill::*;
// Hammer
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_hammer".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.hammer_render, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Descriptive {
title: "hud.skill.hmr_single_strike_title",
desc: "hud.skill.hmr_single_strike",
image: self.imgs.twohhammer_m1,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_hammer_combo_0,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(SsKnockback),
image: self.imgs.physical_knockback_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_hammer_combo_1,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(SsDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[2], 3.0),
id: state.ids.skill_hammer_combo_2,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(SsSpeed),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[3], 3.0),
id: state.ids.skill_hammer_combo_3,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(SsRegen),
image: self.imgs.physical_energy_regen_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[4], 3.0),
id: state.ids.skill_hammer_combo_4,
},
// Top right skills
SkillIcon::Descriptive {
title: "hud.skill.hmr_charged_melee_title",
desc: "hud.skill.hmr_charged_melee",
image: self.imgs.hammergolf,
position: MidTopWithMarginOn(state.ids.skills_top_r[0], 3.0),
id: state.ids.skill_hammer_charged_0,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(CKnockback),
image: self.imgs.physical_knockback_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[1], 3.0),
id: state.ids.skill_hammer_charged_1,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(CDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[2], 3.0),
id: state.ids.skill_hammer_charged_2,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(CDrain),
image: self.imgs.physical_energy_drain_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[3], 3.0),
id: state.ids.skill_hammer_charged_3,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(CSpeed),
image: self.imgs.physical_amount_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[4], 3.0),
id: state.ids.skill_hammer_charged_4,
},
// Bottom left skills
SkillIcon::Unlockable {
skill: Skill::Hammer(UnlockLeap),
image: self.imgs.hammerleap,
position: MidTopWithMarginOn(state.ids.skills_bot_l[0], 3.0),
id: state.ids.skill_hammer_leap_0,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(LDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[1], 3.0),
id: state.ids.skill_hammer_leap_1,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(LKnockback),
image: self.imgs.physical_knockback_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[2], 3.0),
id: state.ids.skill_hammer_leap_2,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(LCost),
image: self.imgs.physical_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[3], 3.0),
id: state.ids.skill_hammer_leap_3,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(LDistance),
image: self.imgs.physical_distance_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[4], 3.0),
id: state.ids.skill_hammer_leap_4,
},
SkillIcon::Unlockable {
skill: Skill::Hammer(LRange),
image: self.imgs.physical_radius_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[5], 3.0),
id: state.ids.skill_hammer_leap_5,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_axe_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
// Title text
let tree_title = self.localized_strings.get("common.weapons.axe");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 5;
let skills_top_r = 6;
let skills_bot_l = 5;
let skills_bot_r = 0;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
// Skill icons and buttons
use skills::AxeSkill::*;
// Axe
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_axe".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.axe_render, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Descriptive {
title: "hud.skill.axe_double_strike_title",
desc: "hud.skill.axe_double_strike",
image: self.imgs.twohaxe_m1,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_axe_combo_0,
},
SkillIcon::Unlockable {
skill: Skill::Axe(DsCombo),
image: self.imgs.physical_combo_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_axe_combo_1,
},
SkillIcon::Unlockable {
skill: Skill::Axe(DsDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[2], 3.0),
id: state.ids.skill_axe_combo_2,
},
SkillIcon::Unlockable {
skill: Skill::Axe(DsSpeed),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[3], 3.0),
id: state.ids.skill_axe_combo_3,
},
SkillIcon::Unlockable {
skill: Skill::Axe(DsRegen),
image: self.imgs.physical_energy_regen_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[4], 3.0),
id: state.ids.skill_axe_combo_4,
},
// Top right skills
SkillIcon::Descriptive {
title: "hud.skill.axe_spin_title",
desc: "hud.skill.axe_spin",
image: self.imgs.axespin,
position: MidTopWithMarginOn(state.ids.skills_top_r[0], 3.0),
id: state.ids.skill_axe_spin_0,
},
SkillIcon::Unlockable {
skill: Skill::Axe(SInfinite),
image: self.imgs.physical_infinite_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[1], 3.0),
id: state.ids.skill_axe_spin_1,
},
SkillIcon::Unlockable {
skill: Skill::Axe(SDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[2], 3.0),
id: state.ids.skill_axe_spin_2,
},
SkillIcon::Unlockable {
skill: Skill::Axe(SHelicopter),
image: self.imgs.physical_helicopter_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[3], 3.0),
id: state.ids.skill_axe_spin_3,
},
SkillIcon::Unlockable {
skill: Skill::Axe(SSpeed),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[4], 3.0),
id: state.ids.skill_axe_spin_4,
},
SkillIcon::Unlockable {
skill: Skill::Axe(SCost),
image: self.imgs.physical_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[5], 3.0),
id: state.ids.skill_axe_spin_5,
},
// Bottom left skills
SkillIcon::Unlockable {
skill: Skill::Axe(UnlockLeap),
image: self.imgs.skill_axe_leap_slash,
position: MidTopWithMarginOn(state.ids.skills_bot_l[0], 3.0),
id: state.ids.skill_axe_leap_0,
},
SkillIcon::Unlockable {
skill: Skill::Axe(LDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[1], 3.0),
id: state.ids.skill_axe_leap_1,
},
SkillIcon::Unlockable {
skill: Skill::Axe(LKnockback),
image: self.imgs.physical_knockback_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[2], 3.0),
id: state.ids.skill_axe_leap_2,
},
SkillIcon::Unlockable {
skill: Skill::Axe(LCost),
image: self.imgs.physical_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[3], 3.0),
id: state.ids.skill_axe_leap_3,
},
SkillIcon::Unlockable {
skill: Skill::Axe(LDistance),
image: self.imgs.physical_distance_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[4], 3.0),
id: state.ids.skill_axe_leap_4,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_sceptre_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
// Title text
let tree_title = self.localized_strings.get("common.weapons.sceptre");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 5;
let skills_top_r = 5;
let skills_bot_l = 5;
let skills_bot_r = 0;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
// Skill icons and buttons
use skills::SceptreSkill::*;
// Sceptre
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_sceptre".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.sceptre_render, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Descriptive {
title: "hud.skill.sc_lifesteal_title",
desc: "hud.skill.sc_lifesteal",
image: self.imgs.skill_sceptre_lifesteal,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_sceptre_lifesteal_0,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(LDamage),
image: self.imgs.magic_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_sceptre_lifesteal_1,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(LRange),
image: self.imgs.magic_distance_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[2], 3.0),
id: state.ids.skill_sceptre_lifesteal_2,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(LLifesteal),
image: self.imgs.magic_lifesteal_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[3], 3.0),
id: state.ids.skill_sceptre_lifesteal_3,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(LRegen),
image: self.imgs.magic_energy_regen_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[4], 3.0),
id: state.ids.skill_sceptre_lifesteal_4,
},
// Top right skills
SkillIcon::Descriptive {
title: "hud.skill.sc_heal_title",
desc: "hud.skill.sc_heal",
image: self.imgs.skill_sceptre_heal,
position: MidTopWithMarginOn(state.ids.skills_top_r[0], 3.0),
id: state.ids.skill_sceptre_heal_0,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(HHeal),
image: self.imgs.heal_heal_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[1], 3.0),
id: state.ids.skill_sceptre_heal_1,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(HDuration),
image: self.imgs.heal_duration_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[2], 3.0),
id: state.ids.skill_sceptre_heal_2,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(HRange),
image: self.imgs.heal_radius_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[3], 3.0),
id: state.ids.skill_sceptre_heal_3,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(HCost),
image: self.imgs.heal_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[4], 3.0),
id: state.ids.skill_sceptre_heal_4,
},
// Bottom left skills
SkillIcon::Unlockable {
skill: Skill::Sceptre(UnlockAura),
image: self.imgs.skill_sceptre_aura,
position: MidTopWithMarginOn(state.ids.skills_bot_l[0], 3.0),
id: state.ids.skill_sceptre_aura_0,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(AStrength),
image: self.imgs.buff_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[1], 3.0),
id: state.ids.skill_sceptre_aura_1,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(ADuration),
image: self.imgs.buff_duration_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[2], 3.0),
id: state.ids.skill_sceptre_aura_2,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(ARange),
image: self.imgs.buff_radius_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[3], 3.0),
id: state.ids.skill_sceptre_aura_3,
},
SkillIcon::Unlockable {
skill: Skill::Sceptre(ACost),
image: self.imgs.buff_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[4], 3.0),
id: state.ids.skill_sceptre_aura_4,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_bow_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
// Title text
let tree_title = self.localized_strings.get("common.weapons.bow");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 6;
let skills_top_r = 4;
let skills_bot_l = 5;
let skills_bot_r = 1;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
// Skill icons and buttons
use skills::BowSkill::*;
// Bow
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_bow".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.set(state.ids.bow_render, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Descriptive {
title: "hud.skill.bow_charged_title",
desc: "hud.skill.bow_charged",
image: self.imgs.bow_m1,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_bow_charged_0,
},
SkillIcon::Unlockable {
skill: Skill::Bow(CDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_bow_charged_1,
},
SkillIcon::Unlockable {
skill: Skill::Bow(CRegen),
image: self.imgs.physical_energy_regen_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[2], 3.0),
id: state.ids.skill_bow_charged_2,
},
SkillIcon::Unlockable {
skill: Skill::Bow(CKnockback),
image: self.imgs.physical_knockback_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[3], 3.0),
id: state.ids.skill_bow_charged_3,
},
SkillIcon::Unlockable {
skill: Skill::Bow(CSpeed),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[4], 3.0),
id: state.ids.skill_bow_charged_4,
},
SkillIcon::Unlockable {
skill: Skill::Bow(CMove),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[5], 3.0),
id: state.ids.skill_bow_charged_5,
},
// Top right skills
SkillIcon::Descriptive {
title: "hud.skill.bow_repeater_title",
desc: "hud.skill.bow_repeater",
image: self.imgs.bow_m2,
position: MidTopWithMarginOn(state.ids.skills_top_r[0], 3.0),
id: state.ids.skill_bow_repeater_0,
},
SkillIcon::Unlockable {
skill: Skill::Bow(RDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[1], 3.0),
id: state.ids.skill_bow_repeater_1,
},
SkillIcon::Unlockable {
skill: Skill::Bow(RCost),
image: self.imgs.physical_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[2], 3.0),
id: state.ids.skill_bow_repeater_2,
},
SkillIcon::Unlockable {
skill: Skill::Bow(RSpeed),
image: self.imgs.physical_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[3], 3.0),
id: state.ids.skill_bow_repeater_3,
},
// Bottom left skills
SkillIcon::Unlockable {
skill: Skill::Bow(UnlockShotgun),
image: self.imgs.skill_bow_jump_burst,
position: MidTopWithMarginOn(state.ids.skills_bot_l[0], 3.0),
id: state.ids.skill_bow_shotgun_0,
},
SkillIcon::Unlockable {
skill: Skill::Bow(SDamage),
image: self.imgs.physical_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[1], 3.0),
id: state.ids.skill_bow_shotgun_1,
},
SkillIcon::Unlockable {
skill: Skill::Bow(SCost),
image: self.imgs.physical_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[2], 3.0),
id: state.ids.skill_bow_shotgun_2,
},
SkillIcon::Unlockable {
skill: Skill::Bow(SArrows),
image: self.imgs.physical_amount_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[3], 3.0),
id: state.ids.skill_bow_shotgun_3,
},
SkillIcon::Unlockable {
skill: Skill::Bow(SSpread),
image: self.imgs.physical_explosion_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[4], 3.0),
id: state.ids.skill_bow_shotgun_4,
},
// Bottom right skills
SkillIcon::Unlockable {
skill: Skill::Bow(ProjSpeed),
image: self.imgs.physical_projectile_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_r[0], 3.0),
id: state.ids.skill_bow_passive_0,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_staff_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
// Title text
let tree_title = self.localized_strings.get("common.weapons.staff");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 4;
let skills_top_r = 5;
let skills_bot_l = 5;
let skills_bot_r = 0;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
// Skill icons and buttons
use skills::StaffSkill::*;
// Staff
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_staff_fire".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.staff_render, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Descriptive {
title: "hud.skill.st_fireball_title",
desc: "hud.skill.st_fireball",
image: self.imgs.fireball,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_staff_basic_0,
},
SkillIcon::Unlockable {
skill: Skill::Staff(BDamage),
image: self.imgs.magic_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_staff_basic_1,
},
SkillIcon::Unlockable {
skill: Skill::Staff(BRegen),
image: self.imgs.magic_energy_regen_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[2], 3.0),
id: state.ids.skill_staff_basic_2,
},
SkillIcon::Unlockable {
skill: Skill::Staff(BRadius),
image: self.imgs.magic_radius_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[3], 3.0),
id: state.ids.skill_staff_basic_3,
},
// Top right skills
SkillIcon::Descriptive {
title: "hud.skill.st_flamethrower_title",
desc: "hud.skill.st_flamethrower",
image: self.imgs.flamethrower,
position: MidTopWithMarginOn(state.ids.skills_top_r[0], 3.0),
id: state.ids.skill_staff_beam_0,
},
SkillIcon::Unlockable {
skill: Skill::Staff(FDamage),
image: self.imgs.magic_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[1], 3.0),
id: state.ids.skill_staff_beam_1,
},
SkillIcon::Unlockable {
skill: Skill::Staff(FDrain),
image: self.imgs.magic_energy_drain_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[2], 3.0),
id: state.ids.skill_staff_beam_2,
},
SkillIcon::Unlockable {
skill: Skill::Staff(FRange),
image: self.imgs.magic_radius_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[3], 3.0),
id: state.ids.skill_staff_beam_3,
},
SkillIcon::Unlockable {
skill: Skill::Staff(FVelocity),
image: self.imgs.magic_projectile_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_r[4], 3.0),
id: state.ids.skill_staff_beam_4,
},
// Bottom left skills
SkillIcon::Unlockable {
skill: Skill::Staff(UnlockShockwave),
image: self.imgs.fire_aoe,
position: MidTopWithMarginOn(state.ids.skills_bot_l[0], 3.0),
id: state.ids.skill_staff_shockwave_0,
},
SkillIcon::Unlockable {
skill: Skill::Staff(SDamage),
image: self.imgs.magic_damage_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[1], 3.0),
id: state.ids.skill_staff_shockwave_1,
},
SkillIcon::Unlockable {
skill: Skill::Staff(SKnockback),
image: self.imgs.magic_knockback_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[2], 3.0),
id: state.ids.skill_staff_shockwave_2,
},
SkillIcon::Unlockable {
skill: Skill::Staff(SCost),
image: self.imgs.magic_cost_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[3], 3.0),
id: state.ids.skill_staff_shockwave_3,
},
SkillIcon::Unlockable {
skill: Skill::Staff(SRange),
image: self.imgs.magic_radius_skill,
position: MidTopWithMarginOn(state.ids.skills_bot_l[4], 3.0),
id: state.ids.skill_staff_shockwave_4,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_mining_skills_window(
&mut self,
diary_tooltip: &Tooltip,
state: &mut State<DiaryState>,
ui: &mut UiCell,
mut events: Vec<Event>,
) -> Vec<Event> {
// Title text
let tree_title = self.localized_strings.get("common.tool.mining");
Text::new(tree_title)
.mid_top_with_margin_on(state.ids.content_align, 2.0)
.font_id(self.fonts.cyri.conrod_id)
.font_size(self.fonts.cyri.scale(34))
.color(TEXT_COLOR)
.set(state.ids.tree_title_txt, ui);
// Number of skills per rectangle per weapon, start counting at 0
// Maximum of 9 skills/8 indices
let skills_top_l = 4;
let skills_top_r = 0;
let skills_bot_l = 0;
let skills_bot_r = 0;
self.setup_state_for_skill_icons(
state,
ui,
skills_top_l,
skills_top_r,
skills_bot_l,
skills_bot_r,
);
// Skill icons and buttons
use skills::MiningSkill::*;
// Mining
Image::new(animate_by_pulse(
&self
.item_imgs
.img_ids_or_not_found_img(Tool("example_pick".to_string())),
self.pulse,
))
.wh(ART_SIZE)
.middle_of(state.ids.content_align)
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
.set(state.ids.pick_render, ui);
use PositionSpecifier::MidTopWithMarginOn;
let skill_buttons = &[
// Top Left skills
// 5 1 6
// 3 0 4
// 8 2 7
SkillIcon::Descriptive {
title: "hud.skill.pick_strike_title",
desc: "hud.skill.pick_strike",
image: self.imgs.pickaxe,
position: MidTopWithMarginOn(state.ids.skills_top_l[0], 3.0),
id: state.ids.skill_pick_m1,
},
SkillIcon::Unlockable {
skill: Skill::Pick(Speed),
image: self.imgs.pickaxe_speed_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[1], 3.0),
id: state.ids.skill_pick_m1_0,
},
SkillIcon::Unlockable {
skill: Skill::Pick(OreGain),
image: self.imgs.pickaxe_oregain_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[2], 3.0),
id: state.ids.skill_pick_m1_1,
},
SkillIcon::Unlockable {
skill: Skill::Pick(GemGain),
image: self.imgs.pickaxe_gemgain_skill,
position: MidTopWithMarginOn(state.ids.skills_top_l[3], 3.0),
id: state.ids.skill_pick_m1_2,
},
];
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip);
events
}
fn handle_skill_buttons(
&mut self,
icons: &[SkillIcon],
ui: &mut UiCell,
events: &mut Vec<Event>,
diary_tooltip: &Tooltip,
) {
for icon in icons {
match icon {
SkillIcon::Descriptive {
title,
desc,
image,
position,
id,
} => {
// TODO: shouldn't this be a `Image::new`?
Button::image(*image)
.w_h(74.0, 74.0)
.position(*position)
.with_tooltip(
self.tooltip_manager,
self.localized_strings.get(title),
self.localized_strings.get(desc),
diary_tooltip,
TEXT_COLOR,
)
.set(*id, ui);
},
SkillIcon::Unlockable {
skill,
image,
position,
id,
} => self.create_unlock_skill_button(
*skill,
*image,
*position,
*id,
ui,
events,
diary_tooltip,
),
}
}
}
fn setup_state_for_skill_icons(
&mut self,
state: &mut State<DiaryState>,
ui: &mut UiCell,
skills_top_l: usize,
skills_top_r: usize,
skills_bot_l: usize,
skills_bot_r: usize,
) {
// Update widget id array len
state.update(|s| {
s.ids
.skills_top_l
.resize(skills_top_l, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.skills_top_r
.resize(skills_top_r, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.skills_bot_l
.resize(skills_bot_l, &mut ui.widget_id_generator())
});
state.update(|s| {
s.ids
.skills_bot_r
.resize(skills_bot_r, &mut ui.widget_id_generator())
});
// Create Background Images to place skill icons on them later
// Create central skill first, others around it:
//
// 5 1 6
// 3 0 4
// 8 2 7
//
//
let offset_0 = 22.0;
let offset_1 = -122.0;
let offset_2 = offset_1 - -20.0;
let skill_pos = |idx, align, central_skill| {
use PositionSpecifier::*;
match idx {
// Central skill
0 => MiddleOf(align),
// 12:00
1 => UpFrom(central_skill, offset_0),
// 6:00
2 => DownFrom(central_skill, offset_0),
// 3:00
3 => LeftFrom(central_skill, offset_0),
// 9:00
4 => RightFrom(central_skill, offset_0),
// 10:30
5 => TopLeftWithMarginsOn(central_skill, offset_1, offset_2),
// 1:30
6 => TopRightWithMarginsOn(central_skill, offset_1, offset_2),
// 4:30
7 => BottomLeftWithMarginsOn(central_skill, offset_1, offset_2),
// 7:30
8 => BottomRightWithMarginsOn(central_skill, offset_1, offset_2),
buttons => {
panic!("{} > 8 position number", buttons);
},
}
};
// TOP-LEFT Skills
//
// TODO: Why this uses while loop on field of struct and not just
// `for i in 0..skils_top_l`?
while self.created_btns_top_l < skills_top_l {
let pos = skill_pos(
self.created_btns_top_l,
state.ids.skills_top_l_align,
state.ids.skills_top_l[0],
);
Button::image(self.imgs.wpn_icon_border_skills)
.w_h(80.0, 100.0)
.position(pos)
.set(state.ids.skills_top_l[self.created_btns_top_l], ui);
self.created_btns_top_l += 1;
}
// TOP-RIGHT Skills
while self.created_btns_top_r < skills_top_r {
let pos = skill_pos(
self.created_btns_top_r,
state.ids.skills_top_r_align,
state.ids.skills_top_r[0],
);
Button::image(self.imgs.wpn_icon_border_skills)
.w_h(80.0, 100.0)
.position(pos)
.set(state.ids.skills_top_r[self.created_btns_top_r], ui);
self.created_btns_top_r += 1;
}
// BOTTOM-LEFT Skills
while self.created_btns_bot_l < skills_bot_l {
let pos = skill_pos(
self.created_btns_bot_l,
state.ids.skills_bot_l_align,
state.ids.skills_bot_l[0],
);
Button::image(self.imgs.wpn_icon_border_skills)
.w_h(80.0, 100.0)
.position(pos)
.set(state.ids.skills_bot_l[self.created_btns_bot_l], ui);
self.created_btns_bot_l += 1;
}
// BOTTOM-RIGHT Skills
while self.created_btns_bot_r < skills_bot_r {
let pos = skill_pos(
self.created_btns_bot_r,
state.ids.skills_bot_r_align,
state.ids.skills_bot_r[0],
);
Button::image(self.imgs.wpn_icon_border_skills)
.w_h(80.0, 100.0)
.position(pos)
.set(state.ids.skills_bot_r[self.created_btns_bot_r], ui);
self.created_btns_bot_r += 1;
}
}
fn create_unlock_skill_button(
&mut self,
skill: Skill,
skill_image: image::Id,
position: PositionSpecifier,
widget_id: widget::Id,
ui: &mut UiCell,
events: &mut Vec<Event>,
diary_tooltip: &Tooltip,
) {
let label = if self.skill_set.prerequisites_met(skill) {
let current = self.skill_set.skill_level(skill).unwrap_or(0);
let max = skill.max_level();
format!("{}/{}", current, max)
} else {
"".to_owned()
};
let label_color = if self.skill_set.is_at_max_level(skill) {
TEXT_COLOR
} else if self.skill_set.sufficient_skill_points(skill) {
HP_COLOR
} else {
CRITICAL_HP_COLOR
};
let image_color = if self.skill_set.prerequisites_met(skill) {
TEXT_COLOR
} else {
Color::Rgba(0.41, 0.41, 0.41, 0.7)
};
let (title, description) = skill_strings(skill, self.localized_strings);
let description = if description.contains("{SP}") {
Cow::Owned(self.splice_skill_reqs(skill, &description))
} else {
description
};
let button = Button::image(skill_image)
.w_h(74.0, 74.0)
.position(position)
.label(&label)
.label_y(conrod_core::position::Relative::Scalar(-47.0))
.label_x(conrod_core::position::Relative::Scalar(0.0))
.label_color(label_color)
.label_font_size(self.fonts.cyri.scale(15))
.label_font_id(self.fonts.cyri.conrod_id)
.image_color(image_color)
.with_tooltip(
self.tooltip_manager,
title,
&description,
diary_tooltip,
TEXT_COLOR,
)
.set(widget_id, ui);
if button.was_clicked() {
events.push(Event::UnlockSkill(skill));
};
}
fn splice_skill_reqs(&self, skill: Skill, desc: &str) -> String {
let current_level = self.skill_set.skill_level(skill);
if matches!(current_level, Ok(level) if level == skill.max_level()) {
desc.replace("{SP}", "")
} else {
let req_sp_text = self.localized_strings.get("hud.skill.req_sp");
let skill_cost_text = self.skill_set.skill_cost(skill).to_string();
desc.replace("{SP}", &req_sp_text.replace("{number}", &skill_cost_text))
}
}
}
/// Returns skill info as a tuple of title and description.
///
/// Title is ready to use, description may include `"{SP}"` placeholder you
/// will want to handle yourself.
pub fn skill_strings(skill: Skill, i18n: &Localization) -> (&str, Cow<str>) {
match skill {
// general tree
Skill::General(s) => general_skill_strings(s, i18n),
Skill::UnlockGroup(s) => unlock_skill_strings(s, i18n),
// weapon trees
Skill::Sword(s) => sword_skill_strings(s, i18n),
Skill::Axe(s) => axe_skill_strings(s, i18n),
Skill::Hammer(s) => hammer_skill_strings(s, i18n),
Skill::Bow(s) => bow_skill_strings(s, i18n),
Skill::Staff(s) => staff_skill_strings(s, i18n),
Skill::Sceptre(s) => sceptre_skill_strings(s, i18n),
// movement trees
Skill::Roll(s) => roll_skill_strings(s, i18n),
Skill::Climb(s) => climb_skill_strings(s, i18n),
Skill::Swim(s) => swim_skill_strings(s, i18n),
// mining
Skill::Pick(s) => mining_skill_strings(s, i18n),
}
}
fn general_skill_strings(skill: GeneralSkill, i18n: &Localization) -> (&str, Cow<str>) {
match skill {
GeneralSkill::HealthIncrease => splice_constant(
i18n,
"hud.skill.inc_health_title",
"hud.skill.inc_health",
u32::from(HP_PER_LEVEL),
),
GeneralSkill::EnergyIncrease => splice_constant(
i18n,
"hud.skill.inc_energy_title",
"hud.skill.inc_energy",
u32::from(ENERGY_PER_LEVEL),
),
}
}
fn unlock_skill_strings(group: SkillGroupKind, i18n: &Localization) -> (&str, Cow<str>) {
match group {
SkillGroupKind::Weapon(ToolKind::Sword) => {
localize(i18n, "hud.skill.unlck_sword_title", "hud.skill.unlck_sword")
},
SkillGroupKind::Weapon(ToolKind::Axe) => {
localize(i18n, "hud.skill.unlck_axe_title", "hud.skill.unlck_axe")
},
SkillGroupKind::Weapon(ToolKind::Hammer) => localize(
i18n,
"hud.skill.unlck_hammer_title",
"hud.skill.unlck_hammer",
),
SkillGroupKind::Weapon(ToolKind::Bow) => {
localize(i18n, "hud.skill.unlck_bow_title", "hud.skill.unlck_bow")
},
SkillGroupKind::Weapon(ToolKind::Staff) => {
localize(i18n, "hud.skill.unlck_staff_title", "hud.skill.unlck_staff")
},
SkillGroupKind::Weapon(ToolKind::Sceptre) => localize(
i18n,
"hud.skill.unlck_sceptre_title",
"hud.skill.unlck_sceptre",
),
SkillGroupKind::General
| SkillGroupKind::Weapon(
ToolKind::Dagger
| ToolKind::Shield
| ToolKind::Spear
| ToolKind::Blowgun
| ToolKind::Debug
| ToolKind::Farming
| ToolKind::Pick
| ToolKind::Natural
| ToolKind::Empty,
) => {
tracing::warn!("Requesting title for unlocking unexpected skill group");
("", Cow::Owned(String::new()))
},
}
}
fn sword_skill_strings(skill: SwordSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.sword_tree;
match skill {
// triple strike
SwordSkill::TsCombo => localize(
i18n,
"hud.skill.sw_trip_str_combo_title",
"hud.skill.sw_trip_str_combo",
),
SwordSkill::TsDamage => localize(
i18n,
"hud.skill.sw_trip_str_dmg_title",
"hud.skill.sw_trip_str_dmg",
),
SwordSkill::TsSpeed => localize(
i18n,
"hud.skill.sw_trip_str_sp_title",
"hud.skill.sw_trip_str_sp",
),
SwordSkill::TsRegen => localize(
i18n,
"hud.skill.sw_trip_str_reg_title",
"hud.skill.sw_trip_str_reg",
),
// dash
SwordSkill::DDamage => splice_multiplier(
i18n,
"hud.skill.sw_dash_dmg_title",
"hud.skill.sw_dash_dmg",
modifiers.dash.base_damage,
),
SwordSkill::DDrain => splice_multiplier(
i18n,
"hud.skill.sw_dash_drain_title",
"hud.skill.sw_dash_drain",
modifiers.dash.energy_drain,
),
SwordSkill::DCost => splice_multiplier(
i18n,
"hud.skill.sw_dash_cost_title",
"hud.skill.sw_dash_cost",
modifiers.dash.energy_cost,
),
SwordSkill::DSpeed => splice_multiplier(
i18n,
"hud.skill.sw_dash_speed_title",
"hud.skill.sw_dash_speed",
modifiers.dash.forward_speed,
),
SwordSkill::DChargeThrough => localize(
i18n,
"hud.skill.sw_dash_charge_through_title",
"hud.skill.sw_dash_charge_through",
),
SwordSkill::DScaling => splice_multiplier(
i18n,
"hud.skill.sw_dash_scale_title",
"hud.skill.sw_dash_scale",
modifiers.dash.scaled_damage,
),
// spin
SwordSkill::UnlockSpin => localize(i18n, "hud.skill.sw_spin_title", "hud.skill.sw_spin"),
SwordSkill::SDamage => splice_multiplier(
i18n,
"hud.skill.sw_spin_dmg_title",
"hud.skill.sw_spin_dmg",
modifiers.spin.base_damage,
),
SwordSkill::SSpeed => splice_multiplier(
i18n,
"hud.skill.sw_spin_spd_title",
"hud.skill.sw_spin_spd",
modifiers.spin.swing_duration,
),
SwordSkill::SCost => splice_multiplier(
i18n,
"hud.skill.sw_spin_cost_title",
"hud.skill.sw_spin_cost",
modifiers.spin.energy_cost,
),
SwordSkill::SSpins => splice_constant(
i18n,
"hud.skill.sw_spin_spins_title",
"hud.skill.sw_spin_spins",
modifiers.spin.num,
),
// independent skills
SwordSkill::InterruptingAttacks => localize(
i18n,
"hud.skill.sw_interrupt_title",
"hud.skill.sw_interrupt",
),
}
}
fn axe_skill_strings(skill: AxeSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.axe_tree;
match skill {
// Double strike upgrades
AxeSkill::DsCombo => localize(
i18n,
"hud.skill.axe_double_strike_combo_title",
"hud.skill.axe_double_strike_combo",
),
AxeSkill::DsDamage => localize(
i18n,
"hud.skill.axe_double_strike_damage_title",
"hud.skill.axe_double_strike_damage",
),
AxeSkill::DsSpeed => localize(
i18n,
"hud.skill.axe_double_strike_speed_title",
"hud.skill.axe_double_strike_speed",
),
AxeSkill::DsRegen => localize(
i18n,
"hud.skill.axe_double_strike_regen_title",
"hud.skill.axe_double_strike_regen",
),
// Spin upgrades
AxeSkill::SInfinite => localize(
i18n,
"hud.skill.axe_infinite_axe_spin_title",
"hud.skill.axe_infinite_axe_spin",
),
AxeSkill::SHelicopter => localize(
i18n,
"hud.skill.axe_spin_helicopter_title",
"hud.skill.axe_spin_helicopter",
),
AxeSkill::SDamage => splice_multiplier(
i18n,
"hud.skill.axe_spin_damage_title",
"hud.skill.axe_spin_damage",
modifiers.spin.base_damage,
),
AxeSkill::SSpeed => splice_multiplier(
i18n,
"hud.skill.axe_spin_speed_title",
"hud.skill.axe_spin_speed",
modifiers.spin.swing_duration,
),
AxeSkill::SCost => splice_multiplier(
i18n,
"hud.skill.axe_spin_cost_title",
"hud.skill.axe_spin_cost",
modifiers.spin.energy_cost,
),
// Leap upgrades
AxeSkill::UnlockLeap => localize(
i18n,
"hud.skill.axe_unlock_leap_title",
"hud.skill.axe_unlock_leap",
),
AxeSkill::LDamage => splice_multiplier(
i18n,
"hud.skill.axe_leap_damage_title",
"hud.skill.axe_leap_damage",
modifiers.leap.base_damage,
),
AxeSkill::LKnockback => splice_multiplier(
i18n,
"hud.skill.axe_leap_knockback_title",
"hud.skill.axe_leap_knockback",
modifiers.leap.knockback,
),
AxeSkill::LCost => splice_multiplier(
i18n,
"hud.skill.axe_leap_cost_title",
"hud.skill.axe_leap_cost",
modifiers.leap.energy_cost,
),
AxeSkill::LDistance => splice_multiplier(
i18n,
"hud.skill.axe_leap_distance_title",
"hud.skill.axe_leap_distance",
modifiers.leap.leap_strength,
),
}
}
fn hammer_skill_strings(skill: HammerSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.hammer_tree;
// Single strike upgrades
match skill {
HammerSkill::SsKnockback => splice_multiplier(
i18n,
"hud.skill.hmr_single_strike_knockback_title",
"hud.skill.hmr_single_strike_knockback",
modifiers.single_strike.knockback,
),
HammerSkill::SsDamage => localize(
i18n,
"hud.skill.hmr_single_strike_damage_title",
"hud.skill.hmr_single_strike_damage",
),
HammerSkill::SsSpeed => localize(
i18n,
"hud.skill.hmr_single_strike_speed_title",
"hud.skill.hmr_single_strike_speed",
),
HammerSkill::SsRegen => localize(
i18n,
"hud.skill.hmr_single_strike_regen_title",
"hud.skill.hmr_single_strike_regen",
),
// Charged melee upgrades
HammerSkill::CDamage => splice_multiplier(
i18n,
"hud.skill.hmr_charged_melee_damage_title",
"hud.skill.hmr_charged_melee_damage",
modifiers.charged.scaled_damage,
),
HammerSkill::CKnockback => splice_multiplier(
i18n,
"hud.skill.hmr_charged_melee_knockback_title",
"hud.skill.hmr_charged_melee_knockback",
modifiers.charged.scaled_knockback,
),
HammerSkill::CDrain => splice_multiplier(
i18n,
"hud.skill.hmr_charged_melee_nrg_drain_title",
"hud.skill.hmr_charged_melee_nrg_drain",
modifiers.charged.energy_drain,
),
HammerSkill::CSpeed => splice_multiplier(
i18n,
"hud.skill.hmr_charged_rate_title",
"hud.skill.hmr_charged_rate",
modifiers.charged.charge_rate,
),
// Leap upgrades
HammerSkill::UnlockLeap => localize(
i18n,
"hud.skill.hmr_unlock_leap_title",
"hud.skill.hmr_unlock_leap",
),
HammerSkill::LDamage => splice_multiplier(
i18n,
"hud.skill.hmr_leap_damage_title",
"hud.skill.hmr_leap_damage",
modifiers.leap.base_damage,
),
HammerSkill::LCost => splice_multiplier(
i18n,
"hud.skill.hmr_leap_cost_title",
"hud.skill.hmr_leap_cost",
modifiers.leap.energy_cost,
),
HammerSkill::LDistance => splice_multiplier(
i18n,
"hud.skill.hmr_leap_distance_title",
"hud.skill.hmr_leap_distance",
modifiers.leap.leap_strength,
),
HammerSkill::LKnockback => splice_multiplier(
i18n,
"hud.skill.hmr_leap_knockback_title",
"hud.skill.hmr_leap_knockback",
modifiers.leap.knockback,
),
HammerSkill::LRange => {
let title = i18n.get("hud.skill.hmr_leap_radius_title");
let desc = i18n.get("hud.skill.hmr_leap_radius");
let desc = desc.replace("{boost}", &format!("{}", modifiers.leap.range));
(title, Cow::Owned(desc))
},
}
}
fn bow_skill_strings(skill: BowSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.bow_tree;
match skill {
// Passives
BowSkill::ProjSpeed => splice_multiplier(
i18n,
"hud.skill.bow_projectile_speed_title",
"hud.skill.bow_projectile_speed",
modifiers.universal.projectile_speed,
),
// Charged upgrades
BowSkill::CDamage => splice_multiplier(
i18n,
"hud.skill.bow_charged_damage_title",
"hud.skill.bow_charged_damage",
modifiers.charged.damage_scaling,
),
BowSkill::CRegen => splice_multiplier(
i18n,
"hud.skill.bow_charged_energy_regen_title",
"hud.skill.bow_charged_energy_regen",
modifiers.charged.regen_scaling,
),
BowSkill::CKnockback => splice_multiplier(
i18n,
"hud.skill.bow_charged_knockback_title",
"hud.skill.bow_charged_knockback",
modifiers.charged.knockback_scaling,
),
BowSkill::CSpeed => splice_multiplier(
i18n,
"hud.skill.bow_charged_speed_title",
"hud.skill.bow_charged_speed",
modifiers.charged.charge_rate,
),
BowSkill::CMove => splice_multiplier(
i18n,
"hud.skill.bow_charged_move_title",
"hud.skill.bow_charged_move",
modifiers.charged.move_speed,
),
// Repeater upgrades
BowSkill::RDamage => splice_multiplier(
i18n,
"hud.skill.bow_repeater_damage_title",
"hud.skill.bow_repeater_damage",
modifiers.repeater.power,
),
BowSkill::RCost => splice_multiplier(
i18n,
"hud.skill.bow_repeater_cost_title",
"hud.skill.bow_repeater_cost",
modifiers.repeater.energy_cost,
),
BowSkill::RSpeed => splice_multiplier(
i18n,
"hud.skill.bow_repeater_speed_title",
"hud.skill.bow_repeater_speed",
modifiers.repeater.max_speed,
),
// Shotgun upgrades
BowSkill::UnlockShotgun => localize(
i18n,
"hud.skill.bow_shotgun_unlock_title",
"hud.skill.bow_shotgun_unlock",
),
BowSkill::SDamage => splice_multiplier(
i18n,
"hud.skill.bow_shotgun_damage_title",
"hud.skill.bow_shotgun_damage",
modifiers.shotgun.power,
),
BowSkill::SCost => splice_multiplier(
i18n,
"hud.skill.bow_shotgun_cost_title",
"hud.skill.bow_shotgun_cost",
modifiers.shotgun.energy_cost,
),
BowSkill::SArrows => splice_constant(
i18n,
"hud.skill.bow_shotgun_arrow_count_title",
"hud.skill.bow_shotgun_arrow_count",
modifiers.shotgun.num_projectiles,
),
BowSkill::SSpread => splice_multiplier(
i18n,
"hud.skill.bow_shotgun_spread_title",
"hud.skill.bow_shotgun_spread",
modifiers.shotgun.spread,
),
}
}
fn staff_skill_strings(skill: StaffSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.staff_tree;
match skill {
// Basic ranged upgrades
StaffSkill::BDamage => splice_multiplier(
i18n,
"hud.skill.st_damage_title",
"hud.skill.st_damage",
modifiers.fireball.power,
),
StaffSkill::BRegen => splice_multiplier(
i18n,
"hud.skill.st_energy_regen_title",
"hud.skill.st_energy_regen",
modifiers.fireball.regen,
),
StaffSkill::BRadius => splice_multiplier(
i18n,
"hud.skill.st_explosion_radius_title",
"hud.skill.st_explosion_radius",
modifiers.fireball.range,
),
// Flamethrower upgrades
StaffSkill::FDamage => splice_multiplier(
i18n,
"hud.skill.st_flamethrower_damage_title",
"hud.skill.st_flamethrower_damage",
modifiers.flamethrower.damage,
),
StaffSkill::FRange => splice_multiplier(
i18n,
"hud.skill.st_flamethrower_range_title",
"hud.skill.st_flamethrower_range",
modifiers.flamethrower.range,
),
StaffSkill::FDrain => splice_multiplier(
i18n,
"hud.skill.st_energy_drain_title",
"hud.skill.st_energy_drain",
modifiers.flamethrower.energy_drain,
),
StaffSkill::FVelocity => splice_multiplier(
i18n,
"hud.skill.st_flame_velocity_title",
"hud.skill.st_flame_velocity",
modifiers.flamethrower.velocity,
),
// Shockwave upgrades
StaffSkill::UnlockShockwave => localize(
i18n,
"hud.skill.st_shockwave_unlock_title",
"hud.skill.st_shockwave_unlock",
),
StaffSkill::SDamage => splice_multiplier(
i18n,
"hud.skill.st_shockwave_damage_title",
"hud.skill.st_shockwave_damage",
modifiers.shockwave.damage,
),
StaffSkill::SKnockback => splice_multiplier(
i18n,
"hud.skill.st_shockwave_knockback_title",
"hud.skill.st_shockwave_knockback",
modifiers.shockwave.knockback,
),
StaffSkill::SRange => splice_multiplier(
i18n,
"hud.skill.st_shockwave_range_title",
"hud.skill.st_shockwave_range",
modifiers.shockwave.duration,
),
StaffSkill::SCost => splice_multiplier(
i18n,
"hud.skill.st_shockwave_cost_title",
"hud.skill.st_shockwave_cost",
modifiers.shockwave.energy_cost,
),
}
}
fn sceptre_skill_strings(skill: SceptreSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.sceptre_tree;
match skill {
// Lifesteal beam upgrades
SceptreSkill::LDamage => splice_multiplier(
i18n,
"hud.skill.sc_lifesteal_damage_title",
"hud.skill.sc_lifesteal_damage",
modifiers.beam.damage,
),
SceptreSkill::LRange => splice_multiplier(
i18n,
"hud.skill.sc_lifesteal_range_title",
"hud.skill.sc_lifesteal_range",
modifiers.beam.range,
),
SceptreSkill::LLifesteal => splice_multiplier(
i18n,
"hud.skill.sc_lifesteal_lifesteal_title",
"hud.skill.sc_lifesteal_lifesteal",
modifiers.beam.lifesteal,
),
SceptreSkill::LRegen => splice_multiplier(
i18n,
"hud.skill.sc_lifesteal_regen_title",
"hud.skill.sc_lifesteal_regen",
modifiers.beam.energy_regen,
),
// Healing aura upgrades
SceptreSkill::HHeal => splice_multiplier(
i18n,
"hud.skill.sc_heal_heal_title",
"hud.skill.sc_heal_heal",
modifiers.healing_aura.strength,
),
SceptreSkill::HRange => splice_multiplier(
i18n,
"hud.skill.sc_heal_range_title",
"hud.skill.sc_heal_range",
modifiers.healing_aura.range,
),
SceptreSkill::HDuration => splice_multiplier(
i18n,
"hud.skill.sc_heal_duration_title",
"hud.skill.sc_heal_duration",
modifiers.healing_aura.duration,
),
SceptreSkill::HCost => splice_multiplier(
i18n,
"hud.skill.sc_heal_cost_title",
"hud.skill.sc_heal_cost",
modifiers.healing_aura.energy_cost,
),
// Warding aura upgrades
SceptreSkill::UnlockAura => localize(
i18n,
"hud.skill.sc_wardaura_unlock_title",
"hud.skill.sc_wardaura_unlock",
),
SceptreSkill::AStrength => splice_multiplier(
i18n,
"hud.skill.sc_wardaura_strength_title",
"hud.skill.sc_wardaura_strength",
modifiers.warding_aura.strength,
),
SceptreSkill::ADuration => splice_multiplier(
i18n,
"hud.skill.sc_wardaura_duration_title",
"hud.skill.sc_wardaura_duration",
modifiers.warding_aura.duration,
),
SceptreSkill::ARange => splice_multiplier(
i18n,
"hud.skill.sc_wardaura_range_title",
"hud.skill.sc_wardaura_range",
modifiers.warding_aura.range,
),
SceptreSkill::ACost => splice_multiplier(
i18n,
"hud.skill.sc_wardaura_cost_title",
"hud.skill.sc_wardaura_cost",
modifiers.warding_aura.energy_cost,
),
}
}
fn roll_skill_strings(skill: RollSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.general_tree.roll;
match skill {
RollSkill::Cost => splice_multiplier(
i18n,
"hud.skill.roll_energy_title",
"hud.skill.roll_energy",
modifiers.energy_cost,
),
RollSkill::Strength => splice_multiplier(
i18n,
"hud.skill.roll_speed_title",
"hud.skill.roll_speed",
modifiers.strength,
),
RollSkill::Duration => splice_multiplier(
i18n,
"hud.skill.roll_dur_title",
"hud.skill.roll_dur",
modifiers.duration,
),
}
}
fn climb_skill_strings(skill: ClimbSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.general_tree.climb;
match skill {
ClimbSkill::Cost => splice_multiplier(
i18n,
"hud.skill.climbing_cost_title",
"hud.skill.climbing_cost",
modifiers.energy_cost,
),
ClimbSkill::Speed => splice_multiplier(
i18n,
"hud.skill.climbing_speed_title",
"hud.skill.climbing_speed",
modifiers.speed,
),
}
}
fn swim_skill_strings(skill: SwimSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.general_tree.swim;
match skill {
SwimSkill::Speed => splice_multiplier(
i18n,
"hud.skill.swim_speed_title",
"hud.skill.swim_speed",
modifiers.speed,
),
}
}
fn mining_skill_strings(skill: MiningSkill, i18n: &Localization) -> (&str, Cow<str>) {
let modifiers = SKILL_MODIFIERS.mining_tree;
match skill {
MiningSkill::Speed => splice_multiplier(
i18n,
"hud.skill.pick_strike_speed_title",
"hud.skill.pick_strike_speed",
modifiers.speed,
),
MiningSkill::OreGain => splice_constant(
i18n,
"hud.skill.pick_strike_oregain_title",
"hud.skill.pick_strike_oregain",
(modifiers.ore_gain * 100.0).round() as u32,
),
MiningSkill::GemGain => splice_constant(
i18n,
"hud.skill.pick_strike_gemgain_title",
"hud.skill.pick_strike_gemgain",
(modifiers.gem_gain * 100.0).round() as u32,
),
}
}
/// Helper function which takes title i18n key and description i18n key
/// and returns localized title and localized description replacing "{boost}"
/// placeholder with passed constant.
// TODO: do something better when we get to multimodifier skills
fn splice_constant<'loc>(
i18n: &'loc Localization,
title: &'loc str,
desc: &str,
constant: u32,
) -> (&'loc str, Cow<'loc, str>) {
let title = i18n.get(title);
let desc = i18n.get(desc);
let desc = desc.replace("{boost}", &constant.to_string());
(title, Cow::Owned(desc))
}
/// Helper function which takes title i18n key and description i18n key
/// and returns localized title and localized description replacing "{boost}"
/// placeholder with absolute value of percentage effect of multiplier.
// TODO: do something better when we get to multimodifier skills
fn splice_multiplier<'loc>(
i18n: &'loc Localization,
title: &'loc str,
desc: &str,
multipler: f32,
) -> (&'loc str, Cow<'loc, str>) {
let percentage = hud::multiplier_to_percentage(multipler).abs();
let title = i18n.get(title);
let desc = i18n.get(desc);
let desc = desc.replace("{boost}", &format!("{:.0}", percentage));
(title, Cow::Owned(desc))
}
// Small helper function to get localized skill text.
fn localize<'loc>(
i18n: &'loc Localization,
title: &'loc str,
desc: &'loc str,
) -> (&'loc str, Cow<'loc, str>) {
let title = i18n.get(title);
let desc = i18n.get(desc);
(title, Cow::Borrowed(desc))
}