mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
3429 lines
131 KiB
Rust
3429 lines
131 KiB
Rust
use super::{
|
|
img_ids::{Imgs, ImgsRot},
|
|
item_imgs::{animate_by_pulse, ItemImgs},
|
|
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 client::{self, Client};
|
|
use common::{
|
|
combat,
|
|
comp::{
|
|
self,
|
|
ability::{Ability, ActiveAbilities, AuxiliaryAbility, BASE_ABILITY_LIMIT},
|
|
inventory::{
|
|
item::{
|
|
item_key::ItemKey,
|
|
tool::{AbilityContext, ToolKind},
|
|
ItemKind, MaterialStatManifest,
|
|
},
|
|
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 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 std::borrow::Cow;
|
|
use vek::*;
|
|
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,
|
|
active_bar_checkbox,
|
|
active_bar_checkbox_label,
|
|
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[],
|
|
skills[],
|
|
skill_lock_imgs[],
|
|
sword_bg,
|
|
axe_bg,
|
|
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,
|
|
sword_path_overlay,
|
|
// 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,
|
|
context: &'a AbilityContext,
|
|
|
|
#[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,
|
|
context: &'a AbilityContext,
|
|
) -> 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,
|
|
context,
|
|
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",
|
|
"Axe",
|
|
"Hammer",
|
|
"Bow",
|
|
"Fire Staff",
|
|
"Sceptre",
|
|
"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),
|
|
SelectExpBar(Option<SkillGroupKind>),
|
|
}
|
|
|
|
#[derive(PartialEq, Eq)]
|
|
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_msg("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,
|
|
"Axe" => self.imgs.axe,
|
|
"Hammer" => self.imgs.hammer,
|
|
"Bow" => self.imgs.bow,
|
|
"Fire Staff" => self.imgs.staff,
|
|
"Sceptre" => self.imgs.sceptre,
|
|
"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_msg("hud-skill-not_unlocked")
|
|
} else {
|
|
Cow::Borrowed("")
|
|
};
|
|
|
|
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);
|
|
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 as Exp bar below skillbar
|
|
let exp_selected =
|
|
self.global_state.settings.interface.xp_bar_skillgroup == Some(*sel_tab);
|
|
if Button::image(if !exp_selected {
|
|
self.imgs.checkbox
|
|
} else {
|
|
self.imgs.checkbox_checked
|
|
})
|
|
.w_h(18.0, 18.0)
|
|
.hover_image(if !exp_selected {
|
|
self.imgs.checkbox_mo
|
|
} else {
|
|
self.imgs.checkbox_checked_mo
|
|
})
|
|
.press_image(if !exp_selected {
|
|
self.imgs.checkbox_press
|
|
} else {
|
|
self.imgs.checkbox_checked
|
|
})
|
|
.top_right_with_margins_on(state.ids.exp_bar_frame, 50.0, -30.0)
|
|
.set(state.ids.active_bar_checkbox, ui)
|
|
.was_clicked()
|
|
{
|
|
if self.global_state.settings.interface.xp_bar_skillgroup != Some(*sel_tab) {
|
|
events.push(Event::SelectExpBar(Some(*sel_tab)));
|
|
} else {
|
|
events.push(Event::SelectExpBar(None));
|
|
}
|
|
}
|
|
|
|
Text::new(&self.localized_strings.get_msg("hud-skill-set_as_exp_bar"))
|
|
.right_from(state.ids.active_bar_checkbox, 10.0)
|
|
.font_size(self.fonts.cyri.scale(14))
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
.graphics_for(state.ids.active_bar_checkbox)
|
|
.color(TEXT_COLOR)
|
|
.set(state.ids.active_bar_checkbox_label, 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_msg_ctx(
|
|
"hud-skill-sp_available",
|
|
&i18n::fluent_args! {
|
|
"number" => available_pts,
|
|
},
|
|
))
|
|
.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::Axe) => {
|
|
self.handle_axe_skills_window(&diary_tooltip, state, ui, events)
|
|
},
|
|
SelectedSkillTree::Weapon(ToolKind::Hammer) => {
|
|
self.handle_hammer_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::Sceptre) => {
|
|
self.handle_sceptre_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(BASE_ABILITY_LIMIT, &mut ui.widget_id_generator())
|
|
});
|
|
state.update(|s| {
|
|
s.ids
|
|
.active_abilities_keys
|
|
.resize(BASE_ABILITY_LIMIT, &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,
|
|
self.context,
|
|
),
|
|
image_source: self.imgs,
|
|
slot_manager: Some(self.slot_manager),
|
|
pulse: 0.0,
|
|
};
|
|
|
|
for i in 0..BASE_ABILITY_LIMIT {
|
|
let ability_id = self
|
|
.active_abilities
|
|
.get_ability(
|
|
AbilityInput::Auxiliary(i),
|
|
Some(self.inventory),
|
|
Some(self.skill_set),
|
|
)
|
|
.ability_id(Some(self.inventory), Some(self.skill_set), self.context);
|
|
let (ability_title, ability_desc) = if let Some(ability_id) = ability_id {
|
|
util::ability_description(ability_id, self.localized_strings)
|
|
} else {
|
|
(
|
|
Cow::Borrowed("Drag an ability here to use it."),
|
|
Cow::Borrowed(""),
|
|
)
|
|
};
|
|
|
|
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_shortest(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_available_abilities(
|
|
Some(self.inventory),
|
|
Some(self.skill_set),
|
|
EquipSlot::ActiveMainhand,
|
|
)
|
|
.map(AuxiliaryAbility::MainWeapon)
|
|
.map(|a| {
|
|
(
|
|
Ability::from(a).ability_id(
|
|
Some(self.inventory),
|
|
Some(self.skill_set),
|
|
self.context,
|
|
),
|
|
a,
|
|
)
|
|
});
|
|
let off_weap_abilities = ActiveAbilities::iter_available_abilities(
|
|
Some(self.inventory),
|
|
Some(self.skill_set),
|
|
EquipSlot::ActiveOffhand,
|
|
)
|
|
.map(AuxiliaryAbility::OffWeapon)
|
|
.map(|a| {
|
|
(
|
|
Ability::from(a).ability_id(
|
|
Some(self.inventory),
|
|
Some(self.skill_set),
|
|
self.context,
|
|
),
|
|
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,
|
|
self.context,
|
|
),
|
|
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(""), self.localized_strings);
|
|
|
|
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(13))
|
|
.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; 12] = [
|
|
"Hitpoints",
|
|
"Energy",
|
|
"Poise",
|
|
"Combat-Rating",
|
|
"Protection",
|
|
"Stun-Resistance",
|
|
"Precision-Power",
|
|
"Energy Reward",
|
|
"Stealth",
|
|
"Weapon Power",
|
|
"Weapon Speed",
|
|
"Weapon Effect Power",
|
|
];
|
|
|
|
// 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(item.stats_durability_multiplier()))
|
|
},
|
|
_ => None,
|
|
});
|
|
|
|
let off_weap_stats = self
|
|
.inventory
|
|
.equipped(EquipSlot::ActiveOffhand)
|
|
.and_then(|item| match &*item.kind() {
|
|
ItemKind::Tool(tool) => {
|
|
Some(tool.stats(item.stats_durability_multiplier()))
|
|
},
|
|
_ => 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), self.msm);
|
|
match protection {
|
|
Some(prot) => format!("{}", prot),
|
|
None => String::from("Invincible"),
|
|
}
|
|
},
|
|
"Stun-Resistance" => {
|
|
let stun_res = Poise::compute_poise_damage_reduction(
|
|
Some(self.inventory),
|
|
self.msm,
|
|
None,
|
|
None,
|
|
);
|
|
format!("{:.2}%", stun_res * 100.0)
|
|
},
|
|
"Precision-Power" => {
|
|
let precision_power =
|
|
combat::compute_precision_mult(Some(self.inventory), self.msm);
|
|
format!("x{:.2}", precision_power)
|
|
},
|
|
"Energy Reward" => {
|
|
let energy_rew =
|
|
combat::compute_energy_reward_mod(Some(self.inventory), self.msm);
|
|
format!("{:+.0}%", (energy_rew - 1.0) * 100.0)
|
|
},
|
|
"Stealth" => {
|
|
let stealth_perception_multiplier =
|
|
combat::perception_dist_multiplier_from_stealth(
|
|
Some(self.inventory),
|
|
None,
|
|
self.msm,
|
|
);
|
|
let txt =
|
|
format!("{:+.1}%", (1.0 - stealth_perception_multiplier) * 100.0);
|
|
|
|
txt
|
|
},
|
|
"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 Effect Power" => 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(),
|
|
},
|
|
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)),
|
|
"Axe" => Some(SelectedSkillTree::Weapon(ToolKind::Axe)),
|
|
"Hammer" => Some(SelectedSkillTree::Weapon(ToolKind::Hammer)),
|
|
"Bow" => Some(SelectedSkillTree::Weapon(ToolKind::Bow)),
|
|
"Fire Staff" => Some(SelectedSkillTree::Weapon(ToolKind::Staff)),
|
|
"Sceptre" => Some(SelectedSkillTree::Weapon(ToolKind::Sceptre)),
|
|
"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,
|
|
},
|
|
Ability {
|
|
skill: Skill,
|
|
ability_id: &'a str,
|
|
position: PositionSpecifier,
|
|
},
|
|
}
|
|
|
|
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_msg("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(ItemKey::Simple(
|
|
"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(ItemKey::Simple(
|
|
"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(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, state);
|
|
events
|
|
}
|
|
|
|
fn handle_sword_skills_window(
|
|
&mut self,
|
|
diary_tooltip: &Tooltip,
|
|
state: &mut State<DiaryState>,
|
|
ui: &mut UiCell,
|
|
mut events: Vec<Event>,
|
|
) -> Vec<Event> {
|
|
Image::new(self.imgs.sword_tree_paths)
|
|
.wh([1042.0, 636.0])
|
|
.mid_top_with_margin_on(state.ids.content_align, 55.0)
|
|
.graphics_for(state.ids.content_align)
|
|
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
|
|
.set(state.ids.sword_path_overlay, ui);
|
|
|
|
// Sword
|
|
Image::new(self.imgs.sword_bg)
|
|
.wh([933.0, 615.0])
|
|
.mid_top_with_margin_on(state.ids.content_align, 65.0)
|
|
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
|
|
.set(state.ids.sword_bg, ui);
|
|
|
|
use PositionSpecifier::TopLeftWithMarginsOn;
|
|
let skill_buttons = &[
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CrescentSlash),
|
|
ability_id: "veloren.core.pseudo_abilities.sword.crescent_slash",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 537.0, 429.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::FellStrike),
|
|
ability_id: "veloren.core.pseudo_abilities.sword.fell_strike",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 457.0, 527.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::Skewer),
|
|
ability_id: "veloren.core.pseudo_abilities.sword.skewer",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 368.0, 527.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::Cascade),
|
|
ability_id: "veloren.core.pseudo_abilities.sword.cascade",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 457.0, 332.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CrossCut),
|
|
ability_id: "veloren.core.pseudo_abilities.sword.cross_cut",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 368.0, 332.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::Finisher),
|
|
ability_id: "veloren.core.pseudo_abilities.sword.finisher",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 263.0, 429.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::HeavySweep),
|
|
ability_id: "common.abilities.sword.heavy_sweep",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 457.0, 2.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::HeavyPommelStrike),
|
|
ability_id: "common.abilities.sword.heavy_pommel_strike",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 457.0, 91.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::AgileQuickDraw),
|
|
ability_id: "common.abilities.sword.agile_quick_draw",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 142.0, 384.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::AgileFeint),
|
|
ability_id: "common.abilities.sword.agile_feint",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 142.0, 472.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::DefensiveRiposte),
|
|
ability_id: "common.abilities.sword.defensive_riposte",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 457.0, 766.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::DefensiveDisengage),
|
|
ability_id: "common.abilities.sword.defensive_disengage",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 457.0, 855.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CripplingGouge),
|
|
ability_id: "common.abilities.sword.crippling_gouge",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 53.0, 766.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CripplingHamstring),
|
|
ability_id: "common.abilities.sword.crippling_hamstring",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 142.0, 766.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CleavingWhirlwindSlice),
|
|
ability_id: "common.abilities.sword.cleaving_whirlwind_slice",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 53.0, 91.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CleavingEarthSplitter),
|
|
ability_id: "common.abilities.sword.cleaving_earth_splitter",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 142.0, 91.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::HeavyFortitude),
|
|
ability_id: "common.abilities.sword.heavy_fortitude",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 368.0, 2.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::HeavyPillarThrust),
|
|
ability_id: "common.abilities.sword.heavy_pillar_thrust",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 368.0, 91.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::AgileDancingEdge),
|
|
ability_id: "common.abilities.sword.agile_dancing_edge",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 53.0, 385.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::AgileFlurry),
|
|
ability_id: "common.abilities.sword.agile_flurry",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 53.0, 473.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::DefensiveStalwartSword),
|
|
ability_id: "common.abilities.sword.defensive_stalwart_sword",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 368.0, 766.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::DefensiveDeflect),
|
|
ability_id: "common.abilities.sword.defensive_deflect",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 368.0, 855.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CripplingEviscerate),
|
|
ability_id: "common.abilities.sword.crippling_eviscerate",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 142.0, 855.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CripplingBloodyGash),
|
|
ability_id: "common.abilities.sword.crippling_bloody_gash",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 53.0, 855.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CleavingBladeFever),
|
|
ability_id: "common.abilities.sword.cleaving_blade_fever",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 53.0, 2.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Sword(SwordSkill::CleavingSkySplitter),
|
|
ability_id: "common.abilities.sword.cleaving_sky_splitter",
|
|
position: TopLeftWithMarginsOn(state.ids.sword_bg, 142.0, 2.0),
|
|
},
|
|
];
|
|
|
|
state.update(|s| {
|
|
s.ids
|
|
.skills
|
|
.resize(skill_buttons.len(), &mut ui.widget_id_generator())
|
|
});
|
|
state.update(|s| {
|
|
s.ids
|
|
.skill_lock_imgs
|
|
.resize(skill_buttons.len(), &mut ui.widget_id_generator())
|
|
});
|
|
|
|
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip, state);
|
|
events
|
|
}
|
|
|
|
fn handle_axe_skills_window(
|
|
&mut self,
|
|
diary_tooltip: &Tooltip,
|
|
state: &mut State<DiaryState>,
|
|
ui: &mut UiCell,
|
|
mut events: Vec<Event>,
|
|
) -> Vec<Event> {
|
|
// Axe
|
|
Image::new(self.imgs.axe_bg)
|
|
.wh([924.0, 619.0])
|
|
.mid_top_with_margin_on(state.ids.content_align, 65.0)
|
|
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 1.0)))
|
|
.set(state.ids.axe_bg, ui);
|
|
|
|
use PositionSpecifier::TopLeftWithMarginsOn;
|
|
let skill_buttons = &[
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::BrutalSwing),
|
|
ability_id: "common.abilities.axe.brutal_swing",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 387.0, 424.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Berserk),
|
|
ability_id: "common.abilities.axe.berserk",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 374.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::RisingTide),
|
|
ability_id: "common.abilities.axe.rising_tide",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 474.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::SavageSense),
|
|
ability_id: "common.abilities.axe.savage_sense",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 187.0, 324.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::AdrenalineRush),
|
|
ability_id: "common.abilities.axe.adrenaline_rush",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 187.0, 524.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Execute),
|
|
ability_id: "common.abilities.axe.execute",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 187.0, 424.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Maelstrom),
|
|
ability_id: "common.abilities.axe.maelstrom",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 4.0, 424.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Rake),
|
|
ability_id: "common.abilities.axe.rake",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 507.0, 325.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Bloodfeast),
|
|
ability_id: "common.abilities.axe.bloodfeast",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 387.0, 74.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::FierceRaze),
|
|
ability_id: "common.abilities.axe.fierce_raze",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 387.0, 174.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Furor),
|
|
ability_id: "common.abilities.axe.furor",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 24.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Fracture),
|
|
ability_id: "common.abilities.axe.fracture",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 224.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Lacerate),
|
|
ability_id: "common.abilities.axe.lacerate",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 124.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Riptide),
|
|
ability_id: "common.abilities.axe.riptide",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 104.0, 124.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::SkullBash),
|
|
ability_id: "common.abilities.axe.skull_bash",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 507.0, 523.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Sunder),
|
|
ability_id: "common.abilities.axe.sunder",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 387.0, 674.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Plunder),
|
|
ability_id: "common.abilities.axe.plunder",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 387.0, 774.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Defiance),
|
|
ability_id: "common.abilities.axe.defiance",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 624.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Keelhaul),
|
|
ability_id: "common.abilities.axe.keelhaul",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 824.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Bulkhead),
|
|
ability_id: "common.abilities.axe.bulkhead",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 287.0, 724.0),
|
|
},
|
|
SkillIcon::Ability {
|
|
skill: Skill::Axe(AxeSkill::Capsize),
|
|
ability_id: "common.abilities.axe.capsize",
|
|
position: TopLeftWithMarginsOn(state.ids.axe_bg, 104.0, 724.0),
|
|
},
|
|
];
|
|
|
|
state.update(|s| {
|
|
s.ids
|
|
.skills
|
|
.resize(skill_buttons.len(), &mut ui.widget_id_generator())
|
|
});
|
|
state.update(|s| {
|
|
s.ids
|
|
.skill_lock_imgs
|
|
.resize(skill_buttons.len(), &mut ui.widget_id_generator())
|
|
});
|
|
|
|
self.handle_skill_buttons(skill_buttons, ui, &mut events, diary_tooltip, state);
|
|
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_msg("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(ItemKey::Simple("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, state);
|
|
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_msg("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(ItemKey::Simple("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, state);
|
|
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_msg("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(ItemKey::Simple("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, state);
|
|
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_msg("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(ItemKey::Simple("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, state);
|
|
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_msg("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(ItemKey::Simple("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, state);
|
|
events
|
|
}
|
|
|
|
fn handle_skill_buttons(
|
|
&mut self,
|
|
icons: &[SkillIcon],
|
|
ui: &mut UiCell,
|
|
events: &mut Vec<Event>,
|
|
diary_tooltip: &Tooltip,
|
|
state: &mut State<DiaryState>,
|
|
) {
|
|
for (i, icon) in icons.iter().enumerate() {
|
|
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_msg(title),
|
|
&self.localized_strings.get_msg(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,
|
|
),
|
|
SkillIcon::Ability {
|
|
skill,
|
|
ability_id,
|
|
position,
|
|
} => self.create_unlock_ability_button(
|
|
*skill,
|
|
ability_id,
|
|
*position,
|
|
i,
|
|
ui,
|
|
events,
|
|
diary_tooltip,
|
|
state,
|
|
),
|
|
}
|
|
}
|
|
}
|
|
|
|
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 skill_strings = skill_strings(skill);
|
|
let (title, description) =
|
|
skill_strings.localize(self.localized_strings, self.skill_set, skill);
|
|
|
|
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 create_unlock_ability_button(
|
|
&mut self,
|
|
skill: Skill,
|
|
ability_id: &str,
|
|
position: PositionSpecifier,
|
|
widget_index: usize,
|
|
ui: &mut UiCell,
|
|
events: &mut Vec<Event>,
|
|
diary_tooltip: &Tooltip,
|
|
state: &mut State<DiaryState>,
|
|
) {
|
|
let locked = !self.skill_set.prerequisites_met(skill);
|
|
let owned = self.skill_set.has_skill(skill);
|
|
let image_color = if owned {
|
|
TEXT_COLOR
|
|
} else {
|
|
Color::Rgba(0.41, 0.41, 0.41, 0.7)
|
|
};
|
|
|
|
let (title, description) = util::ability_description(ability_id, self.localized_strings);
|
|
|
|
let sp_cost = sp(self.localized_strings, self.skill_set, skill);
|
|
|
|
let description = format!("{description}\n{sp_cost}");
|
|
|
|
let button = Button::image(util::ability_image(self.imgs, ability_id))
|
|
.w_h(76.0, 76.0)
|
|
.position(position)
|
|
.image_color(image_color)
|
|
.with_tooltip(
|
|
self.tooltip_manager,
|
|
&title,
|
|
&description,
|
|
diary_tooltip,
|
|
TEXT_COLOR,
|
|
)
|
|
.set(state.ids.skills[widget_index], ui);
|
|
|
|
// Lock Image
|
|
if locked {
|
|
Image::new(self.imgs.lock)
|
|
.w_h(76.0, 76.0)
|
|
.middle_of(state.ids.skills[widget_index])
|
|
.graphics_for(state.ids.skills[widget_index])
|
|
.color(Some(Color::Rgba(1.0, 1.0, 1.0, 0.8)))
|
|
.set(state.ids.skill_lock_imgs[widget_index], ui);
|
|
}
|
|
|
|
if button.was_clicked() {
|
|
events.push(Event::UnlockSkill(skill));
|
|
};
|
|
}
|
|
}
|
|
|
|
/// Returns skill info as a tuple of title and description.
|
|
///
|
|
/// If you want to get localized version, use `SkillStrings::localize` method
|
|
fn skill_strings(skill: Skill) -> SkillStrings<'static> {
|
|
match skill {
|
|
// general tree
|
|
Skill::General(s) => general_skill_strings(s),
|
|
Skill::UnlockGroup(s) => unlock_skill_strings(s),
|
|
// weapon trees
|
|
Skill::Hammer(s) => hammer_skill_strings(s),
|
|
Skill::Bow(s) => bow_skill_strings(s),
|
|
Skill::Staff(s) => staff_skill_strings(s),
|
|
Skill::Sceptre(s) => sceptre_skill_strings(s),
|
|
// movement trees
|
|
Skill::Roll(s) => roll_skill_strings(s),
|
|
Skill::Climb(s) => climb_skill_strings(s),
|
|
Skill::Swim(s) => swim_skill_strings(s),
|
|
// mining
|
|
Skill::Pick(s) => mining_skill_strings(s),
|
|
_ => SkillStrings::plain("", ""),
|
|
}
|
|
}
|
|
|
|
fn general_skill_strings(skill: GeneralSkill) -> SkillStrings<'static> {
|
|
match skill {
|
|
GeneralSkill::HealthIncrease => SkillStrings::with_const(
|
|
"hud-skill-inc_health_title",
|
|
"hud-skill-inc_health",
|
|
u32::from(HP_PER_LEVEL),
|
|
),
|
|
GeneralSkill::EnergyIncrease => SkillStrings::with_const(
|
|
"hud-skill-inc_energy_title",
|
|
"hud-skill-inc_energy",
|
|
u32::from(ENERGY_PER_LEVEL),
|
|
),
|
|
}
|
|
}
|
|
|
|
fn unlock_skill_strings(group: SkillGroupKind) -> SkillStrings<'static> {
|
|
match group {
|
|
SkillGroupKind::Weapon(ToolKind::Sword) => {
|
|
SkillStrings::plain("hud-skill-unlck_sword_title", "hud-skill-unlck_sword")
|
|
},
|
|
SkillGroupKind::Weapon(ToolKind::Axe) => {
|
|
SkillStrings::plain("hud-skill-unlck_axe_title", "hud-skill-unlck_axe")
|
|
},
|
|
SkillGroupKind::Weapon(ToolKind::Hammer) => {
|
|
SkillStrings::plain("hud-skill-unlck_hammer_title", "hud-skill-unlck_hammer")
|
|
},
|
|
SkillGroupKind::Weapon(ToolKind::Bow) => {
|
|
SkillStrings::plain("hud-skill-unlck_bow_title", "hud-skill-unlck_bow")
|
|
},
|
|
SkillGroupKind::Weapon(ToolKind::Staff) => {
|
|
SkillStrings::plain("hud-skill-unlck_staff_title", "hud-skill-unlck_staff")
|
|
},
|
|
SkillGroupKind::Weapon(ToolKind::Sceptre) => {
|
|
SkillStrings::plain("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::Instrument
|
|
| ToolKind::Pick
|
|
| ToolKind::Shovel
|
|
| ToolKind::Natural
|
|
| ToolKind::Empty,
|
|
) => {
|
|
tracing::warn!("Requesting title for unlocking unexpected skill group");
|
|
SkillStrings::Empty
|
|
},
|
|
}
|
|
}
|
|
|
|
fn hammer_skill_strings(skill: HammerSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.hammer_tree;
|
|
// Single strike upgrades
|
|
match skill {
|
|
HammerSkill::SsKnockback => SkillStrings::with_mult(
|
|
"hud-skill-hmr_single_strike_knockback_title",
|
|
"hud-skill-hmr_single_strike_knockback",
|
|
modifiers.single_strike.knockback,
|
|
),
|
|
HammerSkill::SsDamage => SkillStrings::plain(
|
|
"hud-skill-hmr_single_strike_damage_title",
|
|
"hud-skill-hmr_single_strike_damage",
|
|
),
|
|
HammerSkill::SsSpeed => SkillStrings::plain(
|
|
"hud-skill-hmr_single_strike_speed_title",
|
|
"hud-skill-hmr_single_strike_speed",
|
|
),
|
|
HammerSkill::SsRegen => SkillStrings::plain(
|
|
"hud-skill-hmr_single_strike_regen_title",
|
|
"hud-skill-hmr_single_strike_regen",
|
|
),
|
|
// Charged melee upgrades
|
|
HammerSkill::CDamage => SkillStrings::with_mult(
|
|
"hud-skill-hmr_charged_melee_damage_title",
|
|
"hud-skill-hmr_charged_melee_damage",
|
|
modifiers.charged.scaled_damage,
|
|
),
|
|
HammerSkill::CKnockback => SkillStrings::with_mult(
|
|
"hud-skill-hmr_charged_melee_knockback_title",
|
|
"hud-skill-hmr_charged_melee_knockback",
|
|
modifiers.charged.scaled_knockback,
|
|
),
|
|
HammerSkill::CDrain => SkillStrings::with_mult(
|
|
"hud-skill-hmr_charged_melee_nrg_drain_title",
|
|
"hud-skill-hmr_charged_melee_nrg_drain",
|
|
modifiers.charged.energy_drain,
|
|
),
|
|
HammerSkill::CSpeed => SkillStrings::with_mult(
|
|
"hud-skill-hmr_charged_rate_title",
|
|
"hud-skill-hmr_charged_rate",
|
|
modifiers.charged.charge_rate,
|
|
),
|
|
// Leap upgrades
|
|
HammerSkill::UnlockLeap => SkillStrings::plain(
|
|
"hud-skill-hmr_unlock_leap_title",
|
|
"hud-skill-hmr_unlock_leap",
|
|
),
|
|
HammerSkill::LDamage => SkillStrings::with_mult(
|
|
"hud-skill-hmr_leap_damage_title",
|
|
"hud-skill-hmr_leap_damage",
|
|
modifiers.leap.base_damage,
|
|
),
|
|
HammerSkill::LCost => SkillStrings::with_mult(
|
|
"hud-skill-hmr_leap_cost_title",
|
|
"hud-skill-hmr_leap_cost",
|
|
modifiers.leap.energy_cost,
|
|
),
|
|
HammerSkill::LDistance => SkillStrings::with_mult(
|
|
"hud-skill-hmr_leap_distance_title",
|
|
"hud-skill-hmr_leap_distance",
|
|
modifiers.leap.leap_strength,
|
|
),
|
|
HammerSkill::LKnockback => SkillStrings::with_mult(
|
|
"hud-skill-hmr_leap_knockback_title",
|
|
"hud-skill-hmr_leap_knockback",
|
|
modifiers.leap.knockback,
|
|
),
|
|
HammerSkill::LRange => SkillStrings::with_const_float(
|
|
"hud-skill-hmr_leap_radius_title",
|
|
"hud-skill-hmr_leap_radius",
|
|
modifiers.leap.range,
|
|
),
|
|
}
|
|
}
|
|
|
|
fn bow_skill_strings(skill: BowSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.bow_tree;
|
|
match skill {
|
|
// Passives
|
|
BowSkill::ProjSpeed => SkillStrings::with_mult(
|
|
"hud-skill-bow_projectile_speed_title",
|
|
"hud-skill-bow_projectile_speed",
|
|
modifiers.universal.projectile_speed,
|
|
),
|
|
// Charged upgrades
|
|
BowSkill::CDamage => SkillStrings::with_mult(
|
|
"hud-skill-bow_charged_damage_title",
|
|
"hud-skill-bow_charged_damage",
|
|
modifiers.charged.damage_scaling,
|
|
),
|
|
BowSkill::CRegen => SkillStrings::with_mult(
|
|
"hud-skill-bow_charged_energy_regen_title",
|
|
"hud-skill-bow_charged_energy_regen",
|
|
modifiers.charged.regen_scaling,
|
|
),
|
|
BowSkill::CKnockback => SkillStrings::with_mult(
|
|
"hud-skill-bow_charged_knockback_title",
|
|
"hud-skill-bow_charged_knockback",
|
|
modifiers.charged.knockback_scaling,
|
|
),
|
|
BowSkill::CSpeed => SkillStrings::with_mult(
|
|
"hud-skill-bow_charged_speed_title",
|
|
"hud-skill-bow_charged_speed",
|
|
modifiers.charged.charge_rate,
|
|
),
|
|
BowSkill::CMove => SkillStrings::with_mult(
|
|
"hud-skill-bow_charged_move_title",
|
|
"hud-skill-bow_charged_move",
|
|
modifiers.charged.move_speed,
|
|
),
|
|
// Repeater upgrades
|
|
BowSkill::RDamage => SkillStrings::with_mult(
|
|
"hud-skill-bow_repeater_damage_title",
|
|
"hud-skill-bow_repeater_damage",
|
|
modifiers.repeater.power,
|
|
),
|
|
BowSkill::RCost => SkillStrings::with_mult(
|
|
"hud-skill-bow_repeater_cost_title",
|
|
"hud-skill-bow_repeater_cost",
|
|
modifiers.repeater.energy_cost,
|
|
),
|
|
BowSkill::RSpeed => SkillStrings::with_mult(
|
|
"hud-skill-bow_repeater_speed_title",
|
|
"hud-skill-bow_repeater_speed",
|
|
modifiers.repeater.max_speed,
|
|
),
|
|
// Shotgun upgrades
|
|
BowSkill::UnlockShotgun => SkillStrings::plain(
|
|
"hud-skill-bow_shotgun_unlock_title",
|
|
"hud-skill-bow_shotgun_unlock",
|
|
),
|
|
BowSkill::SDamage => SkillStrings::with_mult(
|
|
"hud-skill-bow_shotgun_damage_title",
|
|
"hud-skill-bow_shotgun_damage",
|
|
modifiers.shotgun.power,
|
|
),
|
|
BowSkill::SCost => SkillStrings::with_mult(
|
|
"hud-skill-bow_shotgun_cost_title",
|
|
"hud-skill-bow_shotgun_cost",
|
|
modifiers.shotgun.energy_cost,
|
|
),
|
|
BowSkill::SArrows => SkillStrings::with_const(
|
|
"hud-skill-bow_shotgun_arrow_count_title",
|
|
"hud-skill-bow_shotgun_arrow_count",
|
|
modifiers.shotgun.num_projectiles,
|
|
),
|
|
BowSkill::SSpread => SkillStrings::with_mult(
|
|
"hud-skill-bow_shotgun_spread_title",
|
|
"hud-skill-bow_shotgun_spread",
|
|
modifiers.shotgun.spread,
|
|
),
|
|
}
|
|
}
|
|
|
|
fn staff_skill_strings(skill: StaffSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.staff_tree;
|
|
match skill {
|
|
// Basic ranged upgrades
|
|
StaffSkill::BDamage => SkillStrings::with_mult(
|
|
"hud-skill-st_damage_title",
|
|
"hud-skill-st_damage",
|
|
modifiers.fireball.power,
|
|
),
|
|
StaffSkill::BRegen => SkillStrings::with_mult(
|
|
"hud-skill-st_energy_regen_title",
|
|
"hud-skill-st_energy_regen",
|
|
modifiers.fireball.regen,
|
|
),
|
|
StaffSkill::BRadius => SkillStrings::with_mult(
|
|
"hud-skill-st_explosion_radius_title",
|
|
"hud-skill-st_explosion_radius",
|
|
modifiers.fireball.range,
|
|
),
|
|
// Flamethrower upgrades
|
|
StaffSkill::FDamage => SkillStrings::with_mult(
|
|
"hud-skill-st_flamethrower_damage_title",
|
|
"hud-skill-st_flamethrower_damage",
|
|
modifiers.flamethrower.damage,
|
|
),
|
|
StaffSkill::FRange => SkillStrings::with_mult(
|
|
"hud-skill-st_flamethrower_range_title",
|
|
"hud-skill-st_flamethrower_range",
|
|
modifiers.flamethrower.range,
|
|
),
|
|
StaffSkill::FDrain => SkillStrings::with_mult(
|
|
"hud-skill-st_energy_drain_title",
|
|
"hud-skill-st_energy_drain",
|
|
modifiers.flamethrower.energy_drain,
|
|
),
|
|
StaffSkill::FVelocity => SkillStrings::with_mult(
|
|
"hud-skill-st_flame_velocity_title",
|
|
"hud-skill-st_flame_velocity",
|
|
modifiers.flamethrower.velocity,
|
|
),
|
|
// Shockwave upgrades
|
|
StaffSkill::UnlockShockwave => SkillStrings::plain(
|
|
"hud-skill-st_shockwave_unlock_title",
|
|
"hud-skill-st_shockwave_unlock",
|
|
),
|
|
StaffSkill::SDamage => SkillStrings::with_mult(
|
|
"hud-skill-st_shockwave_damage_title",
|
|
"hud-skill-st_shockwave_damage",
|
|
modifiers.shockwave.damage,
|
|
),
|
|
StaffSkill::SKnockback => SkillStrings::with_mult(
|
|
"hud-skill-st_shockwave_knockback_title",
|
|
"hud-skill-st_shockwave_knockback",
|
|
modifiers.shockwave.knockback,
|
|
),
|
|
StaffSkill::SRange => SkillStrings::with_mult(
|
|
"hud-skill-st_shockwave_range_title",
|
|
"hud-skill-st_shockwave_range",
|
|
modifiers.shockwave.duration,
|
|
),
|
|
StaffSkill::SCost => SkillStrings::with_mult(
|
|
"hud-skill-st_shockwave_cost_title",
|
|
"hud-skill-st_shockwave_cost",
|
|
modifiers.shockwave.energy_cost,
|
|
),
|
|
}
|
|
}
|
|
|
|
fn sceptre_skill_strings(skill: SceptreSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.sceptre_tree;
|
|
match skill {
|
|
// Lifesteal beam upgrades
|
|
SceptreSkill::LDamage => SkillStrings::with_mult(
|
|
"hud-skill-sc_lifesteal_damage_title",
|
|
"hud-skill-sc_lifesteal_damage",
|
|
modifiers.beam.damage,
|
|
),
|
|
SceptreSkill::LRange => SkillStrings::with_mult(
|
|
"hud-skill-sc_lifesteal_range_title",
|
|
"hud-skill-sc_lifesteal_range",
|
|
modifiers.beam.range,
|
|
),
|
|
SceptreSkill::LLifesteal => SkillStrings::with_mult(
|
|
"hud-skill-sc_lifesteal_lifesteal_title",
|
|
"hud-skill-sc_lifesteal_lifesteal",
|
|
modifiers.beam.lifesteal,
|
|
),
|
|
SceptreSkill::LRegen => SkillStrings::with_mult(
|
|
"hud-skill-sc_lifesteal_regen_title",
|
|
"hud-skill-sc_lifesteal_regen",
|
|
modifiers.beam.energy_regen,
|
|
),
|
|
// Healing aura upgrades
|
|
SceptreSkill::HHeal => SkillStrings::with_mult(
|
|
"hud-skill-sc_heal_heal_title",
|
|
"hud-skill-sc_heal_heal",
|
|
modifiers.healing_aura.strength,
|
|
),
|
|
SceptreSkill::HRange => SkillStrings::with_mult(
|
|
"hud-skill-sc_heal_range_title",
|
|
"hud-skill-sc_heal_range",
|
|
modifiers.healing_aura.range,
|
|
),
|
|
SceptreSkill::HDuration => SkillStrings::with_mult(
|
|
"hud-skill-sc_heal_duration_title",
|
|
"hud-skill-sc_heal_duration",
|
|
modifiers.healing_aura.duration,
|
|
),
|
|
SceptreSkill::HCost => SkillStrings::with_mult(
|
|
"hud-skill-sc_heal_cost_title",
|
|
"hud-skill-sc_heal_cost",
|
|
modifiers.healing_aura.energy_cost,
|
|
),
|
|
// Warding aura upgrades
|
|
SceptreSkill::UnlockAura => SkillStrings::plain(
|
|
"hud-skill-sc_wardaura_unlock_title",
|
|
"hud-skill-sc_wardaura_unlock",
|
|
),
|
|
SceptreSkill::AStrength => SkillStrings::with_mult(
|
|
"hud-skill-sc_wardaura_strength_title",
|
|
"hud-skill-sc_wardaura_strength",
|
|
modifiers.warding_aura.strength,
|
|
),
|
|
SceptreSkill::ADuration => SkillStrings::with_mult(
|
|
"hud-skill-sc_wardaura_duration_title",
|
|
"hud-skill-sc_wardaura_duration",
|
|
modifiers.warding_aura.duration,
|
|
),
|
|
SceptreSkill::ARange => SkillStrings::with_mult(
|
|
"hud-skill-sc_wardaura_range_title",
|
|
"hud-skill-sc_wardaura_range",
|
|
modifiers.warding_aura.range,
|
|
),
|
|
SceptreSkill::ACost => SkillStrings::with_mult(
|
|
"hud-skill-sc_wardaura_cost_title",
|
|
"hud-skill-sc_wardaura_cost",
|
|
modifiers.warding_aura.energy_cost,
|
|
),
|
|
}
|
|
}
|
|
|
|
fn roll_skill_strings(skill: RollSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.general_tree.roll;
|
|
match skill {
|
|
RollSkill::Cost => SkillStrings::with_mult(
|
|
"hud-skill-roll_energy_title",
|
|
"hud-skill-roll_energy",
|
|
modifiers.energy_cost,
|
|
),
|
|
RollSkill::Strength => SkillStrings::with_mult(
|
|
"hud-skill-roll_speed_title",
|
|
"hud-skill-roll_speed",
|
|
modifiers.strength,
|
|
),
|
|
RollSkill::Duration => SkillStrings::with_mult(
|
|
"hud-skill-roll_dur_title",
|
|
"hud-skill-roll_dur",
|
|
modifiers.duration,
|
|
),
|
|
}
|
|
}
|
|
|
|
fn climb_skill_strings(skill: ClimbSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.general_tree.climb;
|
|
match skill {
|
|
ClimbSkill::Cost => SkillStrings::with_mult(
|
|
"hud-skill-climbing_cost_title",
|
|
"hud-skill-climbing_cost",
|
|
modifiers.energy_cost,
|
|
),
|
|
ClimbSkill::Speed => SkillStrings::with_mult(
|
|
"hud-skill-climbing_speed_title",
|
|
"hud-skill-climbing_speed",
|
|
modifiers.speed,
|
|
),
|
|
}
|
|
}
|
|
|
|
fn swim_skill_strings(skill: SwimSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.general_tree.swim;
|
|
match skill {
|
|
SwimSkill::Speed => SkillStrings::with_mult(
|
|
"hud-skill-swim_speed_title",
|
|
"hud-skill-swim_speed",
|
|
modifiers.speed,
|
|
),
|
|
}
|
|
}
|
|
|
|
fn mining_skill_strings(skill: MiningSkill) -> SkillStrings<'static> {
|
|
let modifiers = SKILL_MODIFIERS.mining_tree;
|
|
match skill {
|
|
MiningSkill::Speed => SkillStrings::with_mult(
|
|
"hud-skill-pick_strike_speed_title",
|
|
"hud-skill-pick_strike_speed",
|
|
modifiers.speed,
|
|
),
|
|
MiningSkill::OreGain => SkillStrings::with_const(
|
|
"hud-skill-pick_strike_oregain_title",
|
|
"hud-skill-pick_strike_oregain",
|
|
(modifiers.ore_gain * 100.0).round() as u32,
|
|
),
|
|
MiningSkill::GemGain => SkillStrings::with_const(
|
|
"hud-skill-pick_strike_gemgain_title",
|
|
"hud-skill-pick_strike_gemgain",
|
|
(modifiers.gem_gain * 100.0).round() as u32,
|
|
),
|
|
}
|
|
}
|
|
|
|
/// Helper object used returned by `skill_strings` as source for
|
|
/// later internationalization and formatting.
|
|
enum SkillStrings<'a> {
|
|
Plain {
|
|
title: &'a str,
|
|
desc: &'a str,
|
|
},
|
|
WithConst {
|
|
title: &'a str,
|
|
desc: &'a str,
|
|
constant: u32,
|
|
},
|
|
WithConstFloat {
|
|
title: &'a str,
|
|
desc: &'a str,
|
|
constant: f32,
|
|
},
|
|
WithMult {
|
|
title: &'a str,
|
|
desc: &'a str,
|
|
multiplier: f32,
|
|
},
|
|
Empty,
|
|
}
|
|
|
|
impl<'a> SkillStrings<'a> {
|
|
fn plain(title: &'a str, desc: &'a str) -> Self { Self::Plain { title, desc } }
|
|
|
|
fn with_const(title: &'a str, desc: &'a str, constant: u32) -> Self {
|
|
Self::WithConst {
|
|
title,
|
|
desc,
|
|
constant,
|
|
}
|
|
}
|
|
|
|
fn with_const_float(title: &'a str, desc: &'a str, constant: f32) -> Self {
|
|
Self::WithConstFloat {
|
|
title,
|
|
desc,
|
|
constant,
|
|
}
|
|
}
|
|
|
|
fn with_mult(title: &'a str, desc: &'a str, multiplier: f32) -> Self {
|
|
Self::WithMult {
|
|
title,
|
|
desc,
|
|
multiplier,
|
|
}
|
|
}
|
|
|
|
fn localize<'loc>(
|
|
&self,
|
|
i18n: &'loc Localization,
|
|
skill_set: &SkillSet,
|
|
skill: Skill,
|
|
) -> (Cow<'loc, str>, Cow<'loc, str>) {
|
|
match self {
|
|
Self::Plain { title, desc } => {
|
|
let title = i18n.get_msg(title);
|
|
|
|
let args = i18n::fluent_args! {
|
|
"SP" => sp(i18n, skill_set, skill),
|
|
};
|
|
let desc = i18n.get_msg_ctx(desc, &args);
|
|
|
|
(title, desc)
|
|
},
|
|
Self::WithConst {
|
|
title,
|
|
desc,
|
|
constant,
|
|
} => {
|
|
let title = i18n.get_msg(title);
|
|
let args = i18n::fluent_args! {
|
|
"boost" => constant,
|
|
"SP" => sp(i18n, skill_set, skill),
|
|
};
|
|
let desc = i18n.get_msg_ctx(desc, &args);
|
|
|
|
(title, desc)
|
|
},
|
|
Self::WithConstFloat {
|
|
title,
|
|
desc,
|
|
constant,
|
|
} => {
|
|
let title = i18n.get_msg(title);
|
|
let args = i18n::fluent_args! {
|
|
"boost" => constant,
|
|
"SP" => sp(i18n, skill_set, skill),
|
|
};
|
|
let desc = i18n.get_msg_ctx(desc, &args);
|
|
|
|
(title, desc)
|
|
},
|
|
Self::WithMult {
|
|
title,
|
|
desc,
|
|
multiplier,
|
|
} => {
|
|
let percentage = hud::multiplier_to_percentage(*multiplier).abs();
|
|
|
|
let title = i18n.get_msg(title);
|
|
|
|
let args = i18n::fluent_args! {
|
|
"boost" => format!("{percentage:.0}"),
|
|
"SP" => sp(i18n, skill_set, skill),
|
|
};
|
|
let desc = i18n.get_msg_ctx(desc, &args);
|
|
|
|
(title, desc)
|
|
},
|
|
Self::Empty => (Cow::Borrowed(""), Cow::Borrowed("")),
|
|
}
|
|
}
|
|
}
|
|
|
|
fn sp<'loc>(i18n: &'loc Localization, skill_set: &SkillSet, skill: Skill) -> Cow<'loc, str> {
|
|
let current_level = skill_set.skill_level(skill);
|
|
if matches!(current_level, Ok(level) if level == skill.max_level()) {
|
|
Cow::Borrowed("")
|
|
} else {
|
|
i18n.get_msg_ctx("hud-skill-req_sp", &i18n::fluent_args! {
|
|
"number" => skill_set.skill_cost(skill),
|
|
})
|
|
}
|
|
}
|