2020-07-14 20:11:39 +00:00
|
|
|
use super::{
|
|
|
|
img_ids::{Imgs, ImgsRot},
|
2021-02-16 01:05:54 +00:00
|
|
|
item_imgs::{animate_by_pulse, ItemImgs},
|
2021-04-04 13:02:27 +00:00
|
|
|
Show, TEXT_COLOR, TEXT_DULL_RED_COLOR, TEXT_GRAY_COLOR, UI_HIGHLIGHT_0, UI_MAIN,
|
2020-07-14 20:11:39 +00:00
|
|
|
};
|
|
|
|
use crate::{
|
2020-05-25 18:11:39 +00:00
|
|
|
i18n::Localization,
|
2021-04-04 13:02:27 +00:00
|
|
|
ui::{
|
|
|
|
fonts::Fonts, ImageFrame, ItemTooltip, ItemTooltipManager, ItemTooltipable, Tooltip,
|
|
|
|
TooltipManager, Tooltipable,
|
|
|
|
},
|
2020-07-14 20:11:39 +00:00
|
|
|
};
|
|
|
|
use client::{self, Client};
|
2021-02-16 01:05:54 +00:00
|
|
|
use common::{
|
|
|
|
assets::AssetExt,
|
|
|
|
comp::{
|
2021-04-04 13:02:27 +00:00
|
|
|
item::{
|
|
|
|
ItemDef, ItemDesc, ItemKind, ItemTag, MaterialStatManifest, Quality, TagExampleInfo,
|
|
|
|
},
|
2021-02-16 01:05:54 +00:00
|
|
|
Inventory,
|
|
|
|
},
|
|
|
|
recipe::RecipeInput,
|
2020-10-07 02:23:20 +00:00
|
|
|
};
|
2020-07-14 20:11:39 +00:00
|
|
|
use conrod_core::{
|
|
|
|
color,
|
|
|
|
widget::{self, Button, Image, Rectangle, Scrollbar, Text},
|
|
|
|
widget_ids, Color, Colorable, Labelable, Positionable, Sizeable, Widget, WidgetCommon,
|
|
|
|
};
|
2021-02-16 01:05:54 +00:00
|
|
|
use std::sync::Arc;
|
|
|
|
|
2021-04-04 13:02:27 +00:00
|
|
|
use strum::IntoEnumIterator;
|
|
|
|
use strum_macros::EnumIter;
|
|
|
|
|
2020-07-14 20:11:39 +00:00
|
|
|
widget_ids! {
|
|
|
|
pub struct Ids {
|
|
|
|
window,
|
|
|
|
window_frame,
|
|
|
|
close,
|
|
|
|
icon,
|
|
|
|
title_main,
|
|
|
|
title_rec,
|
|
|
|
align_rec,
|
|
|
|
scrollbar_rec,
|
|
|
|
title_ing,
|
|
|
|
align_ing,
|
|
|
|
scrollbar_ing,
|
|
|
|
btn_craft,
|
|
|
|
recipe_names[],
|
|
|
|
recipe_img_frame[],
|
|
|
|
recipe_img[],
|
|
|
|
ingredients[],
|
|
|
|
ingredient_frame[],
|
|
|
|
ingredient_img[],
|
|
|
|
req_text[],
|
|
|
|
ingredients_txt,
|
|
|
|
output_img_frame,
|
|
|
|
output_img,
|
2020-10-07 02:23:20 +00:00
|
|
|
output_amount,
|
2021-04-04 13:02:27 +00:00
|
|
|
category_bgs[],
|
|
|
|
category_tabs[],
|
|
|
|
category_imgs[],
|
2020-07-14 20:11:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum Event {
|
|
|
|
CraftRecipe(String),
|
2021-04-04 13:02:27 +00:00
|
|
|
ChangeCraftingTab(SelectedCraftingTab),
|
2020-07-14 20:11:39 +00:00
|
|
|
Close,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(WidgetCommon)]
|
|
|
|
pub struct Crafting<'a> {
|
|
|
|
client: &'a Client,
|
|
|
|
imgs: &'a Imgs,
|
2020-05-25 18:11:39 +00:00
|
|
|
fonts: &'a Fonts,
|
|
|
|
localized_strings: &'a Localization,
|
2021-02-16 01:05:54 +00:00
|
|
|
pulse: f32,
|
2020-07-14 20:11:39 +00:00
|
|
|
rot_imgs: &'a ImgsRot,
|
2021-03-23 01:40:54 +00:00
|
|
|
item_tooltip_manager: &'a mut ItemTooltipManager,
|
2020-07-14 20:11:39 +00:00
|
|
|
item_imgs: &'a ItemImgs,
|
|
|
|
inventory: &'a Inventory,
|
2021-02-25 19:04:09 +00:00
|
|
|
msm: &'a MaterialStatManifest,
|
2020-07-14 20:11:39 +00:00
|
|
|
#[conrod(common_builder)]
|
|
|
|
common: widget::CommonBuilder,
|
2021-04-04 13:02:27 +00:00
|
|
|
tooltip_manager: &'a mut TooltipManager,
|
|
|
|
show: &'a mut Show,
|
2020-07-14 20:11:39 +00:00
|
|
|
}
|
|
|
|
#[allow(clippy::too_many_arguments)]
|
|
|
|
impl<'a> Crafting<'a> {
|
|
|
|
pub fn new(
|
|
|
|
client: &'a Client,
|
|
|
|
imgs: &'a Imgs,
|
2020-05-25 18:11:39 +00:00
|
|
|
fonts: &'a Fonts,
|
|
|
|
localized_strings: &'a Localization,
|
2021-02-16 01:05:54 +00:00
|
|
|
pulse: f32,
|
2020-07-14 20:11:39 +00:00
|
|
|
rot_imgs: &'a ImgsRot,
|
2021-03-23 01:40:54 +00:00
|
|
|
item_tooltip_manager: &'a mut ItemTooltipManager,
|
2020-07-14 20:11:39 +00:00
|
|
|
item_imgs: &'a ItemImgs,
|
|
|
|
inventory: &'a Inventory,
|
2021-02-25 19:04:09 +00:00
|
|
|
msm: &'a MaterialStatManifest,
|
2021-04-04 13:02:27 +00:00
|
|
|
tooltip_manager: &'a mut TooltipManager,
|
|
|
|
show: &'a mut Show,
|
2020-07-14 20:11:39 +00:00
|
|
|
) -> Self {
|
|
|
|
Self {
|
|
|
|
client,
|
|
|
|
imgs,
|
|
|
|
fonts,
|
|
|
|
localized_strings,
|
2021-02-16 01:05:54 +00:00
|
|
|
pulse,
|
2020-07-14 20:11:39 +00:00
|
|
|
rot_imgs,
|
2021-03-23 01:40:54 +00:00
|
|
|
item_tooltip_manager,
|
2021-04-04 13:02:27 +00:00
|
|
|
tooltip_manager,
|
2020-07-14 20:11:39 +00:00
|
|
|
item_imgs,
|
|
|
|
inventory,
|
2021-02-25 19:04:09 +00:00
|
|
|
msm,
|
2021-04-04 13:02:27 +00:00
|
|
|
show,
|
2020-07-14 20:11:39 +00:00
|
|
|
common: widget::CommonBuilder::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-04 13:02:27 +00:00
|
|
|
#[derive(Debug, EnumIter, PartialEq)]
|
|
|
|
pub enum SelectedCraftingTab {
|
|
|
|
Armor,
|
|
|
|
Weapon,
|
|
|
|
Food,
|
|
|
|
Dismantle,
|
|
|
|
Potion,
|
|
|
|
Bag,
|
|
|
|
Tool,
|
|
|
|
Utility,
|
|
|
|
Glider,
|
|
|
|
}
|
|
|
|
|
2020-07-14 20:11:39 +00:00
|
|
|
pub struct State {
|
|
|
|
ids: Ids,
|
|
|
|
selected_recipe: Option<String>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Widget for Crafting<'a> {
|
|
|
|
type Event = Vec<Event>;
|
|
|
|
type State = State;
|
|
|
|
type Style = ();
|
|
|
|
|
|
|
|
fn init_state(&self, id_gen: widget::id::Generator) -> Self::State {
|
|
|
|
State {
|
|
|
|
ids: Ids::new(id_gen),
|
|
|
|
selected_recipe: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(clippy::unused_unit)] // TODO: Pending review in #587
|
|
|
|
fn style(&self) -> Self::Style { () }
|
|
|
|
|
|
|
|
fn update(self, args: widget::UpdateArgs<Self>) -> Self::Event {
|
|
|
|
let widget::UpdateArgs { state, ui, .. } = args;
|
|
|
|
|
|
|
|
if state.ids.recipe_names.len() < self.client.recipe_book().iter().len() {
|
|
|
|
state.update(|state| {
|
|
|
|
state.ids.recipe_names.resize(
|
|
|
|
self.client.recipe_book().iter().len(),
|
|
|
|
&mut ui.widget_id_generator(),
|
|
|
|
)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut events = Vec::new();
|
|
|
|
|
|
|
|
// Tooltips
|
2021-03-24 22:17:25 +00:00
|
|
|
let item_tooltip = ItemTooltip::new(
|
2021-03-23 01:40:54 +00:00
|
|
|
{
|
|
|
|
// 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,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
self.client,
|
|
|
|
self.imgs,
|
|
|
|
self.item_imgs,
|
|
|
|
self.pulse,
|
|
|
|
self.msm,
|
2021-03-24 01:05:14 +00:00
|
|
|
self.localized_strings,
|
2021-03-23 01:40:54 +00:00
|
|
|
)
|
|
|
|
.title_font_size(self.fonts.cyri.scale(20))
|
|
|
|
.parent(ui.window)
|
|
|
|
.desc_font_size(self.fonts.cyri.scale(12))
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.desc_text_color(TEXT_COLOR);
|
2021-04-04 13:02:27 +00:00
|
|
|
// Tab tooltips
|
|
|
|
let tabs_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);
|
2021-03-23 01:40:54 +00:00
|
|
|
|
2020-07-14 20:11:39 +00:00
|
|
|
Image::new(self.imgs.crafting_window)
|
|
|
|
.bottom_right_with_margins_on(ui.window, 308.0, 450.0)
|
|
|
|
.color(Some(UI_MAIN))
|
|
|
|
.w_h(422.0, 460.0)
|
2021-04-04 13:02:27 +00:00
|
|
|
.set(state.ids.window, ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
Image::new(self.imgs.crafting_frame)
|
2021-04-04 13:02:27 +00:00
|
|
|
.middle_of(state.ids.window)
|
2020-07-14 20:11:39 +00:00
|
|
|
.color(Some(UI_HIGHLIGHT_0))
|
|
|
|
.w_h(422.0, 460.0)
|
2021-04-04 13:02:27 +00:00
|
|
|
.set(state.ids.window_frame, ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
Image::new(self.imgs.crafting_icon_bordered)
|
|
|
|
.w_h(38.0, 38.0)
|
|
|
|
.top_left_with_margins_on(state.ids.window_frame, 4.0, 4.0)
|
|
|
|
.set(state.ids.icon, ui);
|
|
|
|
// Close Button
|
|
|
|
if Button::image(self.imgs.close_button)
|
|
|
|
.w_h(24.0, 25.0)
|
|
|
|
.hover_image(self.imgs.close_button_hover)
|
|
|
|
.press_image(self.imgs.close_button_press)
|
2021-04-04 13:02:27 +00:00
|
|
|
.top_right_with_margins_on(state.ids.window, 0.0, 0.0)
|
|
|
|
.set(state.ids.close, ui)
|
2020-07-14 20:11:39 +00:00
|
|
|
.was_clicked()
|
|
|
|
{
|
|
|
|
events.push(Event::Close);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Title
|
|
|
|
Text::new(&self.localized_strings.get("hud.crafting"))
|
2021-04-04 13:02:27 +00:00
|
|
|
.mid_top_with_margin_on(state.ids.window_frame, 9.0)
|
2020-07-14 20:11:39 +00:00
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
2020-08-02 23:53:02 +00:00
|
|
|
.font_size(self.fonts.cyri.scale(20))
|
2020-07-14 20:11:39 +00:00
|
|
|
.color(TEXT_COLOR)
|
2021-04-04 13:02:27 +00:00
|
|
|
.set(state.ids.title_main, ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
|
|
|
|
// Alignment
|
|
|
|
Rectangle::fill_with([136.0, 378.0], color::TRANSPARENT)
|
2021-04-04 13:02:27 +00:00
|
|
|
.top_left_with_margins_on(state.ids.window_frame, 74.0, 5.0)
|
2020-08-18 21:32:34 +00:00
|
|
|
.scroll_kids_vertically()
|
2021-04-04 13:02:27 +00:00
|
|
|
.set(state.ids.align_rec, ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
Rectangle::fill_with([274.0, 340.0], color::TRANSPARENT)
|
2021-04-04 13:02:27 +00:00
|
|
|
.top_right_with_margins_on(state.ids.window, 74.0, 5.0)
|
2020-12-24 02:54:18 +00:00
|
|
|
.scroll_kids_vertically()
|
2021-04-04 13:02:27 +00:00
|
|
|
.set(state.ids.align_ing, ui);
|
|
|
|
// Category Tabs
|
|
|
|
if state.ids.category_bgs.len() < SelectedCraftingTab::iter().enumerate().len() {
|
|
|
|
state.update(|s| {
|
|
|
|
s.ids.category_bgs.resize(
|
|
|
|
SelectedCraftingTab::iter().enumerate().len(),
|
|
|
|
&mut ui.widget_id_generator(),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
};
|
|
|
|
if state.ids.category_tabs.len() < SelectedCraftingTab::iter().enumerate().len() {
|
|
|
|
state.update(|s| {
|
|
|
|
s.ids.category_tabs.resize(
|
|
|
|
SelectedCraftingTab::iter().enumerate().len(),
|
|
|
|
&mut ui.widget_id_generator(),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
};
|
|
|
|
if state.ids.category_imgs.len() < SelectedCraftingTab::iter().enumerate().len() {
|
|
|
|
state.update(|s| {
|
|
|
|
s.ids.category_imgs.resize(
|
|
|
|
SelectedCraftingTab::iter().enumerate().len(),
|
|
|
|
&mut ui.widget_id_generator(),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
};
|
|
|
|
let sel_crafting_tab = &self.show.crafting_tab;
|
|
|
|
for i in SelectedCraftingTab::iter().enumerate() {
|
|
|
|
// TODO: i18n!
|
|
|
|
let tab_name = match i.1 {
|
|
|
|
SelectedCraftingTab::Armor => "Armor",
|
|
|
|
SelectedCraftingTab::Dismantle => "Dismantle",
|
|
|
|
SelectedCraftingTab::Food => "Food",
|
|
|
|
SelectedCraftingTab::Glider => "Gliders",
|
|
|
|
SelectedCraftingTab::Potion => "Potions",
|
|
|
|
SelectedCraftingTab::Tool => "Tools",
|
|
|
|
SelectedCraftingTab::Utility => "Utility",
|
|
|
|
SelectedCraftingTab::Weapon => "Weapons",
|
|
|
|
SelectedCraftingTab::Bag => "Weapons",
|
|
|
|
};
|
|
|
|
let tab_img = match i.1 {
|
|
|
|
SelectedCraftingTab::Armor => self.imgs.icon_armor,
|
|
|
|
SelectedCraftingTab::Dismantle => self.imgs.icon_dismantle,
|
|
|
|
SelectedCraftingTab::Food => self.imgs.icon_food,
|
|
|
|
SelectedCraftingTab::Glider => self.imgs.icon_glider,
|
|
|
|
SelectedCraftingTab::Potion => self.imgs.icon_potion,
|
|
|
|
SelectedCraftingTab::Tool => self.imgs.icon_tools,
|
|
|
|
SelectedCraftingTab::Utility => self.imgs.icon_utility,
|
|
|
|
SelectedCraftingTab::Weapon => self.imgs.icon_weapon,
|
|
|
|
SelectedCraftingTab::Bag => self.imgs.icon_bag,
|
|
|
|
};
|
|
|
|
// Button Background
|
|
|
|
let mut bg = Image::new(self.imgs.pixel)
|
|
|
|
.w_h(40.0, 30.0)
|
|
|
|
.color(Some(UI_MAIN));
|
|
|
|
if i.0 == 0 {
|
|
|
|
bg = bg.top_left_with_margins_on(state.ids.window_frame, 50.0, -40.0)
|
|
|
|
} else {
|
|
|
|
bg = bg.down_from(state.ids.category_bgs[i.0 - 1], 0.0)
|
|
|
|
};
|
|
|
|
bg.set(state.ids.category_bgs[i.0], ui);
|
|
|
|
// Category Button
|
|
|
|
if Button::image(if i.1 == *sel_crafting_tab {
|
|
|
|
self.imgs.wpn_icon_border_pressed
|
|
|
|
} else {
|
|
|
|
self.imgs.wpn_icon_border
|
|
|
|
})
|
|
|
|
.wh_of(state.ids.category_bgs[i.0])
|
|
|
|
.middle_of(state.ids.category_bgs[i.0])
|
|
|
|
.hover_image(if i.1 == *sel_crafting_tab {
|
|
|
|
self.imgs.wpn_icon_border_pressed
|
|
|
|
} else {
|
|
|
|
self.imgs.wpn_icon_border_mo
|
|
|
|
})
|
|
|
|
.press_image(if i.1 == *sel_crafting_tab {
|
|
|
|
self.imgs.wpn_icon_border_pressed
|
|
|
|
} else {
|
|
|
|
self.imgs.wpn_icon_border_press
|
|
|
|
})
|
|
|
|
.with_tooltip(
|
|
|
|
self.tooltip_manager,
|
|
|
|
tab_name,
|
|
|
|
"",
|
|
|
|
&tabs_tooltip,
|
|
|
|
TEXT_COLOR,
|
|
|
|
)
|
|
|
|
.set(state.ids.category_tabs[i.0], ui)
|
|
|
|
.was_clicked()
|
|
|
|
{
|
|
|
|
events.push(Event::ChangeCraftingTab(i.1))
|
|
|
|
};
|
|
|
|
// Tab images
|
|
|
|
Image::new(tab_img)
|
|
|
|
.middle_of(state.ids.category_tabs[i.0])
|
|
|
|
.w_h(20.0, 20.0)
|
|
|
|
.graphics_for(state.ids.category_tabs[i.0])
|
|
|
|
.set(state.ids.category_imgs[i.0], ui);
|
|
|
|
}
|
|
|
|
|
2020-07-14 20:11:39 +00:00
|
|
|
let client = &self.client;
|
2021-02-09 02:51:26 +00:00
|
|
|
// First available recipes, then unavailable ones, each alphabetically
|
|
|
|
// In the triples, "name" is the recipe book key, and "recipe.output.0.name()"
|
|
|
|
// is the display name (as stored in the item descriptors)
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
|
|
|
|
enum RecipeIngredientQuantity {
|
|
|
|
All,
|
|
|
|
Some,
|
|
|
|
None,
|
|
|
|
}
|
|
|
|
let mut ordered_recipes: Vec<_> = self
|
2020-07-14 20:11:39 +00:00
|
|
|
.client
|
|
|
|
.recipe_book()
|
|
|
|
.iter()
|
2021-02-09 02:51:26 +00:00
|
|
|
.map(|(name, recipe)| {
|
2021-02-16 01:05:54 +00:00
|
|
|
let at_least_some_ingredients = recipe.inputs.iter().any(|(input, amount)| {
|
|
|
|
*amount > 0
|
|
|
|
&& self.inventory.slots().any(|slot| {
|
|
|
|
slot.as_ref()
|
|
|
|
.map(|item| item.matches_recipe_input(input))
|
|
|
|
.unwrap_or(false)
|
|
|
|
})
|
|
|
|
});
|
2021-02-09 02:51:26 +00:00
|
|
|
let state = if client.available_recipes().contains(name.as_str()) {
|
|
|
|
RecipeIngredientQuantity::All
|
|
|
|
} else if at_least_some_ingredients {
|
|
|
|
RecipeIngredientQuantity::Some
|
|
|
|
} else {
|
|
|
|
RecipeIngredientQuantity::None
|
|
|
|
};
|
|
|
|
(name, recipe, state)
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
ordered_recipes.sort_by_key(|(_, recipe, state)| (*state, recipe.output.0.name()));
|
2020-07-14 20:11:39 +00:00
|
|
|
match &state.selected_recipe {
|
|
|
|
None => {},
|
|
|
|
Some(recipe) => {
|
|
|
|
let can_perform = client.available_recipes().contains(recipe.as_str());
|
|
|
|
// Ingredients Text
|
|
|
|
Text::new(&self.localized_strings.get("hud.crafting.ingredients"))
|
|
|
|
.top_left_with_margins_on(state.ids.align_ing, 10.0, 5.0)
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(self.fonts.cyri.scale(18))
|
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.set(state.ids.ingredients_txt, ui);
|
|
|
|
// Craft button
|
|
|
|
if Button::image(self.imgs.button)
|
|
|
|
.w_h(105.0, 25.0)
|
|
|
|
.hover_image(
|
|
|
|
can_perform
|
|
|
|
.then_some(self.imgs.button_hover)
|
|
|
|
.unwrap_or(self.imgs.button),
|
|
|
|
)
|
|
|
|
.press_image(
|
|
|
|
can_perform
|
|
|
|
.then_some(self.imgs.button_press)
|
|
|
|
.unwrap_or(self.imgs.button),
|
|
|
|
)
|
|
|
|
.label(&self.localized_strings.get("hud.crafting.craft"))
|
|
|
|
.label_y(conrod_core::position::Relative::Scalar(1.0))
|
|
|
|
.label_color(can_perform.then_some(TEXT_COLOR).unwrap_or(TEXT_GRAY_COLOR))
|
|
|
|
.label_font_size(self.fonts.cyri.scale(12))
|
|
|
|
.label_font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.image_color(can_perform.then_some(TEXT_COLOR).unwrap_or(TEXT_GRAY_COLOR))
|
2021-04-04 13:02:27 +00:00
|
|
|
.mid_bottom_with_margin_on(state.ids.align_ing, -31.0)
|
|
|
|
.parent(state.ids.window_frame)
|
|
|
|
.set(state.ids.btn_craft, ui)
|
2020-07-14 20:11:39 +00:00
|
|
|
.was_clicked()
|
|
|
|
{
|
|
|
|
events.push(Event::CraftRecipe(recipe.clone()));
|
|
|
|
}
|
|
|
|
// Result Image BG
|
2020-10-07 02:23:20 +00:00
|
|
|
let quality_col_img = if let Some(recipe) = state
|
|
|
|
.selected_recipe
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|r| self.client.recipe_book().get(r.as_str()))
|
|
|
|
{
|
|
|
|
match recipe.output.0.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,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
self.imgs.inv_slot
|
|
|
|
};
|
|
|
|
Image::new(quality_col_img)
|
2020-07-14 20:11:39 +00:00
|
|
|
.w_h(60.0, 60.0)
|
|
|
|
.top_right_with_margins_on(state.ids.align_ing, 15.0, 10.0)
|
2021-04-04 13:02:27 +00:00
|
|
|
.parent(state.ids.align_ing)
|
|
|
|
.set(state.ids.output_img_frame, ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
|
|
|
|
if let Some(recipe) = state
|
|
|
|
.selected_recipe
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|r| self.client.recipe_book().get(r.as_str()))
|
|
|
|
{
|
|
|
|
let output_text = format!("x{}", &recipe.output.1.to_string());
|
|
|
|
// Output Image
|
2021-02-16 01:05:54 +00:00
|
|
|
Button::image(animate_by_pulse(
|
|
|
|
&self
|
|
|
|
.item_imgs
|
|
|
|
.img_ids_or_not_found_img((&*recipe.output.0).into()),
|
|
|
|
self.pulse,
|
|
|
|
))
|
2020-07-14 20:11:39 +00:00
|
|
|
.w_h(55.0, 55.0)
|
2020-08-21 20:37:08 +00:00
|
|
|
.label(&output_text)
|
2020-07-14 20:11:39 +00:00
|
|
|
.label_color(TEXT_COLOR)
|
|
|
|
.label_font_size(self.fonts.cyri.scale(14))
|
|
|
|
.label_font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.label_y(conrod_core::position::Relative::Scalar(-24.0))
|
|
|
|
.label_x(conrod_core::position::Relative::Scalar(24.0))
|
2020-10-07 02:23:20 +00:00
|
|
|
.middle_of(state.ids.output_img_frame)
|
2021-03-23 01:40:54 +00:00
|
|
|
.with_item_tooltip(
|
|
|
|
self.item_tooltip_manager,
|
|
|
|
&*recipe.output.0,
|
2021-04-02 19:27:02 +00:00
|
|
|
&None,
|
2021-03-24 22:17:25 +00:00
|
|
|
&item_tooltip,
|
2020-10-07 02:23:20 +00:00
|
|
|
)
|
2020-07-14 20:11:39 +00:00
|
|
|
.set(state.ids.output_img, ui);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recipe list
|
2021-04-04 13:02:27 +00:00
|
|
|
for (i, (name, recipe, quantity)) in ordered_recipes
|
|
|
|
.into_iter()
|
|
|
|
.filter(|(_name, recipe, _quantity)| match &self.show.crafting_tab {
|
|
|
|
SelectedCraftingTab::Food => recipe.output.0.tags().contains(&ItemTag::Food),
|
|
|
|
SelectedCraftingTab::Armor => match recipe.output.0.kind() {
|
|
|
|
ItemKind::Armor(_) => !recipe.output.0.tags().contains(&ItemTag::Bag),
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
SelectedCraftingTab::Glider => {
|
|
|
|
matches!(recipe.output.0.kind(), ItemKind::Glider(_))
|
|
|
|
},
|
|
|
|
SelectedCraftingTab::Potion => recipe.output.0.tags().contains(&ItemTag::Potion),
|
|
|
|
SelectedCraftingTab::Bag => recipe.output.0.tags().contains(&ItemTag::Bag),
|
|
|
|
SelectedCraftingTab::Tool => {
|
|
|
|
recipe.output.0.tags().contains(&ItemTag::CraftingTool)
|
|
|
|
},
|
|
|
|
SelectedCraftingTab::Utility => recipe.output.0.tags().contains(&ItemTag::Utility),
|
|
|
|
SelectedCraftingTab::Weapon => match recipe.output.0.kind() {
|
|
|
|
ItemKind::Tool(_) => !recipe.output.0.tags().contains(&ItemTag::CraftingTool),
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
SelectedCraftingTab::Dismantle => match recipe.output.0.kind() {
|
|
|
|
ItemKind::Ingredient { .. } => {
|
|
|
|
!recipe.output.0.tags().contains(&ItemTag::CraftingTool)
|
|
|
|
},
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
.enumerate()
|
|
|
|
{
|
2020-07-14 20:11:39 +00:00
|
|
|
let button = Button::image(
|
|
|
|
if state
|
|
|
|
.selected_recipe
|
|
|
|
.as_ref()
|
|
|
|
.map(|s| s != name)
|
|
|
|
.unwrap_or(false)
|
|
|
|
{
|
|
|
|
self.imgs.nothing
|
|
|
|
} else {
|
|
|
|
match state.selected_recipe {
|
|
|
|
None => self.imgs.nothing,
|
|
|
|
Some(_) => self.imgs.selection,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
// Recipe Button
|
|
|
|
let button = if i == 0 {
|
|
|
|
button.mid_top_with_margin_on(state.ids.align_rec, 2.0)
|
|
|
|
} else {
|
|
|
|
button.mid_bottom_with_margin_on(state.ids.recipe_names[i - 1], -25.0)
|
|
|
|
};
|
2021-02-09 02:51:26 +00:00
|
|
|
let text_color = match quantity {
|
|
|
|
RecipeIngredientQuantity::All => TEXT_COLOR,
|
|
|
|
RecipeIngredientQuantity::Some => TEXT_GRAY_COLOR,
|
|
|
|
RecipeIngredientQuantity::None => TEXT_DULL_RED_COLOR,
|
|
|
|
};
|
2020-07-14 20:11:39 +00:00
|
|
|
if button
|
|
|
|
.label(recipe.output.0.name())
|
|
|
|
.w_h(130.0, 20.0)
|
|
|
|
.hover_image(self.imgs.selection_hover)
|
|
|
|
.press_image(self.imgs.selection_press)
|
2021-02-09 02:51:26 +00:00
|
|
|
.label_color(text_color)
|
2020-07-14 20:11:39 +00:00
|
|
|
.label_font_size(self.fonts.cyri.scale(12))
|
|
|
|
.label_font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.label_y(conrod_core::position::Relative::Scalar(2.0))
|
|
|
|
.set(state.ids.recipe_names[i], ui)
|
|
|
|
.was_clicked()
|
|
|
|
{
|
|
|
|
if state
|
|
|
|
.selected_recipe
|
|
|
|
.as_ref()
|
|
|
|
.map(|s| s == name)
|
|
|
|
.unwrap_or(false)
|
|
|
|
{
|
|
|
|
state.update(|s| s.selected_recipe = None);
|
|
|
|
} else {
|
|
|
|
state.update(|s| s.selected_recipe = Some(name.clone()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Ingredients
|
|
|
|
if let Some(recipe) = state
|
|
|
|
.selected_recipe
|
|
|
|
.as_ref()
|
|
|
|
.and_then(|r| self.client.recipe_book().get(r.as_str()))
|
|
|
|
{
|
|
|
|
// Title
|
2020-09-17 23:02:14 +00:00
|
|
|
Text::new(&recipe.output.0.name())
|
2020-07-14 20:11:39 +00:00
|
|
|
.mid_top_with_margin_on(state.ids.align_ing, -22.0)
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(self.fonts.cyri.scale(14))
|
|
|
|
.color(TEXT_COLOR)
|
|
|
|
.parent(state.ids.window)
|
|
|
|
.set(state.ids.title_ing, ui);
|
|
|
|
// Ingredient images with tooltip
|
|
|
|
if state.ids.ingredient_frame.len() < recipe.inputs().len() {
|
|
|
|
state.update(|state| {
|
|
|
|
state
|
|
|
|
.ids
|
|
|
|
.ingredient_frame
|
|
|
|
.resize(recipe.inputs().len(), &mut ui.widget_id_generator())
|
|
|
|
});
|
|
|
|
};
|
|
|
|
if state.ids.ingredients.len() < recipe.inputs().len() {
|
|
|
|
state.update(|state| {
|
|
|
|
state
|
|
|
|
.ids
|
|
|
|
.ingredients
|
|
|
|
.resize(recipe.inputs().len(), &mut ui.widget_id_generator())
|
|
|
|
});
|
|
|
|
};
|
|
|
|
if state.ids.ingredient_img.len() < recipe.inputs().len() {
|
|
|
|
state.update(|state| {
|
|
|
|
state
|
|
|
|
.ids
|
|
|
|
.ingredient_img
|
|
|
|
.resize(recipe.inputs().len(), &mut ui.widget_id_generator())
|
|
|
|
});
|
|
|
|
};
|
|
|
|
if state.ids.req_text.len() < recipe.inputs().len() {
|
|
|
|
state.update(|state| {
|
|
|
|
state
|
|
|
|
.ids
|
|
|
|
.req_text
|
|
|
|
.resize(recipe.inputs().len(), &mut ui.widget_id_generator())
|
|
|
|
});
|
|
|
|
};
|
|
|
|
// Widget generation for every ingredient
|
2021-02-16 01:05:54 +00:00
|
|
|
for (i, (recipe_input, amount)) in recipe.inputs.iter().enumerate() {
|
|
|
|
let item_def = match recipe_input {
|
|
|
|
RecipeInput::Item(item_def) => Arc::clone(item_def),
|
|
|
|
RecipeInput::Tag(tag) => Arc::<ItemDef>::load_expect_cloned(
|
|
|
|
&self
|
|
|
|
.inventory
|
|
|
|
.slots()
|
|
|
|
.filter_map(|slot| {
|
|
|
|
slot.as_ref().and_then(|item| {
|
|
|
|
if item.matches_recipe_input(recipe_input) {
|
|
|
|
Some(item.item_definition_id().to_string())
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.next()
|
|
|
|
.unwrap_or_else(|| tag.exemplar_identifier().to_string()),
|
|
|
|
),
|
|
|
|
};
|
|
|
|
|
2020-07-14 20:11:39 +00:00
|
|
|
// Grey color for images and text if their amount is too low to craft the item
|
2021-02-16 01:05:54 +00:00
|
|
|
let item_count_in_inventory = self.inventory.item_count(&*item_def);
|
2020-10-15 12:36:59 +00:00
|
|
|
let col = if item_count_in_inventory >= u64::from(*amount.max(&1)) {
|
2020-07-14 20:11:39 +00:00
|
|
|
TEXT_COLOR
|
|
|
|
} else {
|
|
|
|
TEXT_DULL_RED_COLOR
|
|
|
|
};
|
|
|
|
// Slot BG
|
|
|
|
let frame_pos = if i == 0 {
|
|
|
|
state.ids.ingredients_txt
|
|
|
|
} else {
|
|
|
|
state.ids.ingredient_frame[i - 1]
|
|
|
|
};
|
|
|
|
// add a larger offset for the the first ingredient and the "Required Text for
|
|
|
|
// Catalysts/Tools"
|
|
|
|
let frame_offset = if i == 0 {
|
|
|
|
10.0
|
|
|
|
} else if *amount == 0 {
|
|
|
|
5.0
|
|
|
|
} else {
|
|
|
|
0.0
|
|
|
|
};
|
2020-10-07 02:23:20 +00:00
|
|
|
let quality_col_img = match &item_def.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,
|
|
|
|
};
|
|
|
|
let frame = Image::new(quality_col_img).w_h(25.0, 25.0);
|
2020-07-14 20:11:39 +00:00
|
|
|
let frame = if *amount == 0 {
|
|
|
|
frame.down_from(state.ids.req_text[i], 10.0 + frame_offset)
|
|
|
|
} else {
|
|
|
|
frame.down_from(frame_pos, 10.0 + frame_offset)
|
|
|
|
};
|
|
|
|
frame.set(state.ids.ingredient_frame[i], ui);
|
|
|
|
//Item Image
|
2021-02-16 01:05:54 +00:00
|
|
|
Button::image(animate_by_pulse(
|
|
|
|
&self.item_imgs.img_ids_or_not_found_img((&*item_def).into()),
|
|
|
|
self.pulse,
|
|
|
|
))
|
|
|
|
.w_h(22.0, 22.0)
|
|
|
|
.middle_of(state.ids.ingredient_frame[i])
|
2021-04-02 19:27:02 +00:00
|
|
|
.with_item_tooltip(self.item_tooltip_manager, &*item_def, &None, &item_tooltip)
|
2021-02-16 01:05:54 +00:00
|
|
|
.set(state.ids.ingredient_img[i], ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
// Ingredients text and amount
|
|
|
|
// Don't show inventory amounts above 999 to avoid the widget clipping
|
2020-08-20 23:55:18 +00:00
|
|
|
let over9k = "99+";
|
2020-09-17 23:02:14 +00:00
|
|
|
let in_inv: &str = &item_count_in_inventory.to_string();
|
2020-07-14 20:11:39 +00:00
|
|
|
// Show Ingredients
|
|
|
|
// Align "Required" Text below last ingredient
|
|
|
|
if *amount == 0 {
|
|
|
|
// Catalysts/Tools
|
|
|
|
Text::new(&self.localized_strings.get("hud.crafting.tool_cata"))
|
|
|
|
.down_from(state.ids.ingredient_frame[i - 1], 20.0)
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(self.fonts.cyri.scale(14))
|
2020-10-12 15:41:17 +00:00
|
|
|
.color(TEXT_COLOR)
|
2020-07-14 20:11:39 +00:00
|
|
|
.set(state.ids.req_text[i], ui);
|
2020-09-26 15:20:46 +00:00
|
|
|
Text::new(&item_def.name())
|
2020-07-14 20:11:39 +00:00
|
|
|
.right_from(state.ids.ingredient_frame[i], 10.0)
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(self.fonts.cyri.scale(14))
|
|
|
|
.color(col)
|
|
|
|
.set(state.ids.ingredients[i], ui);
|
|
|
|
} else {
|
|
|
|
// Ingredients
|
2021-02-16 01:05:54 +00:00
|
|
|
let name = match recipe_input {
|
|
|
|
RecipeInput::Item(_) => item_def.name().to_string(),
|
|
|
|
RecipeInput::Tag(tag) => format!("Any {}", tag.name()),
|
|
|
|
};
|
2020-07-14 20:11:39 +00:00
|
|
|
let input = format!(
|
|
|
|
"{}x {} ({})",
|
|
|
|
amount,
|
2021-02-16 01:05:54 +00:00
|
|
|
name,
|
2020-09-17 23:02:14 +00:00
|
|
|
if item_count_in_inventory > 99 {
|
2020-07-14 20:11:39 +00:00
|
|
|
over9k
|
|
|
|
} else {
|
|
|
|
in_inv
|
|
|
|
}
|
|
|
|
);
|
|
|
|
// Ingredient Text
|
|
|
|
Text::new(&input)
|
|
|
|
.right_from(state.ids.ingredient_frame[i], 10.0)
|
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
2020-08-20 23:55:18 +00:00
|
|
|
.font_size(self.fonts.cyri.scale(12))
|
2020-07-14 20:11:39 +00:00
|
|
|
.color(col)
|
|
|
|
.set(state.ids.ingredients[i], ui);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Scrollbars
|
2021-04-04 13:02:27 +00:00
|
|
|
Scrollbar::y_axis(state.ids.align_rec)
|
2020-07-14 20:11:39 +00:00
|
|
|
.thickness(5.0)
|
|
|
|
.rgba(0.33, 0.33, 0.33, 1.0)
|
2021-04-04 13:02:27 +00:00
|
|
|
.set(state.ids.scrollbar_rec, ui);
|
|
|
|
Scrollbar::y_axis(state.ids.align_ing)
|
2020-07-14 20:11:39 +00:00
|
|
|
.thickness(5.0)
|
|
|
|
.rgba(0.33, 0.33, 0.33, 1.0)
|
2021-04-04 13:02:27 +00:00
|
|
|
.set(state.ids.scrollbar_ing, ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
|
|
|
|
// Title Recipes and Ingredients
|
|
|
|
Text::new(&self.localized_strings.get("hud.crafting.recipes"))
|
2021-04-04 13:02:27 +00:00
|
|
|
.mid_top_with_margin_on(state.ids.align_rec, -22.0)
|
2020-07-14 20:11:39 +00:00
|
|
|
.font_id(self.fonts.cyri.conrod_id)
|
|
|
|
.font_size(self.fonts.cyri.scale(14))
|
|
|
|
.color(TEXT_COLOR)
|
2021-04-04 13:02:27 +00:00
|
|
|
.parent(state.ids.window)
|
|
|
|
.set(state.ids.title_rec, ui);
|
2020-07-14 20:11:39 +00:00
|
|
|
|
|
|
|
events
|
|
|
|
}
|
|
|
|
}
|