use super::{ cr_color, img_ids::{Imgs, ImgsRot}, item_imgs::ItemImgs, slots::{ArmorSlot, EquipSlot, InventorySlot, SlotManager}, util::loadout_slot_text, Show, CRITICAL_HP_COLOR, LOW_HP_COLOR, QUALITY_COMMON, TEXT_COLOR, UI_HIGHLIGHT_0, UI_MAIN, }; use crate::{ hud::get_quality_col, i18n::Localization, ui::{ fonts::Fonts, slot::{ContentSize, SlotMaker}, ImageFrame, Tooltip, TooltipManager, Tooltipable, }, }; use client::Client; use common::{ combat::{combat_rating, Damage}, comp::{item::Quality, Body, Energy, Health, Stats}, }; use conrod_core::{ color, widget::{self, Button, Image, Rectangle, Scrollbar, Text}, widget_ids, Color, Colorable, Positionable, Sizeable, Widget, WidgetCommon, }; use crate::hud::slots::SlotKind; use vek::Vec2; widget_ids! { pub struct Ids { test, bag_close, inv_alignment, inv_grid_1, inv_grid_2, inv_scrollbar, inv_slots_0, inv_slots[], //tooltip[], bg, bg_frame, char_ico, coin_ico, space_txt, currency_txt, inventory_title, inventory_title_bg, scrollbar_bg, scrollbar_slots, tab_1, tab_2, tab_3, tab_4, bag_expand_btn, // Armor Slots slots_bg, head_slot, neck_slot, chest_slot, shoulders_slot, hands_slot, legs_slot, belt_slot, lantern_slot, ring1_slot, ring2_slot, feet_slot, back_slot, tabard_slot, glider_slot, mainhand_slot, offhand_slot, bag1_slot, bag2_slot, bag3_slot, bag4_slot, // Stats stat_icons[], stat_txts[], } } #[derive(WidgetCommon)] pub struct Bag<'a> { client: &'a Client, imgs: &'a Imgs, item_imgs: &'a ItemImgs, fonts: &'a Fonts, #[conrod(common_builder)] common: widget::CommonBuilder, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, slot_manager: &'a mut SlotManager, _pulse: f32, localized_strings: &'a Localization, stats: &'a Stats, health: &'a Health, energy: &'a Energy, show: &'a Show, body: &'a Body, } impl<'a> Bag<'a> { #[allow(clippy::too_many_arguments)] // TODO: Pending review in #587 pub fn new( client: &'a Client, imgs: &'a Imgs, item_imgs: &'a ItemImgs, fonts: &'a Fonts, rot_imgs: &'a ImgsRot, tooltip_manager: &'a mut TooltipManager, slot_manager: &'a mut SlotManager, pulse: f32, localized_strings: &'a Localization, stats: &'a Stats, health: &'a Health, energy: &'a Energy, show: &'a Show, body: &'a Body, ) -> Self { Self { client, imgs, item_imgs, fonts, common: widget::CommonBuilder::default(), rot_imgs, tooltip_manager, slot_manager, _pulse: pulse, localized_strings, stats, energy, health, show, body, } } } const STATS: [&str; 4] = ["Health", "Stamina", "Protection", "Combat Rating"]; pub struct State { ids: Ids, } pub enum Event { BagExpand, Close, } impl<'a> Widget for Bag<'a> { type Event = Option; type State = State; type Style = (); fn init_state(&self, id_gen: widget::id::Generator) -> Self::State { State { ids: Ids::new(id_gen), } } #[allow(clippy::unused_unit)] // TODO: Pending review in #587 fn style(&self) -> Self::Style { () } #[allow(clippy::useless_format)] // TODO: Pending review in #587 fn update(self, args: widget::UpdateArgs) -> Self::Event { let widget::UpdateArgs { state, ui, .. } = args; let mut event = None; let bag_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); let inventories = self.client.inventories(); let inventory = match inventories.get(self.client.entity()) { Some(l) => l, None => return None, }; let space_used = inventory.populated_slots(); let space_max = inventory.slots().count(); let bag_space = format!("{}/{}", space_used, space_max); let bag_space_percentage = space_used as f32 / space_max as f32; let currency = 0; // TODO: Add as a Stat // Tooltips let item_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); // BG Image::new(if self.show.stats { self.imgs.inv_bg_stats } else if self.show.bag_inv { self.imgs.inv_bg_bag } else { self.imgs.inv_bg_armor }) .w_h(424.0, 708.0) .bottom_right_with_margins_on(ui.window, 60.0, 5.0) .color(Some(UI_MAIN)) .set(state.ids.bg, ui); Image::new(if self.show.bag_inv { self.imgs.inv_frame_bag } else { self.imgs.inv_frame }) .w_h(424.0, 708.0) .middle_of(state.ids.bg) .color(Some(UI_HIGHLIGHT_0)) .set(state.ids.bg_frame, ui); // Title Text::new( &self .localized_strings .get("hud.bag.inventory") .replace("{playername}", &self.stats.name.to_string().as_str()), ) .mid_top_with_margin_on(state.ids.bg_frame, 9.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(20)) .color(Color::Rgba(0.0, 0.0, 0.0, 1.0)) .set(state.ids.inventory_title_bg, ui); Text::new( &self .localized_strings .get("hud.bag.inventory") .replace("{playername}", &self.stats.name.to_string().as_str()), ) .top_left_with_margins_on(state.ids.inventory_title_bg, 2.0, 2.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(20)) .color(TEXT_COLOR) .set(state.ids.inventory_title, ui); // Slots Scrollbar if space_max > 45 && !self.show.bag_inv { // Scrollbar-BG Image::new(self.imgs.scrollbar_bg) .w_h(9.0, 173.0) .bottom_right_with_margins_on(state.ids.bg_frame, 42.0, 3.0) .color(Some(UI_HIGHLIGHT_0)) .set(state.ids.scrollbar_bg, ui); // Scrollbar Scrollbar::y_axis(state.ids.inv_alignment) .thickness(5.0) .h(123.0) .color(UI_MAIN) .middle_of(state.ids.scrollbar_bg) .set(state.ids.scrollbar_slots, ui); } else if space_max > 135 { // Scrollbar-BG Image::new(self.imgs.scrollbar_bg_big) .w_h(9.0, 592.0) .bottom_right_with_margins_on(state.ids.bg_frame, 42.0, 3.0) .color(Some(UI_HIGHLIGHT_0)) .set(state.ids.scrollbar_bg, ui); // Scrollbar Scrollbar::y_axis(state.ids.inv_alignment) .thickness(5.0) .h(542.0) .color(UI_MAIN) .middle_of(state.ids.scrollbar_bg) .set(state.ids.scrollbar_slots, ui); }; // Char Pixel-Art Image::new(self.imgs.char_art) .w_h(40.0, 37.0) .top_left_with_margins_on(state.ids.bg, 4.0, 2.0) .set(state.ids.char_ico, ui); // Coin Icon and Currency Text Image::new(self.imgs.coin_ico) .w_h(16.0, 17.0) .bottom_left_with_margins_on(state.ids.bg_frame, 2.0, 43.0) .set(state.ids.coin_ico, ui); Text::new(&format!("{}", currency)) .bottom_left_with_margins_on(state.ids.bg_frame, 6.0, 64.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(14)) .color(Color::Rgba(0.871, 0.863, 0.05, 1.0)) .set(state.ids.currency_txt, ui); //Free Bag-Space Text::new(&bag_space) .bottom_right_with_margins_on(state.ids.bg_frame, 6.0, 43.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(14)) .color(if bag_space_percentage < 0.8 { TEXT_COLOR } else if bag_space_percentage < 1.0 { LOW_HP_COLOR } else { CRITICAL_HP_COLOR }) .set(state.ids.space_txt, ui); // Alignment for Grid Rectangle::fill_with( [362.0, if self.show.bag_inv { 600.0 } else { 200.0 }], color::TRANSPARENT, ) .bottom_left_with_margins_on(state.ids.bg_frame, 29.0, 46.5) .scroll_kids_vertically() .set(state.ids.inv_alignment, ui); // Button to expand bag let txt = if self.show.bag_inv { "Show Loadout" } else { "Expand Bag" }; let expand_btn = Button::image(if self.show.bag_inv { self.imgs.collapse_btn } else { self.imgs.expand_btn }) .w_h(30.0, 17.0) .hover_image(if self.show.bag_inv { self.imgs.collapse_btn_hover } else { self.imgs.expand_btn_hover }) .press_image(if self.show.bag_inv { self.imgs.collapse_btn_press } else { self.imgs.expand_btn_press }); // Only show expand button when it's needed... if space_max > 45 && !self.show.bag_inv { if expand_btn .top_left_with_margins_on(state.ids.bg_frame, 460.0, 211.5) .with_tooltip(self.tooltip_manager, &txt, "", &bag_tooltip, TEXT_COLOR) .set(state.ids.bag_expand_btn, ui) .was_clicked() { event = Some(Event::BagExpand); } } else if self.show.bag_inv { //... but always show it when the bag is expanded if expand_btn .top_left_with_margins_on(state.ids.bg_frame, 53.0, 211.5) .with_tooltip(self.tooltip_manager, &txt, "", &bag_tooltip, TEXT_COLOR) .set(state.ids.bag_expand_btn, ui) .was_clicked() { event = Some(Event::BagExpand); } } // Title Text::new( &self .localized_strings .get("hud.bag.inventory") .replace("{playername}", &self.stats.name.to_string().as_str()), ) .mid_top_with_margin_on(state.ids.bg_frame, 9.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(22)) .color(Color::Rgba(0.0, 0.0, 0.0, 1.0)) .set(state.ids.inventory_title_bg, ui); Text::new( &self .localized_strings .get("hud.bag.inventory") .replace("{playername}", &self.stats.name.to_string().as_str()), ) .top_left_with_margins_on(state.ids.inventory_title_bg, 2.0, 2.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(22)) .color(TEXT_COLOR) .set(state.ids.inventory_title, ui); // Armor Slots let mut slot_maker = SlotMaker { empty_slot: self.imgs.armor_slot_empty, filled_slot: self.imgs.armor_slot, selected_slot: self.imgs.armor_slot_sel, background_color: Some(UI_HIGHLIGHT_0), content_size: ContentSize { width_height_ratio: 1.0, max_fraction: 0.75, /* Changes the item image size by setting a maximum * fraction * of either the width or height */ }, 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: inventory, image_source: self.item_imgs, slot_manager: Some(self.slot_manager), }; let i18n = &self.localized_strings; let filled_slot = self.imgs.armor_slot; if !self.show.bag_inv { // Stat icons and text state.update(|s| { s.ids .stat_icons .resize(STATS.len(), &mut ui.widget_id_generator()) }); state.update(|s| { s.ids .stat_txts .resize(STATS.len(), &mut ui.widget_id_generator()) }); // Stats let combat_rating = combat_rating(inventory, self.health, self.stats, *self.body).min(999.9); let indicator_col = cr_color(combat_rating); for i in STATS.iter().copied().enumerate() { let btn = Button::image(match i.1 { "Health" => self.imgs.health_ico, "Stamina" => self.imgs.stamina_ico, "Combat Rating" => self.imgs.combat_rating_ico, "Protection" => self.imgs.protection_ico, _ => self.imgs.nothing, }) .w_h(20.0, 20.0) .image_color(if i.1 == "Combat Rating" { indicator_col } else { TEXT_COLOR }); let protection_txt = format!( "{}%", (100.0 * Damage::compute_damage_reduction(inventory)) as i32 ); let health_txt = format!("{}", (self.health.maximum() as f32 / 10.0) as usize); let stamina_txt = format!("{}", (self.energy.maximum() as f32 / 10.0) as usize); let combat_rating_txt = format!("{}", (combat_rating * 10.0) as usize); let btn = if i.0 == 0 { btn.top_left_with_margins_on(state.ids.bg_frame, 55.0, 10.0) } else { btn.down_from(state.ids.stat_icons[i.0 - 1], 7.0) }; // TODO: Translation let tooltip_head = match i.1 { "Health" => "Health", "Stamina" => "Stamina", "Combat Rating" => "Combat Rating", "Protection" => "Protection", _ => "", }; // TODO: Translation let tooltip_txt = match i.1 { "Combat Rating" => "Calculated from your\nequipment and health.", "Protection" => "Damage reduction through armor", _ => "", }; btn.with_tooltip( self.tooltip_manager, &tooltip_head, &tooltip_txt, &bag_tooltip, TEXT_COLOR, ) .set(state.ids.stat_icons[i.0], ui); Text::new(match i.1 { "Health" => &health_txt, "Stamina" => &stamina_txt, "Combat Rating" => &combat_rating_txt, "Protection" => &protection_txt, _ => "", }) .right_from(state.ids.stat_icons[i.0], 10.0) .font_id(self.fonts.cyri.conrod_id) .font_size(self.fonts.cyri.scale(14)) .color(TEXT_COLOR) .graphics_for(state.ids.stat_icons[i.0]) .set(state.ids.stat_txts[i.0], ui); } // Loadout Slots // Head let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Head)), || (i18n.get("hud.bag.head"), ""), ); let head_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Head)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Head), [45.0; 2]) .mid_top_with_margin_on(state.ids.bg_frame, 60.0) .with_icon(self.imgs.head_bg, Vec2::new(32.0, 40.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, head_q_col, ) .set(state.ids.head_slot, ui); // Necklace let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Neck)), || (i18n.get("hud.bag.neck"), ""), ); let neck_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Neck)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Neck), [45.0; 2]) .mid_bottom_with_margin_on(state.ids.head_slot, -55.0) .with_icon(self.imgs.necklace_bg, Vec2::new(40.0, 31.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, neck_q_col, ) .set(state.ids.neck_slot, ui); // Chest //Image::new(self.imgs.armor_slot) // different graphics for empty/non empty let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Chest)), || (i18n.get("hud.bag.chest"), ""), ); let chest_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Chest)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Chest), [85.0; 2]) .mid_bottom_with_margin_on(state.ids.neck_slot, -95.0) .with_icon(self.imgs.chest_bg, Vec2::new(64.0, 42.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, chest_q_col, ) .set(state.ids.chest_slot, ui); // Shoulders let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Shoulders)), || (i18n.get("hud.bag.shoulders"), ""), ); let shoulder_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Shoulders)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Shoulders), [70.0; 2]) .bottom_left_with_margins_on(state.ids.chest_slot, 0.0, -80.0) .with_icon(self.imgs.shoulders_bg, Vec2::new(60.0, 36.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, shoulder_q_col, ) .set(state.ids.shoulders_slot, ui); // Hands let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Hands)), || (i18n.get("hud.bag.hands"), ""), ); let chest_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Hands)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Hands), [70.0; 2]) .bottom_right_with_margins_on(state.ids.chest_slot, 0.0, -80.0) .with_icon(self.imgs.hands_bg, Vec2::new(55.0, 60.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, chest_q_col, ) .set(state.ids.hands_slot, ui); // Belt let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Belt)), || (i18n.get("hud.bag.belt"), ""), ); let belt_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Belt)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Belt), [45.0; 2]) .mid_bottom_with_margin_on(state.ids.chest_slot, -55.0) .with_icon(self.imgs.belt_bg, Vec2::new(40.0, 23.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, belt_q_col, ) .set(state.ids.belt_slot, ui); // Legs let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Legs)), || (i18n.get("hud.bag.legs"), ""), ); let legs_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Legs)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Legs), [85.0; 2]) .mid_bottom_with_margin_on(state.ids.belt_slot, -95.0) .with_icon(self.imgs.legs_bg, Vec2::new(48.0, 70.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, legs_q_col, ) .set(state.ids.legs_slot, ui); // Ring let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Ring1)), || (i18n.get("hud.bag.ring"), ""), ); let ring_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Ring1)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Ring1), [45.0; 2]) .bottom_left_with_margins_on(state.ids.hands_slot, -55.0, 0.0) .with_icon(self.imgs.ring_bg, Vec2::new(36.0, 40.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, ring_q_col, ) .set(state.ids.ring1_slot, ui); // Ring 2 let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Ring2)), || (i18n.get("hud.bag.ring"), ""), ); let ring2_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Ring2)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Ring2), [45.0; 2]) .bottom_right_with_margins_on(state.ids.shoulders_slot, -55.0, 0.0) .with_icon(self.imgs.ring_bg, Vec2::new(36.0, 40.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, ring2_q_col, ) .set(state.ids.ring2_slot, ui); // Back let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Back)), || (i18n.get("hud.bag.back"), ""), ); let back_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Back)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Back), [45.0; 2]) .down_from(state.ids.ring2_slot, 10.0) .with_icon(self.imgs.back_bg, Vec2::new(33.0, 40.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, back_q_col, ) .set(state.ids.back_slot, ui); // Foot let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Feet)), || (i18n.get("hud.bag.feet"), ""), ); let foot_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Feet)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Feet), [45.0; 2]) .down_from(state.ids.ring1_slot, 10.0) .with_icon(self.imgs.feet_bg, Vec2::new(32.0, 40.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, foot_q_col, ) .set(state.ids.feet_slot, ui); // Lantern let (title, desc) = loadout_slot_text(inventory.equipped(EquipSlot::Lantern), || { (i18n.get("hud.bag.lantern"), "") }); let lantern_q_col = inventory .equipped(EquipSlot::Lantern) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Lantern, [45.0; 2]) .top_right_with_margins_on(state.ids.bg_frame, 60.0, 5.0) .with_icon(self.imgs.lantern_bg, Vec2::new(24.0, 38.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, lantern_q_col, ) .set(state.ids.lantern_slot, ui); // Glider let (title, desc) = loadout_slot_text(inventory.equipped(EquipSlot::Glider), || { (i18n.get("hud.bag.glider"), "") }); let glider_q_col = inventory .equipped(EquipSlot::Glider) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Glider, [45.0; 2]) .down_from(state.ids.lantern_slot, 5.0) .with_icon(self.imgs.glider_bg, Vec2::new(38.0, 38.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, glider_q_col, ) .set(state.ids.glider_slot, ui); // Tabard let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Tabard)), || (i18n.get("hud.bag.tabard"), ""), ); let tabard_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Tabard)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Tabard), [45.0; 2]) .down_from(state.ids.glider_slot, 5.0) .with_icon(self.imgs.tabard_bg, Vec2::new(38.0, 38.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, tabard_q_col, ) .set(state.ids.tabard_slot, ui); // Mainhand/Left-Slot let (title, desc) = loadout_slot_text(inventory.equipped(EquipSlot::Mainhand), || { (i18n.get("hud.bag.mainhand"), "") }); let mainhand_q_col = inventory .equipped(EquipSlot::Mainhand) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Mainhand, [85.0; 2]) .bottom_right_with_margins_on(state.ids.back_slot, -95.0, 0.0) .with_icon(self.imgs.mainhand_bg, Vec2::new(75.0, 75.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, mainhand_q_col, ) .set(state.ids.mainhand_slot, ui); // Offhand/Right-Slot let (title, desc) = loadout_slot_text(inventory.equipped(EquipSlot::Offhand), || { (i18n.get("hud.bag.offhand"), "") }); let offhand_q_col = inventory .equipped(EquipSlot::Offhand) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Offhand, [85.0; 2]) .bottom_left_with_margins_on(state.ids.feet_slot, -95.0, 0.0) .with_icon(self.imgs.offhand_bg, Vec2::new(75.0, 75.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, offhand_q_col, ) .set(state.ids.offhand_slot, ui); } // Bag 1 let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Bag1)), || (i18n.get("hud.bag.bag"), ""), ); let bag1_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Bag1)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Bag1), [35.0; 2]) .bottom_left_with_margins_on( state.ids.bg_frame, if self.show.bag_inv { 600.0 } else { 167.0 }, 3.0, ) .with_icon(self.imgs.bag_bg, Vec2::new(28.0, 24.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, bag1_q_col, ) .set(state.ids.bag1_slot, ui); // Bag 2 let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Bag2)), || (i18n.get("hud.bag.bag"), ""), ); let bag2_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Bag2)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Bag2), [35.0; 2]) .down_from(state.ids.bag1_slot, 2.0) .with_icon(self.imgs.bag_bg, Vec2::new(28.0, 24.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, bag2_q_col, ) .set(state.ids.bag2_slot, ui); // Bag 3 let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Bag3)), || (i18n.get("hud.bag.bag"), ""), ); let bag3_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Bag3)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Bag3), [35.0; 2]) .down_from(state.ids.bag2_slot, 2.0) .with_icon(self.imgs.bag_bg, Vec2::new(28.0, 24.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, bag3_q_col, ) .set(state.ids.bag3_slot, ui); // Bag 4 let (title, desc) = loadout_slot_text( inventory.equipped(EquipSlot::Armor(ArmorSlot::Bag4)), || (i18n.get("hud.bag.bag"), ""), ); let bag4_q_col = inventory .equipped(EquipSlot::Armor(ArmorSlot::Bag4)) .map(|item| get_quality_col(item)) .unwrap_or(QUALITY_COMMON); slot_maker .fabricate(EquipSlot::Armor(ArmorSlot::Bag4), [35.0; 2]) .down_from(state.ids.bag3_slot, 2.0) .with_icon(self.imgs.bag_bg, Vec2::new(28.0, 24.0), Some(UI_MAIN)) .filled_slot(filled_slot) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, bag4_q_col, ) .set(state.ids.bag4_slot, ui); // Bag Slots // Create available inventory slot widgets if state.ids.inv_slots.len() < inventory.capacity() { state.update(|s| { s.ids .inv_slots .resize(inventory.capacity(), &mut ui.widget_id_generator()); }); } // Determine the range of inventory slots that are provided by the loadout item // that the mouse is over let mouseover_loadout_slots = self .slot_manager .mouse_over_slot .and_then(|x| { if let SlotKind::Equip(e) = x { inventory.get_slot_range_for_equip_slot(e) } else { None } }) .unwrap_or(0usize..0usize); // Display inventory contents 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.75, }, 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: inventory, image_source: self.item_imgs, slot_manager: Some(self.slot_manager), }; for (i, (pos, item)) in inventory.slots_with_id().enumerate() { let x = i % 9; let y = i / 9; // Slot let mut slot_widget = slot_maker .fabricate(InventorySlot(pos), [40.0; 2]) .top_left_with_margins_on( state.ids.inv_alignment, 0.0 + y as f64 * (40.0), 0.0 + x as f64 * (40.0), ); // Highlight slots are provided by the loadout item that the mouse is over if mouseover_loadout_slots.contains(&i) { slot_widget = slot_widget.with_background_color(Color::Rgba(1.0, 1.0, 1.0, 1.0)); } if let Some(item) = item { let (title, desc) = super::util::item_text(item); let quality_col = get_quality_col(item); let quality_col_img = match item.quality() { Quality::Low => self.imgs.inv_slot_grey, Quality::Common => self.imgs.inv_slot, Quality::Moderate => self.imgs.inv_slot_green, Quality::High => self.imgs.inv_slot_blue, Quality::Epic => self.imgs.inv_slot_purple, Quality::Legendary => self.imgs.inv_slot_gold, Quality::Artifact => self.imgs.inv_slot_orange, _ => self.imgs.inv_slot_red, }; slot_widget .filled_slot(quality_col_img) .with_tooltip( self.tooltip_manager, title, &*desc, &item_tooltip, quality_col, ) .set(state.ids.inv_slots[i], ui); } else { slot_widget.set(state.ids.inv_slots[i], ui); } } // Close button if Button::image(self.imgs.close_btn) .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.bg, 0.0, 0.0) .set(state.ids.bag_close, ui) .was_clicked() { event = Some(Event::Close); } event } }