2023-07-13 03:54:27 +03:00
|
|
|
// Copyright 2023 The Grim Developers
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
use std::sync::atomic::{AtomicBool, Ordering};
|
2023-07-14 00:55:31 +03:00
|
|
|
use egui::os::OperatingSystem;
|
|
|
|
use egui::RichText;
|
2023-07-13 03:54:27 +03:00
|
|
|
|
|
|
|
use lazy_static::lazy_static;
|
2023-07-14 00:55:31 +03:00
|
|
|
use crate::gui::Colors;
|
2023-07-13 03:54:27 +03:00
|
|
|
|
|
|
|
use crate::gui::platform::PlatformCallbacks;
|
2023-07-14 00:55:31 +03:00
|
|
|
use crate::gui::views::{Accounts, Modal, ModalContainer, Network, View};
|
|
|
|
use crate::node::Node;
|
2023-07-13 03:54:27 +03:00
|
|
|
|
|
|
|
lazy_static! {
|
|
|
|
/// To check if side panel is open from any part of ui.
|
2023-07-14 00:55:31 +03:00
|
|
|
static ref SIDE_PANEL_OPEN: AtomicBool = AtomicBool::new(false);
|
2023-07-13 03:54:27 +03:00
|
|
|
}
|
|
|
|
|
2023-07-14 00:55:31 +03:00
|
|
|
/// Contains main ui content, handles side panel state.
|
2023-07-13 03:54:27 +03:00
|
|
|
pub struct Root {
|
2023-07-14 00:55:31 +03:00
|
|
|
/// Side panel content.
|
|
|
|
side_panel: Network,
|
|
|
|
/// Central panel content.
|
|
|
|
central_content: Accounts,
|
|
|
|
|
|
|
|
/// Check if app exit is allowed on close event of [`eframe::App`] platform implementation.
|
|
|
|
pub(crate) exit_allowed: bool,
|
|
|
|
|
|
|
|
/// Flag to show exit progress at [`Modal`].
|
|
|
|
show_exit_progress: bool,
|
|
|
|
|
|
|
|
/// List of allowed [`Modal`] ids for this [`ModalContainer`].
|
|
|
|
allowed_modal_ids: Vec<&'static str>
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Root {
|
|
|
|
fn default() -> Self {
|
|
|
|
// Exit from eframe only for non-mobile platforms.
|
|
|
|
let os = OperatingSystem::from_target_os();
|
|
|
|
let exit_allowed = os == OperatingSystem::Android || os == OperatingSystem::IOS;
|
|
|
|
Self {
|
|
|
|
side_panel: Network::default(),
|
|
|
|
central_content: Accounts::default(),
|
|
|
|
exit_allowed,
|
|
|
|
show_exit_progress: false,
|
|
|
|
allowed_modal_ids: vec![
|
|
|
|
Self::EXIT_MODAL_ID
|
|
|
|
],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ModalContainer for Root {
|
|
|
|
fn modal_ids(&self) -> &Vec<&'static str> {
|
|
|
|
&self.allowed_modal_ids
|
|
|
|
}
|
2023-07-13 03:54:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Root {
|
2023-07-14 00:55:31 +03:00
|
|
|
/// Identifier for exit confirmation [`Modal`].
|
|
|
|
pub const EXIT_MODAL_ID: &'static str = "exit_confirmation";
|
|
|
|
|
2023-07-13 03:54:27 +03:00
|
|
|
/// Default width of side panel at application UI.
|
2023-07-16 11:23:56 +03:00
|
|
|
pub const SIDE_PANEL_MIN_WIDTH: f32 = 400.0;
|
2023-07-13 03:54:27 +03:00
|
|
|
|
|
|
|
pub fn ui(&mut self, ui: &mut egui::Ui, frame: &mut eframe::Frame, cb: &dyn PlatformCallbacks) {
|
2023-07-14 00:55:31 +03:00
|
|
|
// Show opened exit confirmation Modal content.
|
|
|
|
if self.can_draw_modal() {
|
|
|
|
self.exit_modal_content(ui, frame, cb);
|
|
|
|
}
|
|
|
|
|
2023-07-13 03:54:27 +03:00
|
|
|
let (is_panel_open, panel_width) = Self::side_panel_state_width(frame);
|
|
|
|
egui::SidePanel::left("network_panel")
|
|
|
|
.resizable(false)
|
|
|
|
.exact_width(panel_width)
|
2023-07-14 00:55:31 +03:00
|
|
|
.frame(egui::Frame::none())
|
2023-07-13 03:54:27 +03:00
|
|
|
.show_animated_inside(ui, is_panel_open, |ui| {
|
2023-07-16 11:23:56 +03:00
|
|
|
// Show network content on side panel.
|
2023-07-14 00:55:31 +03:00
|
|
|
self.side_panel.ui(ui, frame, cb);
|
2023-07-13 03:54:27 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
egui::CentralPanel::default()
|
2023-07-14 00:55:31 +03:00
|
|
|
.frame(egui::Frame::none())
|
2023-07-13 03:54:27 +03:00
|
|
|
.show_inside(ui, |ui| {
|
2023-07-16 11:23:56 +03:00
|
|
|
// Show accounts content on central panel.
|
2023-07-14 00:55:31 +03:00
|
|
|
self.central_content.ui(ui, frame, cb);
|
2023-07-13 03:54:27 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get side panel state and width.
|
|
|
|
fn side_panel_state_width(frame: &mut eframe::Frame) -> (bool, f32) {
|
|
|
|
let dual_panel_mode = Self::is_dual_panel_mode(frame);
|
2023-07-14 00:55:31 +03:00
|
|
|
let is_panel_open = dual_panel_mode || Self::is_side_panel_open();
|
2023-07-13 03:54:27 +03:00
|
|
|
let panel_width = if dual_panel_mode {
|
2023-07-16 11:23:56 +03:00
|
|
|
Self::SIDE_PANEL_MIN_WIDTH + View::get_left_inset()
|
2023-07-13 03:54:27 +03:00
|
|
|
} else {
|
2023-07-16 11:23:56 +03:00
|
|
|
frame.info().window_info.size.x
|
2023-07-13 03:54:27 +03:00
|
|
|
};
|
|
|
|
(is_panel_open, panel_width)
|
|
|
|
}
|
|
|
|
|
2023-07-14 00:55:31 +03:00
|
|
|
/// Check if ui can show [`Network`] and [`Accounts`] at same time.
|
2023-07-13 03:54:27 +03:00
|
|
|
pub fn is_dual_panel_mode(frame: &mut eframe::Frame) -> bool {
|
|
|
|
let w = frame.info().window_info.size.x;
|
|
|
|
let h = frame.info().window_info.size.y;
|
|
|
|
// Screen is wide if width is greater than height or just 20% smaller.
|
|
|
|
let is_wide_screen = w > h || w + (w * 0.2) >= h;
|
|
|
|
// Dual panel mode is available when window is wide and its width is at least 2 times
|
2023-07-16 11:23:56 +03:00
|
|
|
// greater than minimal width of the side panel plus display insets from both sides.
|
|
|
|
let side_insets = View::get_left_inset() + View::get_right_inset();
|
|
|
|
is_wide_screen && w >= (Self::SIDE_PANEL_MIN_WIDTH * 2.0) + side_insets
|
2023-07-13 03:54:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Toggle [`Network`] panel state.
|
2023-07-14 00:55:31 +03:00
|
|
|
pub fn toggle_side_panel() {
|
|
|
|
let is_open = SIDE_PANEL_OPEN.load(Ordering::Relaxed);
|
|
|
|
SIDE_PANEL_OPEN.store(!is_open, Ordering::Relaxed);
|
2023-07-13 03:54:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Check if side panel is open.
|
2023-07-14 00:55:31 +03:00
|
|
|
pub fn is_side_panel_open() -> bool {
|
|
|
|
SIDE_PANEL_OPEN.load(Ordering::Relaxed)
|
2023-07-13 03:54:27 +03:00
|
|
|
}
|
|
|
|
|
2023-07-14 00:55:31 +03:00
|
|
|
/// Show exit confirmation modal.
|
|
|
|
pub fn show_exit_modal() {
|
2023-07-14 03:51:06 +03:00
|
|
|
let exit_modal = Modal::new(Self::EXIT_MODAL_ID).title(t!("modal.confirmation"));
|
2023-07-14 00:55:31 +03:00
|
|
|
Modal::show(exit_modal);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Draw exit confirmation modal content.
|
|
|
|
fn exit_modal_content(&mut self,
|
|
|
|
ui: &mut egui::Ui,
|
|
|
|
frame: &mut eframe::Frame,
|
|
|
|
cb: &dyn PlatformCallbacks) {
|
|
|
|
Modal::ui(ui, |ui, modal| {
|
|
|
|
if self.show_exit_progress {
|
|
|
|
if !Node::is_running() {
|
|
|
|
self.exit(frame, cb);
|
|
|
|
modal.close();
|
|
|
|
}
|
|
|
|
ui.add_space(16.0);
|
|
|
|
ui.vertical_centered(|ui| {
|
|
|
|
View::small_loading_spinner(ui);
|
|
|
|
ui.add_space(12.0);
|
|
|
|
ui.label(RichText::new(t!("sync_status.shutdown"))
|
2023-07-14 03:51:06 +03:00
|
|
|
.size(17.0)
|
2023-07-14 00:55:31 +03:00
|
|
|
.color(Colors::TEXT));
|
|
|
|
});
|
|
|
|
ui.add_space(10.0);
|
|
|
|
} else {
|
|
|
|
ui.add_space(8.0);
|
|
|
|
ui.vertical_centered(|ui| {
|
|
|
|
ui.label(RichText::new(t!("modal_exit.description"))
|
2023-07-14 03:51:06 +03:00
|
|
|
.size(17.0)
|
2023-07-14 00:55:31 +03:00
|
|
|
.color(Colors::TEXT));
|
|
|
|
});
|
|
|
|
ui.add_space(10.0);
|
|
|
|
|
|
|
|
// Show modal buttons.
|
|
|
|
ui.scope(|ui| {
|
|
|
|
// Setup spacing between buttons.
|
|
|
|
ui.spacing_mut().item_spacing = egui::Vec2::new(6.0, 0.0);
|
|
|
|
|
|
|
|
ui.columns(2, |columns| {
|
|
|
|
columns[0].vertical_centered_justified(|ui| {
|
|
|
|
View::button(ui, t!("modal_exit.exit"), Colors::WHITE, || {
|
|
|
|
if !Node::is_running() {
|
|
|
|
self.exit(frame, cb);
|
|
|
|
modal.close();
|
|
|
|
} else {
|
|
|
|
Node::stop(true);
|
|
|
|
modal.disable_closing();
|
|
|
|
self.show_exit_progress = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
columns[1].vertical_centered_justified(|ui| {
|
|
|
|
View::button(ui, t!("modal.cancel"), Colors::WHITE, || {
|
|
|
|
modal.close();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
ui.add_space(6.0);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Platform-specific exit from the application.
|
|
|
|
fn exit(&mut self, frame: &mut eframe::Frame, cb: &dyn PlatformCallbacks) {
|
|
|
|
match OperatingSystem::from_target_os() {
|
|
|
|
OperatingSystem::Android => {
|
|
|
|
cb.exit();
|
|
|
|
}
|
|
|
|
OperatingSystem::IOS => {
|
|
|
|
//TODO: exit on iOS.
|
|
|
|
}
|
|
|
|
OperatingSystem::Nix | OperatingSystem::Mac | OperatingSystem::Windows => {
|
|
|
|
self.exit_allowed = true;
|
|
|
|
frame.close();
|
|
|
|
}
|
|
|
|
OperatingSystem::Unknown => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handle platform-specific Back key code event.
|
|
|
|
pub fn on_back() {
|
2023-07-13 03:54:27 +03:00
|
|
|
if Modal::on_back() {
|
2023-07-14 00:55:31 +03:00
|
|
|
Self::show_exit_modal()
|
2023-07-13 03:54:27 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(dead_code)]
|
|
|
|
#[cfg(target_os = "android")]
|
|
|
|
#[allow(non_snake_case)]
|
|
|
|
#[no_mangle]
|
2023-07-14 00:55:31 +03:00
|
|
|
/// Handle Back key code event from Android.
|
|
|
|
pub extern "C" fn Java_mw_gri_android_MainActivity_onBack(
|
2023-07-13 03:54:27 +03:00
|
|
|
_env: jni::JNIEnv,
|
|
|
|
_class: jni::objects::JObject,
|
|
|
|
_activity: jni::objects::JObject,
|
|
|
|
) {
|
|
|
|
Root::on_back();
|
2023-07-14 00:55:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|