mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
Initial implementation of starting site choice
This commit is contained in:
parent
d1a7884ac8
commit
acec45b756
@ -19,5 +19,8 @@ char_selection-eye_color = Eye Color
|
|||||||
char_selection-skin = Skin
|
char_selection-skin = Skin
|
||||||
char_selection-eyeshape = Eye Details
|
char_selection-eyeshape = Eye Details
|
||||||
char_selection-accessories = Accessories
|
char_selection-accessories = Accessories
|
||||||
|
char_selection-starting_site = Starting Site
|
||||||
|
char_selection-starting_site_name = Name: { $name }
|
||||||
|
char_selection-starting_site_kind = Kind: { $kind }
|
||||||
char_selection-create_info_name = Your Character needs a name!
|
char_selection-create_info_name = Your Character needs a name!
|
||||||
char_selection-version_mismatch = WARNING! This server is running a different, possibly incompatible game version. Please update your game.
|
char_selection-version_mismatch = WARNING! This server is running a different, possibly incompatible game version. Please update your game.
|
||||||
|
@ -189,6 +189,7 @@ impl BotClient {
|
|||||||
Some("common.items.weapons.sword.starter".to_string()),
|
Some("common.items.weapons.sword.starter".to_string()),
|
||||||
None,
|
None,
|
||||||
body.into(),
|
body.into(),
|
||||||
|
None,
|
||||||
);
|
);
|
||||||
client.load_character_list();
|
client.load_character_list();
|
||||||
}
|
}
|
||||||
|
@ -148,6 +148,7 @@ fn run_client(
|
|||||||
Some("common.items.weapons.sword.starter".into()),
|
Some("common.items.weapons.sword.starter".into()),
|
||||||
None,
|
None,
|
||||||
body(),
|
body(),
|
||||||
|
None,
|
||||||
);
|
);
|
||||||
|
|
||||||
client.load_character_list();
|
client.load_character_list();
|
||||||
|
@ -951,6 +951,7 @@ impl Client {
|
|||||||
mainhand: Option<String>,
|
mainhand: Option<String>,
|
||||||
offhand: Option<String>,
|
offhand: Option<String>,
|
||||||
body: comp::Body,
|
body: comp::Body,
|
||||||
|
start_site: Option<SiteId>,
|
||||||
) {
|
) {
|
||||||
self.character_list.loading = true;
|
self.character_list.loading = true;
|
||||||
self.send_msg(ClientGeneral::CreateCharacter {
|
self.send_msg(ClientGeneral::CreateCharacter {
|
||||||
@ -958,6 +959,7 @@ impl Client {
|
|||||||
mainhand,
|
mainhand,
|
||||||
offhand,
|
offhand,
|
||||||
body,
|
body,
|
||||||
|
start_site,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,6 +48,7 @@ pub enum ClientGeneral {
|
|||||||
mainhand: Option<String>,
|
mainhand: Option<String>,
|
||||||
offhand: Option<String>,
|
offhand: Option<String>,
|
||||||
body: comp::Body,
|
body: comp::Body,
|
||||||
|
start_site: Option<SiteId>,
|
||||||
},
|
},
|
||||||
DeleteCharacter(CharacterId),
|
DeleteCharacter(CharacterId),
|
||||||
EditCharacter {
|
EditCharacter {
|
||||||
|
@ -1,7 +1,10 @@
|
|||||||
use crate::persistence::{character_updater::CharacterUpdater, PersistedComponents};
|
use crate::persistence::{character_updater::CharacterUpdater, PersistedComponents};
|
||||||
use common::{
|
use common::{
|
||||||
character::CharacterId,
|
character::CharacterId,
|
||||||
comp::{inventory::loadout_builder::LoadoutBuilder, Body, Inventory, Item, SkillSet, Stats},
|
comp::{
|
||||||
|
inventory::loadout_builder::LoadoutBuilder, Body, Inventory, Item, SkillSet, Stats,
|
||||||
|
Waypoint,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
use specs::{Entity, WriteExpect};
|
use specs::{Entity, WriteExpect};
|
||||||
|
|
||||||
@ -32,6 +35,7 @@ pub fn create_character(
|
|||||||
character_offhand: Option<String>,
|
character_offhand: Option<String>,
|
||||||
body: Body,
|
body: Body,
|
||||||
character_updater: &mut WriteExpect<'_, CharacterUpdater>,
|
character_updater: &mut WriteExpect<'_, CharacterUpdater>,
|
||||||
|
waypoint: Option<Waypoint>,
|
||||||
) -> Result<(), CreationError> {
|
) -> Result<(), CreationError> {
|
||||||
// quick fix whitelist validation for now; eventually replace the
|
// quick fix whitelist validation for now; eventually replace the
|
||||||
// `Option<String>` with an index into a server-provided list of starter
|
// `Option<String>` with an index into a server-provided list of starter
|
||||||
@ -63,7 +67,6 @@ pub fn create_character(
|
|||||||
.push(Item::new_from_asset_expect("common.items.food.cheese"))
|
.push(Item::new_from_asset_expect("common.items.food.cheese"))
|
||||||
.expect("Inventory has at least 1 slot left!");
|
.expect("Inventory has at least 1 slot left!");
|
||||||
|
|
||||||
let waypoint = None;
|
|
||||||
let map_marker = None;
|
let map_marker = None;
|
||||||
|
|
||||||
character_updater.create_character(entity, player_uuid, character_alias, PersistedComponents {
|
character_updater.create_character(entity, player_uuid, character_alias, PersistedComponents {
|
||||||
|
@ -1,3 +1,8 @@
|
|||||||
|
#[cfg(not(feature = "worldgen"))]
|
||||||
|
use crate::test_world::{IndexOwned, World};
|
||||||
|
#[cfg(feature = "worldgen")]
|
||||||
|
use world::{IndexOwned, World};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
automod::AutoMod,
|
automod::AutoMod,
|
||||||
character_creator,
|
character_creator,
|
||||||
@ -7,15 +12,17 @@ use crate::{
|
|||||||
EditableSettings,
|
EditableSettings,
|
||||||
};
|
};
|
||||||
use common::{
|
use common::{
|
||||||
comp::{Admin, AdminRole, ChatType, Player, UnresolvedChatMsg},
|
comp::{Admin, AdminRole, ChatType, Player, UnresolvedChatMsg, Waypoint},
|
||||||
event::{EventBus, ServerEvent},
|
event::{EventBus, ServerEvent},
|
||||||
|
resources::Time,
|
||||||
|
terrain::TerrainChunkSize,
|
||||||
uid::Uid,
|
uid::Uid,
|
||||||
};
|
};
|
||||||
use common_ecs::{Job, Origin, Phase, System};
|
use common_ecs::{Job, Origin, Phase, System};
|
||||||
use common_net::msg::{ClientGeneral, ServerGeneral};
|
use common_net::msg::{ClientGeneral, ServerGeneral};
|
||||||
use specs::{Entities, Join, Read, ReadExpect, ReadStorage, WriteExpect, WriteStorage};
|
use specs::{Entities, Join, Read, ReadExpect, ReadStorage, WriteExpect, WriteStorage};
|
||||||
use std::sync::{atomic::Ordering, Arc};
|
use std::sync::{atomic::Ordering, Arc};
|
||||||
use tracing::debug;
|
use tracing::{debug, error};
|
||||||
|
|
||||||
impl Sys {
|
impl Sys {
|
||||||
fn handle_client_character_screen_msg(
|
fn handle_client_character_screen_msg(
|
||||||
@ -32,6 +39,9 @@ impl Sys {
|
|||||||
censor: &ReadExpect<'_, Arc<censor::Censor>>,
|
censor: &ReadExpect<'_, Arc<censor::Censor>>,
|
||||||
automod: &AutoMod,
|
automod: &AutoMod,
|
||||||
msg: ClientGeneral,
|
msg: ClientGeneral,
|
||||||
|
time: Time,
|
||||||
|
index: &ReadExpect<'_, IndexOwned>,
|
||||||
|
world: &ReadExpect<'_, Arc<World>>,
|
||||||
) -> Result<(), crate::error::Error> {
|
) -> Result<(), crate::error::Error> {
|
||||||
let mut send_join_messages = || -> Result<(), crate::error::Error> {
|
let mut send_join_messages = || -> Result<(), crate::error::Error> {
|
||||||
// Give the player a welcome message
|
// Give the player a welcome message
|
||||||
@ -135,6 +145,7 @@ impl Sys {
|
|||||||
mainhand,
|
mainhand,
|
||||||
offhand,
|
offhand,
|
||||||
body,
|
body,
|
||||||
|
start_site,
|
||||||
} => {
|
} => {
|
||||||
if censor.check(&alias) {
|
if censor.check(&alias) {
|
||||||
debug!(?alias, "denied alias as it contained a banned word");
|
debug!(?alias, "denied alias as it contained a banned word");
|
||||||
@ -151,6 +162,22 @@ impl Sys {
|
|||||||
offhand.clone(),
|
offhand.clone(),
|
||||||
body,
|
body,
|
||||||
character_updater,
|
character_updater,
|
||||||
|
start_site.and_then(|site_idx| {
|
||||||
|
// TODO: This corresponds to the ID generation logic in `world/src/lib.rs`
|
||||||
|
// Really, we should have a way to consistently refer to sites, but that's a job for rtsim2
|
||||||
|
// and the site changes that it will require. Until then, this code is very hacky.
|
||||||
|
world.civs().sites.iter()
|
||||||
|
.find(|(_, site)| site.site_tmp.map(|i| i.id()) == Some(site_idx))
|
||||||
|
.map(Some)
|
||||||
|
.unwrap_or_else(|| {
|
||||||
|
error!("Tried to create character with starting site index {}, but such a site does not exist", site_idx);
|
||||||
|
None
|
||||||
|
})
|
||||||
|
.map(|(_, site)| {
|
||||||
|
let wpos2d = TerrainChunkSize::center_wpos(site.center);
|
||||||
|
Waypoint::new(world.find_accessible_pos(index.as_index_ref(), wpos2d, true), time)
|
||||||
|
})
|
||||||
|
}),
|
||||||
) {
|
) {
|
||||||
debug!(
|
debug!(
|
||||||
?error,
|
?error,
|
||||||
@ -226,6 +253,9 @@ impl<'a> System<'a> for Sys {
|
|||||||
ReadExpect<'a, EditableSettings>,
|
ReadExpect<'a, EditableSettings>,
|
||||||
ReadExpect<'a, Arc<censor::Censor>>,
|
ReadExpect<'a, Arc<censor::Censor>>,
|
||||||
ReadExpect<'a, AutoMod>,
|
ReadExpect<'a, AutoMod>,
|
||||||
|
ReadExpect<'a, Time>,
|
||||||
|
ReadExpect<'a, IndexOwned>,
|
||||||
|
ReadExpect<'a, Arc<World>>,
|
||||||
);
|
);
|
||||||
|
|
||||||
const NAME: &'static str = "msg::character_screen";
|
const NAME: &'static str = "msg::character_screen";
|
||||||
@ -247,6 +277,9 @@ impl<'a> System<'a> for Sys {
|
|||||||
editable_settings,
|
editable_settings,
|
||||||
censor,
|
censor,
|
||||||
automod,
|
automod,
|
||||||
|
time,
|
||||||
|
index,
|
||||||
|
world,
|
||||||
): Self::SystemData,
|
): Self::SystemData,
|
||||||
) {
|
) {
|
||||||
let mut server_emitter = server_event_bus.emitter();
|
let mut server_emitter = server_event_bus.emitter();
|
||||||
@ -267,6 +300,9 @@ impl<'a> System<'a> for Sys {
|
|||||||
&censor,
|
&censor,
|
||||||
&automod,
|
&automod,
|
||||||
msg,
|
msg,
|
||||||
|
*time,
|
||||||
|
&index,
|
||||||
|
&world,
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -74,8 +74,8 @@ use crate::{
|
|||||||
},
|
},
|
||||||
settings::chat::ChatFilter,
|
settings::chat::ChatFilter,
|
||||||
ui::{
|
ui::{
|
||||||
self, fonts::Fonts, img_ids::Rotations, slot, slot::SlotKey, Graphic, Ingameable,
|
self, default_water_color, fonts::Fonts, img_ids::Rotations, slot, slot::SlotKey, Graphic,
|
||||||
ScaleMode, Ui,
|
Ingameable, ScaleMode, Ui,
|
||||||
},
|
},
|
||||||
window::Event as WinEvent,
|
window::Event as WinEvent,
|
||||||
GlobalState,
|
GlobalState,
|
||||||
@ -106,7 +106,7 @@ use common::{
|
|||||||
terrain::{SpriteKind, TerrainChunk, UnlockKind},
|
terrain::{SpriteKind, TerrainChunk, UnlockKind},
|
||||||
trade::{ReducedInventory, TradeAction},
|
trade::{ReducedInventory, TradeAction},
|
||||||
uid::Uid,
|
uid::Uid,
|
||||||
util::{srgba_to_linear, Dir},
|
util::Dir,
|
||||||
vol::RectRasterableVol,
|
vol::RectRasterableVol,
|
||||||
};
|
};
|
||||||
use common_base::{prof_span, span};
|
use common_base::{prof_span, span};
|
||||||
@ -1303,15 +1303,15 @@ impl Hud {
|
|||||||
ui.set_scaling_mode(settings.interface.ui_scale);
|
ui.set_scaling_mode(settings.interface.ui_scale);
|
||||||
// Generate ids.
|
// Generate ids.
|
||||||
let ids = Ids::new(ui.id_generator());
|
let ids = Ids::new(ui.id_generator());
|
||||||
// NOTE: Use a border the same color as the LOD ocean color (but with a
|
|
||||||
// translucent alpha since UI have transparency and LOD doesn't).
|
|
||||||
let water_color = srgba_to_linear(Rgba::new(0.0, 0.18, 0.37, 1.0));
|
|
||||||
// Load world map
|
// Load world map
|
||||||
let mut layers = Vec::new();
|
let mut layers = Vec::new();
|
||||||
for layer in client.world_data().map_layers() {
|
for layer in client.world_data().map_layers() {
|
||||||
layers.push(
|
// NOTE: Use a border the same color as the LOD ocean color (but with a
|
||||||
ui.add_graphic_with_rotations(Graphic::Image(Arc::clone(layer), Some(water_color))),
|
// translucent alpha since UI have transparency and LOD doesn't).
|
||||||
);
|
layers.push(ui.add_graphic_with_rotations(Graphic::Image(
|
||||||
|
Arc::clone(layer),
|
||||||
|
Some(default_water_color()),
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
let world_map = (layers, client.world_data().chunk_size().map(|e| e as u32));
|
let world_map = (layers, client.world_data().chunk_size().map(|e| e as u32));
|
||||||
// Load images.
|
// Load images.
|
||||||
|
@ -117,10 +117,11 @@ impl PlayState for CharSelectionState {
|
|||||||
mainhand,
|
mainhand,
|
||||||
offhand,
|
offhand,
|
||||||
body,
|
body,
|
||||||
|
start_site,
|
||||||
} => {
|
} => {
|
||||||
self.client
|
self.client
|
||||||
.borrow_mut()
|
.borrow_mut()
|
||||||
.create_character(alias, mainhand, offhand, body);
|
.create_character(alias, mainhand, offhand, body, start_site);
|
||||||
},
|
},
|
||||||
ui::Event::EditCharacter {
|
ui::Event::EditCharacter {
|
||||||
alias,
|
alias,
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
render::UiDrawer,
|
render::UiDrawer,
|
||||||
ui::{
|
ui::{
|
||||||
self,
|
self, default_water_color,
|
||||||
fonts::IcedFonts as Fonts,
|
fonts::IcedFonts as Fonts,
|
||||||
ice::{
|
ice::{
|
||||||
component::{
|
component::{
|
||||||
@ -15,7 +15,8 @@ use crate::{
|
|||||||
},
|
},
|
||||||
Element, IcedRenderer, IcedUi as Ui,
|
Element, IcedRenderer, IcedUi as Ui,
|
||||||
},
|
},
|
||||||
img_ids::ImageGraphic,
|
img_ids::{GraphicCreator, ImageGraphic, PixelGraphic},
|
||||||
|
GraphicId,
|
||||||
},
|
},
|
||||||
window, GlobalState,
|
window, GlobalState,
|
||||||
};
|
};
|
||||||
@ -25,6 +26,7 @@ use common::{
|
|||||||
comp::{self, humanoid, inventory::slot::EquipSlot, Inventory, Item},
|
comp::{self, humanoid, inventory::slot::EquipSlot, Inventory, Item},
|
||||||
LoadoutBuilder,
|
LoadoutBuilder,
|
||||||
};
|
};
|
||||||
|
use common_net::msg::world_msg::{SiteId, SiteInfo, SiteKind};
|
||||||
use i18n::{Localization, LocalizationHandle};
|
use i18n::{Localization, LocalizationHandle};
|
||||||
//ImageFrame, Tooltip,
|
//ImageFrame, Tooltip,
|
||||||
use crate::settings::Settings;
|
use crate::settings::Settings;
|
||||||
@ -133,6 +135,7 @@ pub enum Event {
|
|||||||
mainhand: Option<String>,
|
mainhand: Option<String>,
|
||||||
offhand: Option<String>,
|
offhand: Option<String>,
|
||||||
body: comp::Body,
|
body: comp::Body,
|
||||||
|
start_site: Option<SiteId>,
|
||||||
},
|
},
|
||||||
EditCharacter {
|
EditCharacter {
|
||||||
alias: String,
|
alias: String,
|
||||||
@ -168,13 +171,15 @@ enum Mode {
|
|||||||
species_buttons: [button::State; 6],
|
species_buttons: [button::State; 6],
|
||||||
tool_buttons: [button::State; 6],
|
tool_buttons: [button::State; 6],
|
||||||
sliders: Sliders,
|
sliders: Sliders,
|
||||||
scroll: scrollable::State,
|
left_scroll: scrollable::State,
|
||||||
|
right_scroll: scrollable::State,
|
||||||
name_input: text_input::State,
|
name_input: text_input::State,
|
||||||
back_button: button::State,
|
back_button: button::State,
|
||||||
create_button: button::State,
|
create_button: button::State,
|
||||||
rand_character_button: button::State,
|
rand_character_button: button::State,
|
||||||
rand_name_button: button::State,
|
rand_name_button: button::State,
|
||||||
character_id: Option<CharacterId>,
|
character_id: Option<CharacterId>,
|
||||||
|
start_site_idx: usize,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -217,13 +222,15 @@ impl Mode {
|
|||||||
species_buttons: Default::default(),
|
species_buttons: Default::default(),
|
||||||
tool_buttons: Default::default(),
|
tool_buttons: Default::default(),
|
||||||
sliders: Default::default(),
|
sliders: Default::default(),
|
||||||
scroll: Default::default(),
|
left_scroll: Default::default(),
|
||||||
|
right_scroll: Default::default(),
|
||||||
name_input: Default::default(),
|
name_input: Default::default(),
|
||||||
back_button: Default::default(),
|
back_button: Default::default(),
|
||||||
create_button: Default::default(),
|
create_button: Default::default(),
|
||||||
rand_character_button: Default::default(),
|
rand_character_button: Default::default(),
|
||||||
rand_name_button: Default::default(),
|
rand_name_button: Default::default(),
|
||||||
character_id: None,
|
character_id: None,
|
||||||
|
start_site_idx: 0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -243,13 +250,15 @@ impl Mode {
|
|||||||
species_buttons: Default::default(),
|
species_buttons: Default::default(),
|
||||||
tool_buttons: Default::default(),
|
tool_buttons: Default::default(),
|
||||||
sliders: Default::default(),
|
sliders: Default::default(),
|
||||||
scroll: Default::default(),
|
left_scroll: Default::default(),
|
||||||
|
right_scroll: Default::default(),
|
||||||
name_input: Default::default(),
|
name_input: Default::default(),
|
||||||
back_button: Default::default(),
|
back_button: Default::default(),
|
||||||
create_button: Default::default(),
|
create_button: Default::default(),
|
||||||
rand_character_button: Default::default(),
|
rand_character_button: Default::default(),
|
||||||
rand_name_button: Default::default(),
|
rand_name_button: Default::default(),
|
||||||
character_id: Some(character_id),
|
character_id: Some(character_id),
|
||||||
|
start_site_idx: 0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -279,6 +288,8 @@ struct Controls {
|
|||||||
// Id of the selected character
|
// Id of the selected character
|
||||||
selected: Option<CharacterId>,
|
selected: Option<CharacterId>,
|
||||||
default_name: String,
|
default_name: String,
|
||||||
|
map_img: GraphicId,
|
||||||
|
possible_starting_sites: Vec<SiteInfo>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
@ -309,6 +320,7 @@ enum Message {
|
|||||||
EyeColor(u8),
|
EyeColor(u8),
|
||||||
Accessory(u8),
|
Accessory(u8),
|
||||||
Beard(u8),
|
Beard(u8),
|
||||||
|
StartSite(usize),
|
||||||
// Workaround for widgets that require a message but we don't want them to actually do
|
// Workaround for widgets that require a message but we don't want them to actually do
|
||||||
// anything
|
// anything
|
||||||
DoNothing,
|
DoNothing,
|
||||||
@ -321,6 +333,8 @@ impl Controls {
|
|||||||
selected: Option<CharacterId>,
|
selected: Option<CharacterId>,
|
||||||
default_name: String,
|
default_name: String,
|
||||||
server_info: &ServerInfo,
|
server_info: &ServerInfo,
|
||||||
|
map_img: GraphicId,
|
||||||
|
possible_starting_sites: Vec<SiteInfo>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let version = common::util::DISPLAY_VERSION_LONG.clone();
|
let version = common::util::DISPLAY_VERSION_LONG.clone();
|
||||||
let alpha = format!("Veloren {}", common::util::DISPLAY_VERSION.as_str());
|
let alpha = format!("Veloren {}", common::util::DISPLAY_VERSION.as_str());
|
||||||
@ -339,6 +353,8 @@ impl Controls {
|
|||||||
mode: Mode::select(Some(InfoContent::LoadingCharacters)),
|
mode: Mode::select(Some(InfoContent::LoadingCharacters)),
|
||||||
selected,
|
selected,
|
||||||
default_name,
|
default_name,
|
||||||
|
map_img,
|
||||||
|
possible_starting_sites,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -831,7 +847,8 @@ impl Controls {
|
|||||||
inventory: _,
|
inventory: _,
|
||||||
mainhand,
|
mainhand,
|
||||||
offhand: _,
|
offhand: _,
|
||||||
ref mut scroll,
|
ref mut left_scroll,
|
||||||
|
ref mut right_scroll,
|
||||||
ref mut body_type_buttons,
|
ref mut body_type_buttons,
|
||||||
ref mut species_buttons,
|
ref mut species_buttons,
|
||||||
ref mut tool_buttons,
|
ref mut tool_buttons,
|
||||||
@ -842,6 +859,7 @@ impl Controls {
|
|||||||
ref mut rand_character_button,
|
ref mut rand_character_button,
|
||||||
ref mut rand_name_button,
|
ref mut rand_name_button,
|
||||||
character_id,
|
character_id,
|
||||||
|
start_site_idx,
|
||||||
} => {
|
} => {
|
||||||
let unselected_style = style::button::Style::new(imgs.icon_border)
|
let unselected_style = style::button::Style::new(imgs.icon_border)
|
||||||
.hover_image(imgs.icon_border_mo)
|
.hover_image(imgs.icon_border_mo)
|
||||||
@ -1070,19 +1088,19 @@ impl Controls {
|
|||||||
// Height of interactable area
|
// Height of interactable area
|
||||||
const SLIDER_HEIGHT: u16 = 30;
|
const SLIDER_HEIGHT: u16 = 30;
|
||||||
|
|
||||||
fn char_slider<'a>(
|
fn char_slider<'a, T: Copy + Into<u32>>(
|
||||||
text: String,
|
text: String,
|
||||||
state: &'a mut slider::State,
|
state: &'a mut slider::State,
|
||||||
max: u8,
|
max: T,
|
||||||
selected_val: u8,
|
selected_val: T,
|
||||||
on_change: impl 'static + Fn(u8) -> Message,
|
on_change: impl 'static + Fn(u32) -> Message,
|
||||||
(fonts, imgs): (&Fonts, &Imgs),
|
(fonts, imgs): (&Fonts, &Imgs),
|
||||||
) -> Element<'a, Message> {
|
) -> Element<'a, Message> {
|
||||||
Column::with_children(vec![
|
Column::with_children(vec![
|
||||||
Text::new(text)
|
Text::new(text)
|
||||||
.size(fonts.cyri.scale(SLIDER_TEXT_SIZE))
|
.size(fonts.cyri.scale(SLIDER_TEXT_SIZE))
|
||||||
.into(),
|
.into(),
|
||||||
Slider::new(state, 0..=max, selected_val, on_change)
|
Slider::new(state, 0..=max.into(), selected_val.into(), on_change)
|
||||||
.height(SLIDER_HEIGHT)
|
.height(SLIDER_HEIGHT)
|
||||||
.style(style::slider::Style::images(
|
.style(style::slider::Style::images(
|
||||||
imgs.slider_indicator,
|
imgs.slider_indicator,
|
||||||
@ -1096,13 +1114,13 @@ impl Controls {
|
|||||||
.align_items(Align::Center)
|
.align_items(Align::Center)
|
||||||
.into()
|
.into()
|
||||||
}
|
}
|
||||||
fn char_slider_greyable<'a>(
|
fn char_slider_greyable<'a, T: Copy + Into<u32>>(
|
||||||
active: bool,
|
active: bool,
|
||||||
text: String,
|
text: String,
|
||||||
state: &'a mut slider::State,
|
state: &'a mut slider::State,
|
||||||
max: u8,
|
max: T,
|
||||||
selected_val: u8,
|
selected_val: T,
|
||||||
on_change: impl 'static + Fn(u8) -> Message,
|
on_change: impl 'static + Fn(u32) -> Message,
|
||||||
(fonts, imgs): (&Fonts, &Imgs),
|
(fonts, imgs): (&Fonts, &Imgs),
|
||||||
) -> Element<'a, Message> {
|
) -> Element<'a, Message> {
|
||||||
if active {
|
if active {
|
||||||
@ -1115,7 +1133,9 @@ impl Controls {
|
|||||||
.into(),
|
.into(),
|
||||||
// "Disabled" slider
|
// "Disabled" slider
|
||||||
// TODO: add iced support for disabled sliders (like buttons)
|
// TODO: add iced support for disabled sliders (like buttons)
|
||||||
Slider::new(state, 0..=max, selected_val, |_| Message::DoNothing)
|
Slider::new(state, 0..=max.into(), selected_val.into(), |_| {
|
||||||
|
Message::DoNothing
|
||||||
|
})
|
||||||
.height(SLIDER_HEIGHT)
|
.height(SLIDER_HEIGHT)
|
||||||
.style(style::slider::Style {
|
.style(style::slider::Style {
|
||||||
cursor: style::slider::Cursor::Color(Rgba::zero()),
|
cursor: style::slider::Cursor::Color(Rgba::zero()),
|
||||||
@ -1140,7 +1160,7 @@ impl Controls {
|
|||||||
&mut sliders.hair_style,
|
&mut sliders.hair_style,
|
||||||
body.species.num_hair_styles(body.body_type) - 1,
|
body.species.num_hair_styles(body.body_type) - 1,
|
||||||
body.hair_style,
|
body.hair_style,
|
||||||
Message::HairStyle,
|
|x| Message::HairStyle(x as u8),
|
||||||
(fonts, imgs),
|
(fonts, imgs),
|
||||||
),
|
),
|
||||||
char_slider(
|
char_slider(
|
||||||
@ -1148,7 +1168,7 @@ impl Controls {
|
|||||||
&mut sliders.hair_color,
|
&mut sliders.hair_color,
|
||||||
body.species.num_hair_colors() - 1,
|
body.species.num_hair_colors() - 1,
|
||||||
body.hair_color,
|
body.hair_color,
|
||||||
Message::HairColor,
|
|x| Message::HairColor(x as u8),
|
||||||
(fonts, imgs),
|
(fonts, imgs),
|
||||||
),
|
),
|
||||||
char_slider(
|
char_slider(
|
||||||
@ -1156,7 +1176,7 @@ impl Controls {
|
|||||||
&mut sliders.skin,
|
&mut sliders.skin,
|
||||||
body.species.num_skin_colors() - 1,
|
body.species.num_skin_colors() - 1,
|
||||||
body.skin,
|
body.skin,
|
||||||
Message::Skin,
|
|x| Message::Skin(x as u8),
|
||||||
(fonts, imgs),
|
(fonts, imgs),
|
||||||
),
|
),
|
||||||
char_slider(
|
char_slider(
|
||||||
@ -1164,7 +1184,7 @@ impl Controls {
|
|||||||
&mut sliders.eyes,
|
&mut sliders.eyes,
|
||||||
body.species.num_eyes(body.body_type) - 1,
|
body.species.num_eyes(body.body_type) - 1,
|
||||||
body.eyes,
|
body.eyes,
|
||||||
Message::Eyes,
|
|x| Message::Eyes(x as u8),
|
||||||
(fonts, imgs),
|
(fonts, imgs),
|
||||||
),
|
),
|
||||||
char_slider(
|
char_slider(
|
||||||
@ -1172,7 +1192,7 @@ impl Controls {
|
|||||||
&mut sliders.eye_color,
|
&mut sliders.eye_color,
|
||||||
body.species.num_eye_colors() - 1,
|
body.species.num_eye_colors() - 1,
|
||||||
body.eye_color,
|
body.eye_color,
|
||||||
Message::EyeColor,
|
|x| Message::EyeColor(x as u8),
|
||||||
(fonts, imgs),
|
(fonts, imgs),
|
||||||
),
|
),
|
||||||
char_slider_greyable(
|
char_slider_greyable(
|
||||||
@ -1181,7 +1201,7 @@ impl Controls {
|
|||||||
&mut sliders.accessory,
|
&mut sliders.accessory,
|
||||||
body.species.num_accessories(body.body_type) - 1,
|
body.species.num_accessories(body.body_type) - 1,
|
||||||
body.accessory,
|
body.accessory,
|
||||||
Message::Accessory,
|
|x| Message::Accessory(x as u8),
|
||||||
(fonts, imgs),
|
(fonts, imgs),
|
||||||
),
|
),
|
||||||
char_slider_greyable(
|
char_slider_greyable(
|
||||||
@ -1190,7 +1210,7 @@ impl Controls {
|
|||||||
&mut sliders.beard,
|
&mut sliders.beard,
|
||||||
body.species.num_beards(body.body_type) - 1,
|
body.species.num_beards(body.body_type) - 1,
|
||||||
body.beard,
|
body.beard,
|
||||||
Message::Beard,
|
|x| Message::Beard(x as u8),
|
||||||
(fonts, imgs),
|
(fonts, imgs),
|
||||||
),
|
),
|
||||||
])
|
])
|
||||||
@ -1213,7 +1233,7 @@ impl Controls {
|
|||||||
tooltip::text(&tooltip_text, tooltip_style)
|
tooltip::text(&tooltip_text, tooltip_style)
|
||||||
});
|
});
|
||||||
|
|
||||||
let column_content = vec![
|
let left_column_content = vec![
|
||||||
body_type.into(),
|
body_type.into(),
|
||||||
tool.into(),
|
tool.into(),
|
||||||
species.into(),
|
species.into(),
|
||||||
@ -1221,7 +1241,54 @@ impl Controls {
|
|||||||
rand_character.into(),
|
rand_character.into(),
|
||||||
];
|
];
|
||||||
|
|
||||||
let left_column = Container::new(
|
let right_column_content = vec![
|
||||||
|
Image::new(self.map_img)
|
||||||
|
.height(Length::Units(300))
|
||||||
|
.width(Length::Units(300))
|
||||||
|
.into(),
|
||||||
|
Column::with_children(if self.possible_starting_sites.is_empty() {
|
||||||
|
Vec::new()
|
||||||
|
} else {
|
||||||
|
let site_slider = char_slider(
|
||||||
|
i18n.get_msg("char_selection-starting_site").into_owned(),
|
||||||
|
&mut sliders.starting_site,
|
||||||
|
self.possible_starting_sites.len() as u32 - 1,
|
||||||
|
*start_site_idx as u32,
|
||||||
|
|x| Message::StartSite(x as usize),
|
||||||
|
(fonts, imgs),
|
||||||
|
);
|
||||||
|
|
||||||
|
let site_name = Text::new(i18n
|
||||||
|
.get_msg_ctx("char_selection-starting_site_name", &i18n::fluent_args! {
|
||||||
|
"name" => self.possible_starting_sites[*start_site_idx].name.as_deref()
|
||||||
|
.unwrap_or("Unknown"),
|
||||||
|
})
|
||||||
|
.into_owned())
|
||||||
|
.size(fonts.cyri.scale(SLIDER_TEXT_SIZE))
|
||||||
|
.into();
|
||||||
|
|
||||||
|
let site_kind = Text::new(i18n
|
||||||
|
.get_msg_ctx("char_selection-starting_site_kind", &i18n::fluent_args! {
|
||||||
|
"kind" => match self.possible_starting_sites[*start_site_idx].kind {
|
||||||
|
SiteKind::Town => i18n.get_msg("hud-map-town").into_owned(),
|
||||||
|
SiteKind::Castle => i18n.get_msg("hud-map-castle").into_owned(),
|
||||||
|
SiteKind::Bridge => i18n.get_msg("hud-map-bridge").into_owned(),
|
||||||
|
_ => "Unknown".to_string(),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.into_owned())
|
||||||
|
.size(fonts.cyri.scale(SLIDER_TEXT_SIZE))
|
||||||
|
.into();
|
||||||
|
|
||||||
|
vec![site_slider, site_name, site_kind]
|
||||||
|
})
|
||||||
|
.max_width(200)
|
||||||
|
.padding(5)
|
||||||
|
.into(),
|
||||||
|
];
|
||||||
|
|
||||||
|
let column = |column_content, scroll| {
|
||||||
|
let column = Container::new(
|
||||||
Scrollable::new(scroll)
|
Scrollable::new(scroll)
|
||||||
.push(
|
.push(
|
||||||
Column::with_children(column_content)
|
Column::with_children(column_content)
|
||||||
@ -1242,8 +1309,8 @@ impl Controls {
|
|||||||
//.width(Length::Fill)
|
//.width(Length::Fill)
|
||||||
.height(Length::Fill);
|
.height(Length::Fill);
|
||||||
|
|
||||||
let left_column = Column::with_children(vec![
|
Column::with_children(vec![
|
||||||
Container::new(left_column)
|
Container::new(column)
|
||||||
.style(style::container::Style::color(Rgba::from_translucent(
|
.style(style::container::Style::color(Rgba::from_translucent(
|
||||||
0,
|
0,
|
||||||
BANNER_ALPHA,
|
BANNER_ALPHA,
|
||||||
@ -1257,11 +1324,13 @@ impl Controls {
|
|||||||
.color(Rgba::from_translucent(0, BANNER_ALPHA))
|
.color(Rgba::from_translucent(0, BANNER_ALPHA))
|
||||||
.into(),
|
.into(),
|
||||||
])
|
])
|
||||||
.height(Length::Fill);
|
.height(Length::Fill)
|
||||||
|
};
|
||||||
|
|
||||||
let top = Row::with_children(vec![
|
let top = Row::with_children(vec![
|
||||||
left_column.into(),
|
column(left_column_content, left_scroll).into(),
|
||||||
MouseDetector::new(&mut self.mouse_detector, Length::Fill, Length::Fill).into(),
|
MouseDetector::new(&mut self.mouse_detector, Length::Fill, Length::Fill).into(),
|
||||||
|
column(right_column_content, right_scroll).into(),
|
||||||
])
|
])
|
||||||
.padding(10)
|
.padding(10)
|
||||||
.width(Length::Fill)
|
.width(Length::Fill)
|
||||||
@ -1516,6 +1585,7 @@ impl Controls {
|
|||||||
body,
|
body,
|
||||||
mainhand,
|
mainhand,
|
||||||
offhand,
|
offhand,
|
||||||
|
start_site_idx,
|
||||||
..
|
..
|
||||||
} = &self.mode
|
} = &self.mode
|
||||||
{
|
{
|
||||||
@ -1524,6 +1594,10 @@ impl Controls {
|
|||||||
mainhand: mainhand.map(String::from),
|
mainhand: mainhand.map(String::from),
|
||||||
offhand: offhand.map(String::from),
|
offhand: offhand.map(String::from),
|
||||||
body: comp::Body::Humanoid(*body),
|
body: comp::Body::Humanoid(*body),
|
||||||
|
start_site: self
|
||||||
|
.possible_starting_sites
|
||||||
|
.get(*start_site_idx)
|
||||||
|
.map(|info| info.id),
|
||||||
});
|
});
|
||||||
self.mode = Mode::select(Some(InfoContent::CreatingCharacter));
|
self.mode = Mode::select(Some(InfoContent::CreatingCharacter));
|
||||||
}
|
}
|
||||||
@ -1643,6 +1717,11 @@ impl Controls {
|
|||||||
body.validate();
|
body.validate();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
Message::StartSite(idx) => {
|
||||||
|
if let Mode::CreateOrEdit { start_site_idx, .. } = &mut self.mode {
|
||||||
|
*start_site_idx = idx;
|
||||||
|
}
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1707,6 +1786,19 @@ impl CharSelectionUi {
|
|||||||
selected_character,
|
selected_character,
|
||||||
default_name,
|
default_name,
|
||||||
client.server_info(),
|
client.server_info(),
|
||||||
|
ui.add_graphic(
|
||||||
|
PixelGraphic::new_graphic((
|
||||||
|
client.world_data().map_image().clone(),
|
||||||
|
Some(default_water_color()),
|
||||||
|
))
|
||||||
|
.expect("Failed to generate map image"),
|
||||||
|
),
|
||||||
|
client.sites()
|
||||||
|
.values()
|
||||||
|
// TODO: Enforce this server-side and add some way to customise it?
|
||||||
|
.filter(|info| matches!(&info.site.kind, SiteKind::Town | SiteKind::Castle | SiteKind::Bridge))
|
||||||
|
.map(|info| info.site.clone())
|
||||||
|
.collect(),
|
||||||
);
|
);
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
@ -1814,4 +1906,5 @@ struct Sliders {
|
|||||||
eye_color: slider::State,
|
eye_color: slider::State,
|
||||||
accessory: slider::State,
|
accessory: slider::State,
|
||||||
beard: slider::State,
|
beard: slider::State,
|
||||||
|
starting_site: slider::State,
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@ use common::{
|
|||||||
assets::{self, AssetExt, DotVoxAsset, Error},
|
assets::{self, AssetExt, DotVoxAsset, Error},
|
||||||
figure::Segment,
|
figure::Segment,
|
||||||
};
|
};
|
||||||
|
use image::DynamicImage;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use vek::*;
|
use vek::*;
|
||||||
|
|
||||||
@ -112,6 +113,16 @@ impl<'a> GraphicCreator<'a> for VoxelPixArtGraphic {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub enum PixelGraphic {}
|
||||||
|
|
||||||
|
impl<'a> GraphicCreator<'a> for PixelGraphic {
|
||||||
|
type Specifier = (Arc<DynamicImage>, Option<Rgba<f32>>);
|
||||||
|
|
||||||
|
fn new_graphic((img, col): Self::Specifier) -> Result<Graphic, Error> {
|
||||||
|
Ok(Graphic::Image(img, col))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Rotations {
|
pub struct Rotations {
|
||||||
pub none: conrod_core::image::Id,
|
pub none: conrod_core::image::Id,
|
||||||
pub cw90: conrod_core::image::Id,
|
pub cw90: conrod_core::image::Id,
|
||||||
|
@ -1090,3 +1090,6 @@ fn default_scissor(physical_resolution: Vec2<u32>) -> Aabr<u16> {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Don't hard code this
|
||||||
|
pub fn default_water_color() -> Rgba<f32> { srgba_to_linear(Rgba::new(0.0, 0.18, 0.37, 1.0)) }
|
||||||
|
@ -219,6 +219,11 @@ impl World {
|
|||||||
|
|
||||||
pub fn sample_blocks(&self) -> BlockGen { BlockGen::new(ColumnGen::new(&self.sim)) }
|
pub fn sample_blocks(&self) -> BlockGen { BlockGen::new(ColumnGen::new(&self.sim)) }
|
||||||
|
|
||||||
|
/// Find a position that's accessible to a player at the given world
|
||||||
|
/// position by searching blocks vertically.
|
||||||
|
///
|
||||||
|
/// If `ascending` is `true`, we try to find the highest accessible position
|
||||||
|
/// instead of the lowest.
|
||||||
pub fn find_accessible_pos(
|
pub fn find_accessible_pos(
|
||||||
&self,
|
&self,
|
||||||
index: IndexRef,
|
index: IndexRef,
|
||||||
|
Loading…
Reference in New Issue
Block a user