2019-08-19 12:39:23 +00:00
|
|
|
#![deny(unsafe_code)]
|
2020-06-14 16:48:07 +00:00
|
|
|
#![allow(clippy::option_map_unit_fn)]
|
2020-03-31 17:02:26 +00:00
|
|
|
#![feature(bool_to_option)]
|
2019-04-29 20:37:19 +00:00
|
|
|
#![recursion_limit = "2048"]
|
2019-01-23 20:01:58 +00:00
|
|
|
|
2020-02-29 03:59:11 +00:00
|
|
|
use veloren_voxygen::{
|
2020-11-12 03:55:40 +00:00
|
|
|
audio::AudioFrontend,
|
2020-05-25 18:11:39 +00:00
|
|
|
i18n::{self, i18n_asset_key, Localization},
|
2020-06-16 13:55:37 +00:00
|
|
|
profile::Profile,
|
2020-02-08 03:26:17 +00:00
|
|
|
run,
|
2021-04-10 14:53:34 +00:00
|
|
|
scene::terrain::SpriteRenderContext,
|
2021-01-23 08:34:38 +00:00
|
|
|
settings::{get_fps, AudioOutput, Settings},
|
2020-01-17 23:43:18 +00:00
|
|
|
window::Window,
|
2020-02-08 03:26:17 +00:00
|
|
|
GlobalState,
|
2020-01-20 13:37:29 +00:00
|
|
|
};
|
2020-02-29 03:59:11 +00:00
|
|
|
|
2019-10-27 07:11:18 +00:00
|
|
|
use common::{
|
2020-12-12 22:14:24 +00:00
|
|
|
assets::{self, AssetExt},
|
2019-10-27 07:11:18 +00:00
|
|
|
clock::Clock,
|
|
|
|
};
|
2020-02-08 03:26:17 +00:00
|
|
|
use std::panic;
|
2020-10-25 20:19:39 +00:00
|
|
|
use tracing::{error, info, warn};
|
2019-04-17 13:23:34 +00:00
|
|
|
|
2020-12-10 11:50:48 +00:00
|
|
|
#[allow(clippy::manual_unwrap_or)]
|
2019-01-02 21:25:01 +00:00
|
|
|
fn main() {
|
2019-08-18 19:21:16 +00:00
|
|
|
// Load the settings
|
2020-02-01 20:39:39 +00:00
|
|
|
// Note: This won't log anything due to it being called before
|
2020-06-21 10:22:26 +00:00
|
|
|
// `logging::init`. The issue is we need to read a setting to decide
|
2020-02-01 20:39:39 +00:00
|
|
|
// whether we create a log file or not.
|
2019-10-27 07:11:18 +00:00
|
|
|
let mut settings = Settings::load();
|
|
|
|
// Save settings to add new fields or create the file if it is not already there
|
2019-08-18 19:21:16 +00:00
|
|
|
if let Err(err) = settings.save_to_file() {
|
|
|
|
panic!("Failed to save settings: {:?}", err);
|
|
|
|
}
|
2019-12-10 14:02:51 +00:00
|
|
|
|
2019-10-27 07:11:18 +00:00
|
|
|
// Init logging and hold the guards.
|
2021-03-28 23:40:53 +00:00
|
|
|
const LOG_FILENAME: &str = "voxygen.log";
|
|
|
|
let _guards = common_frontend::init_stdout(Some((&settings.log.logs_path, LOG_FILENAME)));
|
2020-01-17 23:43:18 +00:00
|
|
|
|
2020-10-25 20:19:39 +00:00
|
|
|
if let Some(path) = veloren_voxygen::settings::voxygen_data_dir().parent() {
|
|
|
|
info!("Using userdata dir at: {}", path.display());
|
|
|
|
} else {
|
|
|
|
error!("Can't log userdata dir, voxygen data dir has no parent!");
|
|
|
|
}
|
|
|
|
|
2019-05-11 20:39:46 +00:00
|
|
|
// Set up panic handler to relay swish panic messages to the user
|
2019-05-05 18:42:44 +00:00
|
|
|
let default_hook = panic::take_hook();
|
2019-04-25 11:49:36 +00:00
|
|
|
panic::set_hook(Box::new(move |panic_info| {
|
2019-05-08 20:01:02 +00:00
|
|
|
let panic_info_payload = panic_info.payload();
|
|
|
|
let payload_string = panic_info_payload.downcast_ref::<String>();
|
|
|
|
let reason = match payload_string {
|
2019-05-09 17:27:07 +00:00
|
|
|
Some(s) => &s,
|
|
|
|
None => {
|
|
|
|
let payload_str = panic_info_payload.downcast_ref::<&str>();
|
|
|
|
match payload_str {
|
|
|
|
Some(st) => st,
|
|
|
|
None => "Payload is not a string",
|
2019-05-08 20:01:02 +00:00
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-05-08 20:01:02 +00:00
|
|
|
};
|
2019-05-17 07:55:51 +00:00
|
|
|
let msg = format!(
|
2019-05-31 03:01:52 +00:00
|
|
|
"A critical error has occurred and Voxygen has been forced to \
|
2019-05-17 07:55:51 +00:00
|
|
|
terminate in an unusual manner. Details about the error can be \
|
|
|
|
found below.\n\
|
|
|
|
\n\
|
|
|
|
> What should I do?\n\
|
|
|
|
\n\
|
|
|
|
We need your help to fix this! You can help by contacting us and \
|
|
|
|
reporting this problem. To do this, open an issue on the Veloren \
|
|
|
|
issue tracker:\n\
|
|
|
|
\n\
|
|
|
|
https://www.gitlab.com/veloren/veloren/issues/new\n\
|
|
|
|
\n\
|
|
|
|
If you're on the Veloren community Discord server, we'd be \
|
|
|
|
grateful if you could also post a message in the #support channel.
|
|
|
|
\n\
|
|
|
|
> What should I include?\n\
|
|
|
|
\n\
|
|
|
|
The error information below will be useful in finding and fixing \
|
|
|
|
the problem. Please include as much information about your setup \
|
|
|
|
and the events that led up to the panic as possible.
|
|
|
|
\n\
|
|
|
|
Voxygen has logged information about the problem (including this \
|
2020-03-20 08:47:00 +00:00
|
|
|
message) to the file {}. Please include the contents of this \
|
2019-05-17 07:55:51 +00:00
|
|
|
file in your bug report.
|
|
|
|
\n\
|
|
|
|
> Error information\n\
|
|
|
|
\n\
|
|
|
|
The information below is intended for developers and testers.\n\
|
|
|
|
\n\
|
|
|
|
Panic Payload: {:?}\n\
|
2020-03-20 08:47:00 +00:00
|
|
|
PanicInfo: {}\n\
|
|
|
|
Game version: {} [{}]",
|
2020-03-22 22:06:05 +00:00
|
|
|
Settings::load()
|
|
|
|
.log
|
|
|
|
.logs_path
|
2019-12-18 19:35:17 +00:00
|
|
|
.join("voxygen-<date>.log")
|
|
|
|
.display(),
|
|
|
|
reason,
|
|
|
|
panic_info,
|
2020-03-20 08:47:00 +00:00
|
|
|
common::util::GIT_HASH.to_string(),
|
|
|
|
common::util::GIT_DATE.to_string()
|
2019-05-17 07:55:51 +00:00
|
|
|
);
|
2019-04-25 11:49:36 +00:00
|
|
|
|
2019-06-06 14:48:41 +00:00
|
|
|
error!(
|
2019-05-14 07:41:27 +00:00
|
|
|
"VOXYGEN HAS PANICKED\n\n{}\n\nBacktrace:\n{:?}",
|
|
|
|
msg,
|
2019-05-17 07:55:51 +00:00
|
|
|
backtrace::Backtrace::new(),
|
2019-05-14 07:41:27 +00:00
|
|
|
);
|
2019-04-25 11:49:36 +00:00
|
|
|
|
2020-09-08 14:33:06 +00:00
|
|
|
#[cfg(feature = "native-dialog")]
|
2020-01-07 05:45:43 +00:00
|
|
|
{
|
2021-02-07 05:48:57 +00:00
|
|
|
use native_dialog::{MessageDialog, MessageType};
|
2020-09-08 14:33:06 +00:00
|
|
|
|
|
|
|
let mbox = move || {
|
2021-02-07 05:48:57 +00:00
|
|
|
MessageDialog::new()
|
|
|
|
.set_title("Voxygen has panicked")
|
2020-09-08 14:33:06 +00:00
|
|
|
//somehow `<` and `>` are invalid characters and cause the msg to get replaced
|
|
|
|
// by some generic text thus i replace them
|
2021-02-07 05:48:57 +00:00
|
|
|
.set_text(&msg.replace('<', "[").replace('>', "]"))
|
|
|
|
.set_type(MessageType::Error)
|
|
|
|
.show_alert()
|
|
|
|
.unwrap()
|
2020-09-08 14:33:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// On windows we need to spawn a thread as the msg doesn't work otherwise
|
|
|
|
#[cfg(target_os = "windows")]
|
2021-02-18 00:01:57 +00:00
|
|
|
{
|
|
|
|
let builder = std::thread::Builder::new().name("shutdown".into());
|
|
|
|
builder
|
|
|
|
.spawn(move || {
|
|
|
|
mbox();
|
|
|
|
})
|
|
|
|
.unwrap()
|
|
|
|
.join()
|
|
|
|
.unwrap();
|
|
|
|
}
|
2020-09-08 14:33:06 +00:00
|
|
|
|
2020-09-09 10:11:56 +00:00
|
|
|
#[cfg(not(target_os = "windows"))]
|
2020-09-08 14:33:06 +00:00
|
|
|
mbox();
|
2020-01-07 05:45:43 +00:00
|
|
|
}
|
2019-05-05 18:42:44 +00:00
|
|
|
|
|
|
|
default_hook(panic_info);
|
2019-04-25 11:49:36 +00:00
|
|
|
}));
|
|
|
|
|
2020-12-12 22:14:24 +00:00
|
|
|
assets::start_hot_reloading();
|
|
|
|
|
2020-06-17 07:49:14 +00:00
|
|
|
// Initialise watcher for animation hotreloading
|
|
|
|
#[cfg(feature = "hot-anim")]
|
|
|
|
anim::init();
|
|
|
|
|
2019-10-27 07:11:18 +00:00
|
|
|
// Setup audio
|
2020-11-19 18:46:31 +00:00
|
|
|
let mut audio = match settings.audio.output {
|
|
|
|
AudioOutput::Off => AudioFrontend::no_audio(),
|
|
|
|
AudioOutput::Automatic => AudioFrontend::new(settings.audio.max_sfx_channels),
|
|
|
|
// AudioOutput::Device(ref dev) => Some(dev.clone()),
|
|
|
|
};
|
2019-10-27 07:11:18 +00:00
|
|
|
|
2021-04-18 18:05:27 +00:00
|
|
|
audio.set_master_volume(settings.audio.master_volume);
|
2019-10-27 07:11:18 +00:00
|
|
|
audio.set_music_volume(settings.audio.music_volume);
|
|
|
|
audio.set_sfx_volume(settings.audio.sfx_volume);
|
|
|
|
|
|
|
|
// Load the profile.
|
|
|
|
let profile = Profile::load();
|
|
|
|
|
2020-12-13 01:09:57 +00:00
|
|
|
let i18n = Localization::load(&i18n_asset_key(&settings.language.selected_language))
|
|
|
|
.unwrap_or_else(|error| {
|
|
|
|
let selected_language = &settings.language.selected_language;
|
|
|
|
warn!(
|
|
|
|
?error,
|
|
|
|
?selected_language,
|
|
|
|
"Impossible to load language: change to the default language (English) instead.",
|
|
|
|
);
|
|
|
|
settings.language.selected_language = i18n::REFERENCE_LANG.to_owned();
|
|
|
|
Localization::load_expect(&i18n_asset_key(&settings.language.selected_language))
|
|
|
|
});
|
2020-12-12 22:14:24 +00:00
|
|
|
i18n.read().log_missing_entries();
|
2019-01-30 12:11:34 +00:00
|
|
|
|
2019-10-27 07:11:18 +00:00
|
|
|
// Create window
|
2021-04-10 14:53:34 +00:00
|
|
|
let (mut window, event_loop) = Window::new(&settings).expect("Failed to create window!");
|
2019-10-27 07:11:18 +00:00
|
|
|
|
2020-12-02 00:35:32 +00:00
|
|
|
let clipboard = iced_winit::Clipboard::new(window.window());
|
|
|
|
|
2021-04-10 14:53:34 +00:00
|
|
|
let lazy_init = SpriteRenderContext::new(window.renderer_mut());
|
|
|
|
|
2019-10-27 07:11:18 +00:00
|
|
|
let global_state = GlobalState {
|
|
|
|
audio,
|
|
|
|
profile,
|
|
|
|
window,
|
2021-04-10 14:53:34 +00:00
|
|
|
lazy_init,
|
2020-11-12 02:47:22 +00:00
|
|
|
clock: Clock::new(std::time::Duration::from_secs_f64(
|
2021-01-23 08:34:38 +00:00
|
|
|
1.0 / get_fps(settings.graphics.max_fps) as f64,
|
2020-11-12 02:47:22 +00:00
|
|
|
)),
|
2019-10-27 07:11:18 +00:00
|
|
|
settings,
|
|
|
|
info_message: None,
|
|
|
|
#[cfg(feature = "singleplayer")]
|
|
|
|
singleplayer: None,
|
2020-12-12 22:14:24 +00:00
|
|
|
i18n,
|
2020-12-02 00:35:32 +00:00
|
|
|
clipboard,
|
2020-12-30 14:50:17 +00:00
|
|
|
client_error: None,
|
2019-10-27 07:11:18 +00:00
|
|
|
};
|
2019-06-19 19:25:08 +00:00
|
|
|
|
2019-10-27 07:11:18 +00:00
|
|
|
run::run(global_state, event_loop);
|
2019-01-02 21:25:01 +00:00
|
|
|
}
|