2775 lines
122 KiB
Rust
2775 lines
122 KiB
Rust
use std::{
|
|
cmp::{max, min},
|
|
collections::HashMap,
|
|
path::PathBuf,
|
|
time::Instant,
|
|
};
|
|
|
|
use unicode_segmentation::GraphemeCursor;
|
|
use unicode_width::{UnicodeWidthChar, UnicodeWidthStr};
|
|
|
|
use typed_builder::*;
|
|
|
|
use data_farmer::*;
|
|
use data_harvester::temperature;
|
|
use layout_manager::*;
|
|
pub use states::*;
|
|
|
|
use crate::{
|
|
constants,
|
|
data_conversion::ConvertedData,
|
|
options::Config,
|
|
options::ConfigFlags,
|
|
options::WidgetIdEnabled,
|
|
units::data_units::DataUnit,
|
|
utils::error::{BottomError, Result},
|
|
Pid,
|
|
};
|
|
|
|
use self::widgets::{ProcWidget, ProcWidgetMode};
|
|
|
|
pub mod data_farmer;
|
|
pub mod data_harvester;
|
|
pub mod layout_manager;
|
|
mod process_killer;
|
|
pub mod query;
|
|
pub mod states;
|
|
pub mod widgets;
|
|
|
|
const MAX_SEARCH_LENGTH: usize = 200;
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub enum AxisScaling {
|
|
Log,
|
|
Linear,
|
|
}
|
|
|
|
/// AppConfigFields is meant to cover basic fields that would normally be set
|
|
/// by config files or launch options.
|
|
#[derive(Debug)]
|
|
pub struct AppConfigFields {
|
|
pub update_rate_in_milliseconds: u64,
|
|
pub temperature_type: temperature::TemperatureType,
|
|
pub use_dot: bool,
|
|
pub left_legend: bool,
|
|
pub show_average_cpu: bool,
|
|
pub use_current_cpu_total: bool,
|
|
pub use_basic_mode: bool,
|
|
pub default_time_value: u64,
|
|
pub time_interval: u64,
|
|
pub hide_time: bool,
|
|
pub autohide_time: bool,
|
|
pub use_old_network_legend: bool,
|
|
pub table_gap: u16,
|
|
pub disable_click: bool,
|
|
pub no_write: bool,
|
|
pub show_table_scroll_position: bool,
|
|
pub is_advanced_kill: bool,
|
|
// TODO: Remove these, move network details state-side.
|
|
pub network_unit_type: DataUnit,
|
|
pub network_scale_type: AxisScaling,
|
|
pub network_use_binary_prefix: bool,
|
|
}
|
|
|
|
/// For filtering out information
|
|
#[derive(Debug, Clone)]
|
|
pub struct DataFilters {
|
|
pub disk_filter: Option<Filter>,
|
|
pub mount_filter: Option<Filter>,
|
|
pub temp_filter: Option<Filter>,
|
|
pub net_filter: Option<Filter>,
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct Filter {
|
|
pub is_list_ignored: bool,
|
|
pub list: Vec<regex::Regex>,
|
|
}
|
|
|
|
#[derive(TypedBuilder)]
|
|
pub struct App {
|
|
#[builder(default = false, setter(skip))]
|
|
awaiting_second_char: bool,
|
|
|
|
#[builder(default, setter(skip))]
|
|
second_char: Option<char>,
|
|
|
|
#[builder(default, setter(skip))]
|
|
pub dd_err: Option<String>,
|
|
|
|
#[builder(default, setter(skip))]
|
|
to_delete_process_list: Option<(String, Vec<Pid>)>,
|
|
|
|
#[builder(default = false, setter(skip))]
|
|
pub is_frozen: bool,
|
|
|
|
#[builder(default = Instant::now(), setter(skip))]
|
|
last_key_press: Instant,
|
|
|
|
#[builder(default, setter(skip))]
|
|
pub converted_data: ConvertedData,
|
|
|
|
#[builder(default, setter(skip))]
|
|
pub data_collection: DataCollection,
|
|
|
|
#[builder(default, setter(skip))]
|
|
pub delete_dialog_state: AppDeleteDialogState,
|
|
|
|
#[builder(default, setter(skip))]
|
|
pub help_dialog_state: AppHelpDialogState,
|
|
|
|
#[builder(default = false, setter(skip))]
|
|
pub is_expanded: bool,
|
|
|
|
#[builder(default = false, setter(skip))]
|
|
pub is_force_redraw: bool,
|
|
|
|
#[builder(default = false, setter(skip))]
|
|
pub is_determining_widget_boundary: bool,
|
|
|
|
#[builder(default = false, setter(skip))]
|
|
pub basic_mode_use_percent: bool,
|
|
|
|
#[cfg(target_family = "unix")]
|
|
#[builder(default, setter(skip))]
|
|
pub user_table: data_harvester::processes::UserTable,
|
|
|
|
pub cpu_state: CpuState,
|
|
pub mem_state: MemState,
|
|
pub net_state: NetState,
|
|
pub proc_state: ProcState,
|
|
pub temp_state: TempState,
|
|
pub disk_state: DiskState,
|
|
pub battery_state: BatteryState,
|
|
pub basic_table_widget_state: Option<BasicTableWidgetState>,
|
|
pub app_config_fields: AppConfigFields,
|
|
pub widget_map: HashMap<u64, BottomWidget>,
|
|
pub current_widget: BottomWidget,
|
|
pub used_widgets: UsedWidgets,
|
|
pub filters: DataFilters,
|
|
pub config: Config, // TODO: Is this even used...?
|
|
pub config_path: Option<PathBuf>, // TODO: Is this even used...?
|
|
}
|
|
|
|
#[cfg(target_os = "windows")]
|
|
const MAX_SIGNAL: usize = 1;
|
|
#[cfg(target_os = "linux")]
|
|
const MAX_SIGNAL: usize = 64;
|
|
#[cfg(target_os = "macos")]
|
|
const MAX_SIGNAL: usize = 31;
|
|
|
|
impl App {
|
|
pub fn reset(&mut self) {
|
|
// Reset multi
|
|
self.reset_multi_tap_keys();
|
|
|
|
// Reset dialog state
|
|
self.help_dialog_state.is_showing_help = false;
|
|
self.delete_dialog_state.is_showing_dd = false;
|
|
|
|
// Close all searches and reset it
|
|
self.proc_state
|
|
.widget_states
|
|
.values_mut()
|
|
.for_each(|state| {
|
|
state.proc_search.search_state.reset();
|
|
});
|
|
|
|
// Clear current delete list
|
|
self.to_delete_process_list = None;
|
|
self.dd_err = None;
|
|
|
|
// Unfreeze.
|
|
self.is_frozen = false;
|
|
|
|
// Reset zoom
|
|
self.reset_cpu_zoom();
|
|
self.reset_mem_zoom();
|
|
self.reset_net_zoom();
|
|
|
|
// Reset data
|
|
self.data_collection.reset();
|
|
}
|
|
|
|
pub fn should_get_widget_bounds(&self) -> bool {
|
|
self.is_force_redraw || self.is_determining_widget_boundary
|
|
}
|
|
|
|
fn close_dd(&mut self) {
|
|
self.delete_dialog_state.is_showing_dd = false;
|
|
self.delete_dialog_state.selected_signal = KillSignal::default();
|
|
self.delete_dialog_state.scroll_pos = 0;
|
|
self.to_delete_process_list = None;
|
|
self.dd_err = None;
|
|
}
|
|
|
|
pub fn on_esc(&mut self) {
|
|
self.reset_multi_tap_keys();
|
|
if self.is_in_dialog() {
|
|
if self.help_dialog_state.is_showing_help {
|
|
self.help_dialog_state.is_showing_help = false;
|
|
self.help_dialog_state.scroll_state.current_scroll_index = 0;
|
|
} else {
|
|
self.close_dd();
|
|
}
|
|
|
|
self.is_force_redraw = true;
|
|
} else {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Proc => {
|
|
if let Some(current_proc_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if current_proc_state.is_search_enabled() || current_proc_state.is_sort_open
|
|
{
|
|
current_proc_state.proc_search.search_state.is_enabled = false;
|
|
current_proc_state.is_sort_open = false;
|
|
self.is_force_redraw = true;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSearch => {
|
|
if let Some(current_proc_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 1)
|
|
{
|
|
if current_proc_state.is_search_enabled() {
|
|
current_proc_state.proc_search.search_state.is_enabled = false;
|
|
self.move_widget_selection(&WidgetDirection::Up);
|
|
self.is_force_redraw = true;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSort => {
|
|
if let Some(current_proc_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 2)
|
|
{
|
|
if current_proc_state.is_sort_open {
|
|
current_proc_state.is_sort_open = false;
|
|
self.move_widget_selection(&WidgetDirection::Right);
|
|
self.is_force_redraw = true;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
|
|
if self.is_expanded {
|
|
self.is_expanded = false;
|
|
self.is_force_redraw = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn is_in_search_widget(&self) -> bool {
|
|
matches!(
|
|
self.current_widget.widget_type,
|
|
BottomWidgetType::ProcSearch
|
|
)
|
|
}
|
|
|
|
fn reset_multi_tap_keys(&mut self) {
|
|
self.awaiting_second_char = false;
|
|
self.second_char = None;
|
|
}
|
|
|
|
fn is_in_dialog(&self) -> bool {
|
|
self.help_dialog_state.is_showing_help || self.delete_dialog_state.is_showing_dd
|
|
}
|
|
|
|
fn ignore_normal_keybinds(&self) -> bool {
|
|
self.is_in_dialog()
|
|
}
|
|
|
|
pub fn on_tab(&mut self) {
|
|
// Allow usage whilst only in processes
|
|
|
|
if !self.ignore_normal_keybinds() {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Cpu => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
cpu_widget_state.is_multi_graph_mode =
|
|
!cpu_widget_state.is_multi_graph_mode;
|
|
}
|
|
}
|
|
BottomWidgetType::Proc => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.toggle_tab();
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_slash(&mut self) {
|
|
if !self.ignore_normal_keybinds() {
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::Proc | BottomWidgetType::ProcSort => {
|
|
// Toggle on
|
|
if let Some(proc_widget_state) = self.proc_state.get_mut_widget_state(
|
|
self.current_widget.widget_id
|
|
- match &self.current_widget.widget_type {
|
|
BottomWidgetType::ProcSort => 2,
|
|
_ => 0,
|
|
},
|
|
) {
|
|
proc_widget_state.proc_search.search_state.is_enabled = true;
|
|
self.move_widget_selection(&WidgetDirection::Down);
|
|
self.is_force_redraw = true;
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn toggle_sort(&mut self) {
|
|
let widget_id = self.current_widget.widget_id
|
|
- match &self.current_widget.widget_type {
|
|
BottomWidgetType::Proc => 0,
|
|
BottomWidgetType::ProcSort => 2,
|
|
_ => 0,
|
|
};
|
|
|
|
if let Some(pws) = self.proc_state.get_mut_widget_state(widget_id) {
|
|
pws.is_sort_open = !pws.is_sort_open;
|
|
pws.force_update = true;
|
|
|
|
// If the sort is now open, move left. Otherwise, if the proc sort was selected, force move right.
|
|
if pws.is_sort_open {
|
|
if let SortState::Sortable(st) = &pws.table_state.sort_state {
|
|
pws.sort_table_state.current_scroll_position = st
|
|
.current_index
|
|
.clamp(0, pws.num_enabled_columns().saturating_sub(1));
|
|
}
|
|
self.move_widget_selection(&WidgetDirection::Left);
|
|
} else if let BottomWidgetType::ProcSort = self.current_widget.widget_type {
|
|
self.move_widget_selection(&WidgetDirection::Right);
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn invert_sort(&mut self) {
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::Proc | BottomWidgetType::ProcSort => {
|
|
let widget_id = self.current_widget.widget_id
|
|
- match &self.current_widget.widget_type {
|
|
BottomWidgetType::Proc => 0,
|
|
BottomWidgetType::ProcSort => 2,
|
|
_ => 0,
|
|
};
|
|
|
|
if let Some(proc_widget_state) = self.proc_state.get_mut_widget_state(widget_id) {
|
|
if let SortState::Sortable(state) =
|
|
&mut proc_widget_state.table_state.sort_state
|
|
{
|
|
state.toggle_order();
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
pub fn toggle_percentages(&mut self) {
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::BasicMem => {
|
|
self.basic_mode_use_percent = !self.basic_mode_use_percent; // Oh god this is so lazy.
|
|
}
|
|
BottomWidgetType::Proc => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.toggle_mem_percentage();
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
pub fn toggle_ignore_case(&mut self) {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
let mut is_case_sensitive: Option<bool> = None;
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget && proc_widget_state.is_search_enabled() {
|
|
proc_widget_state.proc_search.search_toggle_ignore_case();
|
|
proc_widget_state.update_query();
|
|
proc_widget_state.force_update = true;
|
|
|
|
// Remember, it's the opposite (ignoring case is case "in"sensitive)
|
|
is_case_sensitive = Some(!proc_widget_state.proc_search.is_ignoring_case);
|
|
}
|
|
}
|
|
|
|
// Also toggle it in the config file if we actually changed it.
|
|
if let Some(is_ignoring_case) = is_case_sensitive {
|
|
if let Some(flags) = &mut self.config.flags {
|
|
if let Some(map) = &mut flags.search_case_enabled_widgets_map {
|
|
// Just update the map.
|
|
let mapping = map.entry(self.current_widget.widget_id - 1).or_default();
|
|
*mapping = is_ignoring_case;
|
|
|
|
flags.search_case_enabled_widgets =
|
|
Some(WidgetIdEnabled::create_from_hashmap(map));
|
|
} else {
|
|
// Map doesn't exist yet... initialize ourselves.
|
|
let mut map = HashMap::default();
|
|
map.insert(self.current_widget.widget_id - 1, is_ignoring_case);
|
|
flags.search_case_enabled_widgets =
|
|
Some(WidgetIdEnabled::create_from_hashmap(&map));
|
|
flags.search_case_enabled_widgets_map = Some(map);
|
|
}
|
|
} else {
|
|
// Must initialize it ourselves...
|
|
let mut map = HashMap::default();
|
|
map.insert(self.current_widget.widget_id - 1, is_ignoring_case);
|
|
|
|
self.config.flags = Some(
|
|
ConfigFlags::builder()
|
|
.search_case_enabled_widgets(WidgetIdEnabled::create_from_hashmap(&map))
|
|
.search_case_enabled_widgets_map(map)
|
|
.build(),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn toggle_search_whole_word(&mut self) {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
let mut is_searching_whole_word: Option<bool> = None;
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget && proc_widget_state.is_search_enabled() {
|
|
proc_widget_state.proc_search.search_toggle_whole_word();
|
|
proc_widget_state.update_query();
|
|
proc_widget_state.force_update = true;
|
|
|
|
is_searching_whole_word =
|
|
Some(proc_widget_state.proc_search.is_searching_whole_word);
|
|
}
|
|
}
|
|
|
|
// Also toggle it in the config file if we actually changed it.
|
|
if let Some(is_searching_whole_word) = is_searching_whole_word {
|
|
if let Some(flags) = &mut self.config.flags {
|
|
if let Some(map) = &mut flags.search_whole_word_enabled_widgets_map {
|
|
// Just update the map.
|
|
let mapping = map.entry(self.current_widget.widget_id - 1).or_default();
|
|
*mapping = is_searching_whole_word;
|
|
|
|
flags.search_whole_word_enabled_widgets =
|
|
Some(WidgetIdEnabled::create_from_hashmap(map));
|
|
} else {
|
|
// Map doesn't exist yet... initialize ourselves.
|
|
let mut map = HashMap::default();
|
|
map.insert(self.current_widget.widget_id - 1, is_searching_whole_word);
|
|
flags.search_whole_word_enabled_widgets =
|
|
Some(WidgetIdEnabled::create_from_hashmap(&map));
|
|
flags.search_whole_word_enabled_widgets_map = Some(map);
|
|
}
|
|
} else {
|
|
// Must initialize it ourselves...
|
|
let mut map = HashMap::default();
|
|
map.insert(self.current_widget.widget_id - 1, is_searching_whole_word);
|
|
|
|
self.config.flags = Some(
|
|
ConfigFlags::builder()
|
|
.search_whole_word_enabled_widgets(WidgetIdEnabled::create_from_hashmap(
|
|
&map,
|
|
))
|
|
.search_whole_word_enabled_widgets_map(map)
|
|
.build(),
|
|
);
|
|
}
|
|
|
|
// self.did_config_fail_to_save = self.update_config_file().is_err();
|
|
}
|
|
}
|
|
|
|
pub fn toggle_search_regex(&mut self) {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
let mut is_searching_with_regex: Option<bool> = None;
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget && proc_widget_state.is_search_enabled() {
|
|
proc_widget_state.proc_search.search_toggle_regex();
|
|
proc_widget_state.update_query();
|
|
proc_widget_state.force_update = true;
|
|
|
|
is_searching_with_regex =
|
|
Some(proc_widget_state.proc_search.is_searching_with_regex);
|
|
}
|
|
}
|
|
|
|
// Also toggle it in the config file if we actually changed it.
|
|
if let Some(is_searching_whole_word) = is_searching_with_regex {
|
|
if let Some(flags) = &mut self.config.flags {
|
|
if let Some(map) = &mut flags.search_regex_enabled_widgets_map {
|
|
// Just update the map.
|
|
let mapping = map.entry(self.current_widget.widget_id - 1).or_default();
|
|
*mapping = is_searching_whole_word;
|
|
|
|
flags.search_regex_enabled_widgets =
|
|
Some(WidgetIdEnabled::create_from_hashmap(map));
|
|
} else {
|
|
// Map doesn't exist yet... initialize ourselves.
|
|
let mut map = HashMap::default();
|
|
map.insert(self.current_widget.widget_id - 1, is_searching_whole_word);
|
|
flags.search_regex_enabled_widgets =
|
|
Some(WidgetIdEnabled::create_from_hashmap(&map));
|
|
flags.search_regex_enabled_widgets_map = Some(map);
|
|
}
|
|
} else {
|
|
// Must initialize it ourselves...
|
|
let mut map = HashMap::default();
|
|
map.insert(self.current_widget.widget_id - 1, is_searching_whole_word);
|
|
|
|
self.config.flags = Some(
|
|
ConfigFlags::builder()
|
|
.search_regex_enabled_widgets(WidgetIdEnabled::create_from_hashmap(&map))
|
|
.search_regex_enabled_widgets_map(map)
|
|
.build(),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn toggle_tree_mode(&mut self) {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id))
|
|
{
|
|
match proc_widget_state.mode {
|
|
ProcWidgetMode::Tree { .. } => {
|
|
proc_widget_state.mode = ProcWidgetMode::Normal;
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
ProcWidgetMode::Normal => {
|
|
proc_widget_state.mode = ProcWidgetMode::Tree {
|
|
collapsed_pids: Default::default(),
|
|
};
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
ProcWidgetMode::Grouped => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// One of two functions allowed to run while in a dialog...
|
|
pub fn on_enter(&mut self) {
|
|
if self.delete_dialog_state.is_showing_dd {
|
|
if self.dd_err.is_some() {
|
|
self.close_dd();
|
|
} else if self.delete_dialog_state.selected_signal != KillSignal::Cancel {
|
|
// If within dd...
|
|
if self.dd_err.is_none() {
|
|
// Also ensure that we didn't just fail a dd...
|
|
let dd_result = self.kill_highlighted_process();
|
|
self.delete_dialog_state.scroll_pos = 0;
|
|
self.delete_dialog_state.selected_signal = KillSignal::default();
|
|
|
|
// Check if there was an issue... if so, inform the user.
|
|
if let Err(dd_err) = dd_result {
|
|
self.dd_err = Some(dd_err.to_string());
|
|
} else {
|
|
self.delete_dialog_state.is_showing_dd = false;
|
|
}
|
|
}
|
|
} else {
|
|
self.delete_dialog_state.scroll_pos = 0;
|
|
self.delete_dialog_state.selected_signal = KillSignal::default();
|
|
self.delete_dialog_state.is_showing_dd = false;
|
|
}
|
|
self.is_force_redraw = true;
|
|
} else if !self.is_in_dialog() {
|
|
if let BottomWidgetType::ProcSort = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 2))
|
|
{
|
|
proc_widget_state.use_sort_table_value();
|
|
self.move_widget_selection(&WidgetDirection::Right);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_delete(&mut self) {
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget {
|
|
if proc_widget_state.proc_search.search_state.is_enabled
|
|
&& proc_widget_state.get_search_cursor_position()
|
|
< proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len()
|
|
{
|
|
let current_cursor = proc_widget_state.get_search_cursor_position();
|
|
proc_widget_state
|
|
.search_walk_forward(proc_widget_state.get_search_cursor_position());
|
|
|
|
let _removed_chars: String = proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.drain(current_cursor..proc_widget_state.get_search_cursor_position())
|
|
.collect();
|
|
|
|
proc_widget_state.proc_search.search_state.grapheme_cursor =
|
|
GraphemeCursor::new(
|
|
current_cursor,
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len(),
|
|
true,
|
|
);
|
|
|
|
proc_widget_state.update_query();
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
} else {
|
|
self.start_killing_process()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_backspace(&mut self) {
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget
|
|
&& proc_widget_state.proc_search.search_state.is_enabled
|
|
&& proc_widget_state.get_search_cursor_position() > 0
|
|
{
|
|
let current_cursor = proc_widget_state.get_search_cursor_position();
|
|
proc_widget_state
|
|
.search_walk_back(proc_widget_state.get_search_cursor_position());
|
|
|
|
let removed_chars: String = proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.drain(proc_widget_state.get_search_cursor_position()..current_cursor)
|
|
.collect();
|
|
|
|
proc_widget_state.proc_search.search_state.grapheme_cursor =
|
|
GraphemeCursor::new(
|
|
proc_widget_state.get_search_cursor_position(),
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len(),
|
|
true,
|
|
);
|
|
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.char_cursor_position -= UnicodeWidthStr::width(removed_chars.as_str());
|
|
|
|
proc_widget_state.proc_search.search_state.cursor_direction =
|
|
CursorDirection::Left;
|
|
|
|
proc_widget_state.update_query();
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn get_process_filter(&self, widget_id: u64) -> &Option<query::Query> {
|
|
if let Some(process_widget_state) = self.proc_state.widget_states.get(&widget_id) {
|
|
&process_widget_state.proc_search.search_state.query
|
|
} else {
|
|
&None
|
|
}
|
|
}
|
|
|
|
#[cfg(target_family = "unix")]
|
|
pub fn on_number(&mut self, number_char: char) {
|
|
if self.delete_dialog_state.is_showing_dd {
|
|
if self
|
|
.delete_dialog_state
|
|
.last_number_press
|
|
.map_or(100, |ins| ins.elapsed().as_millis())
|
|
>= 400
|
|
{
|
|
self.delete_dialog_state.keyboard_signal_select = 0;
|
|
}
|
|
let mut kbd_signal = self.delete_dialog_state.keyboard_signal_select * 10;
|
|
kbd_signal += number_char.to_digit(10).unwrap() as usize;
|
|
if kbd_signal > 64 {
|
|
kbd_signal %= 100;
|
|
}
|
|
#[cfg(target_os = "linux")]
|
|
if kbd_signal > 64 || kbd_signal == 32 || kbd_signal == 33 {
|
|
kbd_signal %= 10;
|
|
}
|
|
#[cfg(target_os = "macos")]
|
|
if kbd_signal > 31 {
|
|
kbd_signal %= 10;
|
|
}
|
|
self.delete_dialog_state.selected_signal = KillSignal::Kill(kbd_signal);
|
|
if kbd_signal < 10 {
|
|
self.delete_dialog_state.keyboard_signal_select = kbd_signal;
|
|
} else {
|
|
self.delete_dialog_state.keyboard_signal_select = 0;
|
|
}
|
|
self.delete_dialog_state.last_number_press = Some(Instant::now());
|
|
}
|
|
}
|
|
|
|
pub fn on_up_key(&mut self) {
|
|
if !self.is_in_dialog() {
|
|
self.decrement_position_count();
|
|
} else if self.help_dialog_state.is_showing_help {
|
|
self.help_scroll_up();
|
|
} else if self.delete_dialog_state.is_showing_dd {
|
|
#[cfg(target_os = "windows")]
|
|
self.on_right_key();
|
|
#[cfg(target_family = "unix")]
|
|
{
|
|
if self.app_config_fields.is_advanced_kill {
|
|
self.on_left_key();
|
|
} else {
|
|
self.on_right_key();
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
self.reset_multi_tap_keys();
|
|
}
|
|
|
|
pub fn on_down_key(&mut self) {
|
|
if !self.is_in_dialog() {
|
|
self.increment_position_count();
|
|
} else if self.help_dialog_state.is_showing_help {
|
|
self.help_scroll_down();
|
|
} else if self.delete_dialog_state.is_showing_dd {
|
|
#[cfg(target_os = "windows")]
|
|
self.on_left_key();
|
|
#[cfg(target_family = "unix")]
|
|
{
|
|
if self.app_config_fields.is_advanced_kill {
|
|
self.on_right_key();
|
|
} else {
|
|
self.on_left_key();
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
self.reset_multi_tap_keys();
|
|
}
|
|
|
|
pub fn on_left_key(&mut self) {
|
|
if !self.is_in_dialog() {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::ProcSearch => {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 1)
|
|
{
|
|
if is_in_search_widget {
|
|
let prev_cursor = proc_widget_state.get_search_cursor_position();
|
|
proc_widget_state
|
|
.search_walk_back(proc_widget_state.get_search_cursor_position());
|
|
if proc_widget_state.get_search_cursor_position() < prev_cursor {
|
|
let str_slice = &proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
[proc_widget_state.get_search_cursor_position()..prev_cursor];
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.char_cursor_position -= UnicodeWidthStr::width(str_slice);
|
|
proc_widget_state.proc_search.search_state.cursor_direction =
|
|
CursorDirection::Left;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Battery => {
|
|
if !self.converted_data.battery_data.is_empty() {
|
|
if let Some(battery_widget_state) = self
|
|
.battery_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if battery_widget_state.currently_selected_battery_index > 0 {
|
|
battery_widget_state.currently_selected_battery_index -= 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
} else if self.delete_dialog_state.is_showing_dd {
|
|
#[cfg(target_family = "unix")]
|
|
{
|
|
if self.app_config_fields.is_advanced_kill {
|
|
match self.delete_dialog_state.selected_signal {
|
|
KillSignal::Kill(prev_signal) => {
|
|
self.delete_dialog_state.selected_signal = match prev_signal - 1 {
|
|
0 => KillSignal::Cancel,
|
|
// 32+33 are skipped
|
|
33 => KillSignal::Kill(31),
|
|
signal => KillSignal::Kill(signal),
|
|
};
|
|
}
|
|
KillSignal::Cancel => {}
|
|
};
|
|
} else {
|
|
self.delete_dialog_state.selected_signal = KillSignal::default();
|
|
}
|
|
}
|
|
#[cfg(target_os = "windows")]
|
|
{
|
|
self.delete_dialog_state.selected_signal = KillSignal::Kill(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_right_key(&mut self) {
|
|
if !self.is_in_dialog() {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::ProcSearch => {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 1)
|
|
{
|
|
if is_in_search_widget {
|
|
let prev_cursor = proc_widget_state.get_search_cursor_position();
|
|
proc_widget_state.search_walk_forward(
|
|
proc_widget_state.get_search_cursor_position(),
|
|
);
|
|
if proc_widget_state.get_search_cursor_position() > prev_cursor {
|
|
let str_slice = &proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
[prev_cursor..proc_widget_state.get_search_cursor_position()];
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.char_cursor_position += UnicodeWidthStr::width(str_slice);
|
|
proc_widget_state.proc_search.search_state.cursor_direction =
|
|
CursorDirection::Right;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Battery => {
|
|
if !self.converted_data.battery_data.is_empty() {
|
|
let battery_count = self.converted_data.battery_data.len();
|
|
if let Some(battery_widget_state) = self
|
|
.battery_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if battery_widget_state.currently_selected_battery_index
|
|
< battery_count - 1
|
|
{
|
|
battery_widget_state.currently_selected_battery_index += 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
} else if self.delete_dialog_state.is_showing_dd {
|
|
#[cfg(target_family = "unix")]
|
|
{
|
|
if self.app_config_fields.is_advanced_kill {
|
|
let new_signal = match self.delete_dialog_state.selected_signal {
|
|
KillSignal::Cancel => 1,
|
|
// 32+33 are skipped
|
|
#[cfg(target_os = "linux")]
|
|
KillSignal::Kill(31) => 34,
|
|
#[cfg(target_os = "macos")]
|
|
KillSignal::Kill(31) => 31,
|
|
KillSignal::Kill(64) => 64,
|
|
KillSignal::Kill(signal) => signal + 1,
|
|
};
|
|
self.delete_dialog_state.selected_signal = KillSignal::Kill(new_signal);
|
|
} else {
|
|
self.delete_dialog_state.selected_signal = KillSignal::Cancel;
|
|
}
|
|
}
|
|
#[cfg(target_os = "windows")]
|
|
{
|
|
self.delete_dialog_state.selected_signal = KillSignal::Cancel;
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_page_up(&mut self) {
|
|
if self.delete_dialog_state.is_showing_dd {
|
|
let mut new_signal = match self.delete_dialog_state.selected_signal {
|
|
KillSignal::Cancel => 0,
|
|
KillSignal::Kill(signal) => max(signal, 8) - 8,
|
|
};
|
|
if new_signal > 23 && new_signal < 33 {
|
|
new_signal -= 2;
|
|
}
|
|
self.delete_dialog_state.selected_signal = match new_signal {
|
|
0 => KillSignal::Cancel,
|
|
sig => KillSignal::Kill(sig),
|
|
};
|
|
} else if self.current_widget.widget_type.is_widget_table() {
|
|
if let (Some((_tlc_x, tlc_y)), Some((_brc_x, brc_y))) = (
|
|
&self.current_widget.top_left_corner,
|
|
&self.current_widget.bottom_right_corner,
|
|
) {
|
|
let border_offset = if self.is_drawing_border() { 1 } else { 0 };
|
|
let header_offset = self.header_offset(&self.current_widget);
|
|
let height = brc_y - tlc_y - 2 * border_offset - header_offset;
|
|
self.change_position_count(-(height as i64));
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_page_down(&mut self) {
|
|
if self.delete_dialog_state.is_showing_dd {
|
|
let mut new_signal = match self.delete_dialog_state.selected_signal {
|
|
KillSignal::Cancel => 8,
|
|
KillSignal::Kill(signal) => min(signal + 8, MAX_SIGNAL),
|
|
};
|
|
if new_signal > 31 && new_signal < 42 {
|
|
new_signal += 2;
|
|
}
|
|
self.delete_dialog_state.selected_signal = KillSignal::Kill(new_signal);
|
|
} else if self.current_widget.widget_type.is_widget_table() {
|
|
if let (Some((_tlc_x, tlc_y)), Some((_brc_x, brc_y))) = (
|
|
&self.current_widget.top_left_corner,
|
|
&self.current_widget.bottom_right_corner,
|
|
) {
|
|
let border_offset = if self.is_drawing_border() { 1 } else { 0 };
|
|
let header_offset = self.header_offset(&self.current_widget);
|
|
let height = brc_y - tlc_y - 2 * border_offset - header_offset;
|
|
self.change_position_count(height as i64);
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn skip_cursor_beginning(&mut self) {
|
|
if !self.ignore_normal_keybinds() {
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget {
|
|
proc_widget_state.proc_search.search_state.grapheme_cursor =
|
|
GraphemeCursor::new(
|
|
0,
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len(),
|
|
true,
|
|
);
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.char_cursor_position = 0;
|
|
proc_widget_state.proc_search.search_state.cursor_direction =
|
|
CursorDirection::Left;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn skip_cursor_end(&mut self) {
|
|
if !self.ignore_normal_keybinds() {
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget {
|
|
proc_widget_state.proc_search.search_state.grapheme_cursor =
|
|
GraphemeCursor::new(
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len(),
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len(),
|
|
true,
|
|
);
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.char_cursor_position = UnicodeWidthStr::width(
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.as_str(),
|
|
);
|
|
proc_widget_state.proc_search.search_state.cursor_direction =
|
|
CursorDirection::Right;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn clear_search(&mut self) {
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
proc_widget_state.clear_search();
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn clear_previous_word(&mut self) {
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
// Traverse backwards from the current cursor location until you hit non-whitespace characters,
|
|
// then continue to traverse (and delete) backwards until you hit a whitespace character. Halt.
|
|
|
|
// So... first, let's get our current cursor position using graphemes...
|
|
let end_index = proc_widget_state.get_char_cursor_position();
|
|
|
|
// Then, let's crawl backwards until we hit our location, and store the "head"...
|
|
let query = proc_widget_state.get_current_search_query();
|
|
let mut start_index = 0;
|
|
let mut saw_non_whitespace = false;
|
|
|
|
for (itx, c) in query
|
|
.chars()
|
|
.rev()
|
|
.enumerate()
|
|
.skip(query.len() - end_index)
|
|
{
|
|
if c.is_whitespace() {
|
|
if saw_non_whitespace {
|
|
start_index = query.len() - itx;
|
|
break;
|
|
}
|
|
} else {
|
|
saw_non_whitespace = true;
|
|
}
|
|
}
|
|
|
|
let removed_chars: String = proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.drain(start_index..end_index)
|
|
.collect();
|
|
|
|
proc_widget_state.proc_search.search_state.grapheme_cursor = GraphemeCursor::new(
|
|
start_index,
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len(),
|
|
true,
|
|
);
|
|
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.char_cursor_position -= UnicodeWidthStr::width(removed_chars.as_str());
|
|
|
|
proc_widget_state.proc_search.search_state.cursor_direction = CursorDirection::Left;
|
|
|
|
proc_widget_state.update_query();
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn start_killing_process(&mut self) {
|
|
self.reset_multi_tap_keys();
|
|
|
|
if let Some(pws) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get(&self.current_widget.widget_id)
|
|
{
|
|
if let Some(table_row) = pws
|
|
.table_data
|
|
.data
|
|
.get(pws.table_state.current_scroll_position)
|
|
{
|
|
if let Some(col_value) = table_row.row().get(ProcWidget::PROC_NAME_OR_CMD) {
|
|
let val = col_value.main_text().to_string();
|
|
if pws.is_using_command() {
|
|
if let Some(pids) = self.data_collection.process_data.cmd_pid_map.get(&val)
|
|
{
|
|
let current_process = (val, pids.clone());
|
|
|
|
self.to_delete_process_list = Some(current_process);
|
|
self.delete_dialog_state.is_showing_dd = true;
|
|
self.is_determining_widget_boundary = true;
|
|
}
|
|
} else if let Some(pids) =
|
|
self.data_collection.process_data.name_pid_map.get(&val)
|
|
{
|
|
let current_process = (val, pids.clone());
|
|
|
|
self.to_delete_process_list = Some(current_process);
|
|
self.delete_dialog_state.is_showing_dd = true;
|
|
self.is_determining_widget_boundary = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn on_char_key(&mut self, caught_char: char) {
|
|
// Skip control code chars
|
|
if caught_char.is_control() {
|
|
return;
|
|
}
|
|
|
|
// Forbid any char key presses when showing a dialog box...
|
|
if !self.ignore_normal_keybinds() {
|
|
let current_key_press_inst = Instant::now();
|
|
if current_key_press_inst
|
|
.duration_since(self.last_key_press)
|
|
.as_millis()
|
|
> constants::MAX_KEY_TIMEOUT_IN_MILLISECONDS.into()
|
|
{
|
|
self.reset_multi_tap_keys();
|
|
}
|
|
self.last_key_press = current_key_press_inst;
|
|
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
let is_in_search_widget = self.is_in_search_widget();
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
if is_in_search_widget
|
|
&& proc_widget_state.is_search_enabled()
|
|
&& UnicodeWidthStr::width(
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.as_str(),
|
|
) <= MAX_SEARCH_LENGTH
|
|
{
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.insert(proc_widget_state.get_search_cursor_position(), caught_char);
|
|
|
|
proc_widget_state.proc_search.search_state.grapheme_cursor =
|
|
GraphemeCursor::new(
|
|
proc_widget_state.get_search_cursor_position(),
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.current_search_query
|
|
.len(),
|
|
true,
|
|
);
|
|
proc_widget_state
|
|
.search_walk_forward(proc_widget_state.get_search_cursor_position());
|
|
|
|
proc_widget_state
|
|
.proc_search
|
|
.search_state
|
|
.char_cursor_position +=
|
|
UnicodeWidthChar::width(caught_char).unwrap_or(0);
|
|
|
|
proc_widget_state.update_query();
|
|
proc_widget_state.force_update = true;
|
|
proc_widget_state.proc_search.search_state.cursor_direction =
|
|
CursorDirection::Right;
|
|
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
self.handle_char(caught_char);
|
|
} else if self.help_dialog_state.is_showing_help {
|
|
match caught_char {
|
|
'1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9' => {
|
|
let potential_index = caught_char.to_digit(10);
|
|
if let Some(potential_index) = potential_index {
|
|
let potential_index = potential_index as usize;
|
|
if (potential_index) < self.help_dialog_state.index_shortcuts.len() {
|
|
self.help_scroll_to_or_max(
|
|
self.help_dialog_state.index_shortcuts[potential_index],
|
|
);
|
|
}
|
|
}
|
|
}
|
|
'j' | 'k' | 'g' | 'G' => self.handle_char(caught_char),
|
|
_ => {}
|
|
}
|
|
} else if self.delete_dialog_state.is_showing_dd {
|
|
match caught_char {
|
|
'h' => self.on_left_key(),
|
|
'j' => self.on_down_key(),
|
|
'k' => self.on_up_key(),
|
|
'l' => self.on_right_key(),
|
|
#[cfg(target_family = "unix")]
|
|
'0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9' => {
|
|
self.on_number(caught_char)
|
|
}
|
|
'g' => {
|
|
let mut is_first_g = true;
|
|
if let Some(second_char) = self.second_char {
|
|
if self.awaiting_second_char && second_char == 'g' {
|
|
is_first_g = false;
|
|
self.awaiting_second_char = false;
|
|
self.second_char = None;
|
|
self.skip_to_first();
|
|
}
|
|
}
|
|
|
|
if is_first_g {
|
|
self.awaiting_second_char = true;
|
|
self.second_char = Some('g');
|
|
}
|
|
}
|
|
'G' => self.skip_to_last(),
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn handle_char(&mut self, caught_char: char) {
|
|
match caught_char {
|
|
'/' => {
|
|
self.on_slash();
|
|
}
|
|
'd' => {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
let mut is_first_d = true;
|
|
if let Some(second_char) = self.second_char {
|
|
if self.awaiting_second_char && second_char == 'd' {
|
|
is_first_d = false;
|
|
self.awaiting_second_char = false;
|
|
self.second_char = None;
|
|
|
|
self.start_killing_process();
|
|
}
|
|
}
|
|
|
|
if is_first_d {
|
|
self.awaiting_second_char = true;
|
|
self.second_char = Some('d');
|
|
}
|
|
}
|
|
}
|
|
'g' => {
|
|
let mut is_first_g = true;
|
|
if let Some(second_char) = self.second_char {
|
|
if self.awaiting_second_char && second_char == 'g' {
|
|
is_first_g = false;
|
|
self.awaiting_second_char = false;
|
|
self.second_char = None;
|
|
self.skip_to_first();
|
|
}
|
|
}
|
|
|
|
if is_first_g {
|
|
self.awaiting_second_char = true;
|
|
self.second_char = Some('g');
|
|
}
|
|
}
|
|
'G' => self.skip_to_last(),
|
|
'k' => self.on_up_key(),
|
|
'j' => self.on_down_key(),
|
|
'f' => {
|
|
self.is_frozen = !self.is_frozen;
|
|
if self.is_frozen {
|
|
self.data_collection.freeze();
|
|
} else {
|
|
self.data_collection.thaw();
|
|
}
|
|
}
|
|
'c' => {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.select_column(ProcWidget::CPU);
|
|
}
|
|
}
|
|
}
|
|
'm' => {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.select_column(ProcWidget::MEM);
|
|
}
|
|
}
|
|
}
|
|
'p' => {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.select_column(ProcWidget::PID_OR_COUNT);
|
|
}
|
|
}
|
|
}
|
|
'P' => {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.toggle_command();
|
|
}
|
|
}
|
|
}
|
|
'n' => {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.select_column(ProcWidget::PROC_NAME_OR_CMD);
|
|
}
|
|
}
|
|
}
|
|
'?' => {
|
|
self.help_dialog_state.is_showing_help = true;
|
|
self.is_force_redraw = true;
|
|
}
|
|
'H' | 'A' => self.move_widget_selection(&WidgetDirection::Left),
|
|
'L' | 'D' => self.move_widget_selection(&WidgetDirection::Right),
|
|
'K' | 'W' => self.move_widget_selection(&WidgetDirection::Up),
|
|
'J' | 'S' => self.move_widget_selection(&WidgetDirection::Down),
|
|
't' => self.toggle_tree_mode(),
|
|
'+' => self.on_plus(),
|
|
'-' => self.on_minus(),
|
|
'=' => self.reset_zoom(),
|
|
'e' => self.toggle_expand_widget(),
|
|
's' => self.toggle_sort(),
|
|
'I' => self.invert_sort(),
|
|
'%' => self.toggle_percentages(),
|
|
_ => {}
|
|
}
|
|
|
|
if let Some(second_char) = self.second_char {
|
|
if self.awaiting_second_char && caught_char != second_char {
|
|
self.awaiting_second_char = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn kill_highlighted_process(&mut self) -> Result<()> {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
if let Some(current_selected_processes) = &self.to_delete_process_list {
|
|
#[cfg(target_family = "unix")]
|
|
let signal = match self.delete_dialog_state.selected_signal {
|
|
KillSignal::Kill(sig) => sig,
|
|
KillSignal::Cancel => 15, // should never happen, so just TERM
|
|
};
|
|
for pid in ¤t_selected_processes.1 {
|
|
#[cfg(target_family = "unix")]
|
|
{
|
|
process_killer::kill_process_given_pid(*pid, signal)?;
|
|
}
|
|
#[cfg(target_os = "windows")]
|
|
{
|
|
process_killer::kill_process_given_pid(*pid)?;
|
|
}
|
|
}
|
|
}
|
|
self.to_delete_process_list = None;
|
|
Ok(())
|
|
} else {
|
|
Err(BottomError::GenericError(
|
|
"Cannot kill processes if the current widget is not the Process widget!"
|
|
.to_string(),
|
|
))
|
|
}
|
|
}
|
|
|
|
pub fn get_to_delete_processes(&self) -> Option<(String, Vec<Pid>)> {
|
|
self.to_delete_process_list.clone()
|
|
}
|
|
|
|
fn toggle_expand_widget(&mut self) {
|
|
if self.is_expanded {
|
|
self.is_expanded = false;
|
|
self.is_force_redraw = true;
|
|
} else {
|
|
self.expand_widget();
|
|
}
|
|
}
|
|
|
|
fn expand_widget(&mut self) {
|
|
// TODO: [BASIC] Expansion in basic mode.
|
|
if !self.ignore_normal_keybinds() && !self.app_config_fields.use_basic_mode {
|
|
// Pop-out mode. We ignore if in process search.
|
|
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::ProcSearch => {}
|
|
_ => {
|
|
self.is_expanded = true;
|
|
self.is_force_redraw = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn move_widget_selection(&mut self, direction: &WidgetDirection) {
|
|
// Since we only want to call reset once, we do it like this to avoid
|
|
// redundant calls on recursion.
|
|
self.move_widget_selection_logic(direction);
|
|
self.reset_multi_tap_keys();
|
|
}
|
|
|
|
fn move_widget_selection_logic(&mut self, direction: &WidgetDirection) {
|
|
/*
|
|
The actual logic for widget movement.
|
|
|
|
We follow these following steps:
|
|
1. Send a movement signal in `direction`.
|
|
2. Check if this new widget we've landed on is hidden. If not, halt.
|
|
3. If it hidden, loop and either send:
|
|
- A signal equal to the current direction, if it is opposite of the reflection.
|
|
- Reflection direction.
|
|
*/
|
|
|
|
if !self.ignore_normal_keybinds() && !self.is_expanded {
|
|
if let Some(new_widget_id) = &(match direction {
|
|
WidgetDirection::Left => self.current_widget.left_neighbour,
|
|
WidgetDirection::Right => self.current_widget.right_neighbour,
|
|
WidgetDirection::Up => self.current_widget.up_neighbour,
|
|
WidgetDirection::Down => self.current_widget.down_neighbour,
|
|
}) {
|
|
if let Some(new_widget) = self.widget_map.get(new_widget_id) {
|
|
match &new_widget.widget_type {
|
|
BottomWidgetType::Temp
|
|
| BottomWidgetType::Proc
|
|
| BottomWidgetType::ProcSort
|
|
| BottomWidgetType::Disk
|
|
| BottomWidgetType::Battery
|
|
if self.basic_table_widget_state.is_some()
|
|
&& (*direction == WidgetDirection::Left
|
|
|| *direction == WidgetDirection::Right) =>
|
|
{
|
|
// Gotta do this for the sort widget
|
|
if let BottomWidgetType::ProcSort = new_widget.widget_type {
|
|
if let Some(proc_widget_state) =
|
|
self.proc_state.widget_states.get(&(new_widget_id - 2))
|
|
{
|
|
if proc_widget_state.is_sort_open {
|
|
self.current_widget = new_widget.clone();
|
|
} else if let Some(next_new_widget_id) = match direction {
|
|
WidgetDirection::Left => new_widget.left_neighbour,
|
|
_ => new_widget.right_neighbour,
|
|
} {
|
|
if let Some(next_new_widget) =
|
|
self.widget_map.get(&next_new_widget_id)
|
|
{
|
|
self.current_widget = next_new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
|
|
if let Some(basic_table_widget_state) =
|
|
&mut self.basic_table_widget_state
|
|
{
|
|
basic_table_widget_state.currently_displayed_widget_id =
|
|
self.current_widget.widget_id;
|
|
basic_table_widget_state.currently_displayed_widget_type =
|
|
self.current_widget.widget_type.clone();
|
|
}
|
|
|
|
// And let's not forget:
|
|
self.is_determining_widget_boundary = true;
|
|
}
|
|
BottomWidgetType::BasicTables => {
|
|
match &direction {
|
|
WidgetDirection::Up => {
|
|
// Note this case would fail if it moved up into a hidden
|
|
// widget, but it's for basic so whatever, it's all hard-coded
|
|
// right now anyways...
|
|
if let Some(next_new_widget_id) = new_widget.up_neighbour {
|
|
if let Some(next_new_widget) =
|
|
self.widget_map.get(&next_new_widget_id)
|
|
{
|
|
self.current_widget = next_new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
WidgetDirection::Down => {
|
|
// Assuming we're in basic mode (BasicTables), then
|
|
// we want to move DOWN to the currently shown widget.
|
|
if let Some(basic_table_widget_state) =
|
|
&mut self.basic_table_widget_state
|
|
{
|
|
// We also want to move towards Proc if we had set it to ProcSort.
|
|
if let BottomWidgetType::ProcSort =
|
|
basic_table_widget_state.currently_displayed_widget_type
|
|
{
|
|
basic_table_widget_state
|
|
.currently_displayed_widget_type =
|
|
BottomWidgetType::Proc;
|
|
basic_table_widget_state
|
|
.currently_displayed_widget_id -= 2;
|
|
}
|
|
|
|
if let Some(next_new_widget) = self.widget_map.get(
|
|
&basic_table_widget_state.currently_displayed_widget_id,
|
|
) {
|
|
self.current_widget = next_new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
_ => self.current_widget = new_widget.clone(),
|
|
}
|
|
}
|
|
_ if new_widget.parent_reflector.is_some() => {
|
|
// It may be hidden...
|
|
if let Some((parent_direction, offset)) = &new_widget.parent_reflector {
|
|
if direction.is_opposite(parent_direction) {
|
|
// Keep going in the current direction if hidden...
|
|
// unless we hit a wall of sorts.
|
|
let option_next_neighbour_id = match &direction {
|
|
WidgetDirection::Left => new_widget.left_neighbour,
|
|
WidgetDirection::Right => new_widget.right_neighbour,
|
|
WidgetDirection::Up => new_widget.up_neighbour,
|
|
WidgetDirection::Down => new_widget.down_neighbour,
|
|
};
|
|
match &new_widget.widget_type {
|
|
BottomWidgetType::CpuLegend => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get(&(new_widget_id - *offset))
|
|
{
|
|
if cpu_widget_state.is_legend_hidden {
|
|
if let Some(next_neighbour_id) =
|
|
option_next_neighbour_id
|
|
{
|
|
if let Some(next_neighbour_widget) =
|
|
self.widget_map.get(&next_neighbour_id)
|
|
{
|
|
self.current_widget =
|
|
next_neighbour_widget.clone();
|
|
}
|
|
}
|
|
} else {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSearch
|
|
| BottomWidgetType::ProcSort => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get(&(new_widget_id - *offset))
|
|
{
|
|
match &new_widget.widget_type {
|
|
BottomWidgetType::ProcSearch => {
|
|
if !proc_widget_state.is_search_enabled() {
|
|
if let Some(next_neighbour_id) =
|
|
option_next_neighbour_id
|
|
{
|
|
if let Some(next_neighbour_widget) =
|
|
self.widget_map
|
|
.get(&next_neighbour_id)
|
|
{
|
|
self.current_widget =
|
|
next_neighbour_widget
|
|
.clone();
|
|
}
|
|
}
|
|
} else {
|
|
self.current_widget =
|
|
new_widget.clone();
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSort => {
|
|
if !proc_widget_state.is_sort_open {
|
|
if let Some(next_neighbour_id) =
|
|
option_next_neighbour_id
|
|
{
|
|
if let Some(next_neighbour_widget) =
|
|
self.widget_map
|
|
.get(&next_neighbour_id)
|
|
{
|
|
self.current_widget =
|
|
next_neighbour_widget
|
|
.clone();
|
|
}
|
|
}
|
|
} else {
|
|
self.current_widget =
|
|
new_widget.clone();
|
|
}
|
|
}
|
|
_ => {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
} else {
|
|
// Reflect
|
|
match &new_widget.widget_type {
|
|
BottomWidgetType::CpuLegend => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get(&(new_widget_id - *offset))
|
|
{
|
|
if cpu_widget_state.is_legend_hidden {
|
|
if let Some(parent_cpu_widget) = self
|
|
.widget_map
|
|
.get(&(new_widget_id - *offset))
|
|
{
|
|
self.current_widget =
|
|
parent_cpu_widget.clone();
|
|
}
|
|
} else {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSearch
|
|
| BottomWidgetType::ProcSort => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get(&(new_widget_id - *offset))
|
|
{
|
|
match &new_widget.widget_type {
|
|
BottomWidgetType::ProcSearch => {
|
|
if !proc_widget_state.is_search_enabled() {
|
|
if let Some(parent_proc_widget) = self
|
|
.widget_map
|
|
.get(&(new_widget_id - *offset))
|
|
{
|
|
self.current_widget =
|
|
parent_proc_widget.clone();
|
|
}
|
|
} else {
|
|
self.current_widget =
|
|
new_widget.clone();
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSort => {
|
|
if !proc_widget_state.is_sort_open {
|
|
if let Some(parent_proc_widget) = self
|
|
.widget_map
|
|
.get(&(new_widget_id - *offset))
|
|
{
|
|
self.current_widget =
|
|
parent_proc_widget.clone();
|
|
}
|
|
} else {
|
|
self.current_widget =
|
|
new_widget.clone();
|
|
}
|
|
}
|
|
_ => {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {
|
|
// Cannot be hidden, does not special treatment.
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
|
|
let mut reflection_dir: Option<WidgetDirection> = None;
|
|
if let Some((parent_direction, offset)) = &self.current_widget.parent_reflector
|
|
{
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::CpuLegend => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get(&(self.current_widget.widget_id - *offset))
|
|
{
|
|
if cpu_widget_state.is_legend_hidden {
|
|
reflection_dir = Some(parent_direction.clone());
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSearch | BottomWidgetType::ProcSort => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get(&(self.current_widget.widget_id - *offset))
|
|
{
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::ProcSearch => {
|
|
if !proc_widget_state.is_search_enabled() {
|
|
reflection_dir = Some(parent_direction.clone());
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSort => {
|
|
if !proc_widget_state.is_sort_open {
|
|
reflection_dir = Some(parent_direction.clone());
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
if let Some(ref_dir) = &reflection_dir {
|
|
self.move_widget_selection_logic(ref_dir);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
match direction {
|
|
WidgetDirection::Left => self.handle_left_expanded_movement(),
|
|
WidgetDirection::Right => self.handle_right_expanded_movement(),
|
|
WidgetDirection::Up => {
|
|
if let BottomWidgetType::ProcSearch = self.current_widget.widget_type {
|
|
if let Some(current_widget) =
|
|
self.widget_map.get(&self.current_widget.widget_id)
|
|
{
|
|
if let Some(new_widget_id) = current_widget.up_neighbour {
|
|
if let Some(new_widget) = self.widget_map.get(&new_widget_id) {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
WidgetDirection::Down => match &self.current_widget.widget_type {
|
|
BottomWidgetType::Proc | BottomWidgetType::ProcSort => {
|
|
let widget_id = self.current_widget.widget_id
|
|
- match &self.current_widget.widget_type {
|
|
BottomWidgetType::ProcSort => 2,
|
|
_ => 0,
|
|
};
|
|
if let Some(current_widget) = self.widget_map.get(&widget_id) {
|
|
if let Some(new_widget_id) = current_widget.down_neighbour {
|
|
if let Some(new_widget) = self.widget_map.get(&new_widget_id) {
|
|
if let Some(proc_widget_state) =
|
|
self.proc_state.get_widget_state(widget_id)
|
|
{
|
|
if proc_widget_state.is_search_enabled() {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
fn handle_left_expanded_movement(&mut self) {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
if let Some(new_widget_id) = self.current_widget.left_neighbour {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get(&self.current_widget.widget_id)
|
|
{
|
|
if proc_widget_state.is_sort_open {
|
|
if let Some(proc_sort_widget) = self.widget_map.get(&new_widget_id) {
|
|
self.current_widget = proc_sort_widget.clone(); // TODO: Could I remove this clone w/ static references?
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else if self.app_config_fields.left_legend {
|
|
if let BottomWidgetType::Cpu = self.current_widget.widget_type {
|
|
if let Some(current_widget) = self.widget_map.get(&self.current_widget.widget_id) {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get(&self.current_widget.widget_id)
|
|
{
|
|
if !cpu_widget_state.is_legend_hidden {
|
|
if let Some(new_widget_id) = current_widget.left_neighbour {
|
|
if let Some(new_widget) = self.widget_map.get(&new_widget_id) {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else if let BottomWidgetType::CpuLegend = self.current_widget.widget_type {
|
|
if let Some(current_widget) = self.widget_map.get(&self.current_widget.widget_id) {
|
|
if let Some(new_widget_id) = current_widget.left_neighbour {
|
|
if let Some(new_widget) = self.widget_map.get(&new_widget_id) {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn handle_right_expanded_movement(&mut self) {
|
|
if let BottomWidgetType::ProcSort = self.current_widget.widget_type {
|
|
if let Some(new_widget_id) = self.current_widget.right_neighbour {
|
|
if let Some(proc_sort_widget) = self.widget_map.get(&new_widget_id) {
|
|
self.current_widget = proc_sort_widget.clone();
|
|
}
|
|
}
|
|
} else if self.app_config_fields.left_legend {
|
|
if let BottomWidgetType::CpuLegend = self.current_widget.widget_type {
|
|
if let Some(current_widget) = self.widget_map.get(&self.current_widget.widget_id) {
|
|
if let Some(new_widget_id) = current_widget.right_neighbour {
|
|
if let Some(new_widget) = self.widget_map.get(&new_widget_id) {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else if let BottomWidgetType::Cpu = self.current_widget.widget_type {
|
|
if let Some(current_widget) = self.widget_map.get(&self.current_widget.widget_id) {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get(&self.current_widget.widget_id)
|
|
{
|
|
if !cpu_widget_state.is_legend_hidden {
|
|
if let Some(new_widget_id) = current_widget.right_neighbour {
|
|
if let Some(new_widget) = self.widget_map.get(&new_widget_id) {
|
|
self.current_widget = new_widget.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn skip_to_first(&mut self) {
|
|
if !self.ignore_normal_keybinds() {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Proc => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.table_state.current_scroll_position = 0;
|
|
proc_widget_state.table_state.scroll_direction = ScrollDirection::Up;
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSort => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 2)
|
|
{
|
|
proc_widget_state.sort_table_state.current_scroll_position = 0;
|
|
proc_widget_state.sort_table_state.scroll_direction = ScrollDirection::Up;
|
|
}
|
|
}
|
|
BottomWidgetType::Temp => {
|
|
if let Some(temp_widget_state) = self
|
|
.temp_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
temp_widget_state.table_state.current_scroll_position = 0;
|
|
temp_widget_state.table_state.scroll_direction = ScrollDirection::Up;
|
|
}
|
|
}
|
|
BottomWidgetType::Disk => {
|
|
if let Some(disk_widget_state) = self
|
|
.disk_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
disk_widget_state.table_state.current_scroll_position = 0;
|
|
disk_widget_state.table_state.scroll_direction = ScrollDirection::Up;
|
|
}
|
|
}
|
|
BottomWidgetType::CpuLegend => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 1)
|
|
{
|
|
cpu_widget_state.table_state.current_scroll_position = 0;
|
|
cpu_widget_state.table_state.scroll_direction = ScrollDirection::Up;
|
|
}
|
|
}
|
|
|
|
_ => {}
|
|
}
|
|
self.reset_multi_tap_keys();
|
|
} else if self.help_dialog_state.is_showing_help {
|
|
self.help_dialog_state.scroll_state.current_scroll_index = 0;
|
|
} else if self.delete_dialog_state.is_showing_dd {
|
|
self.delete_dialog_state.selected_signal = KillSignal::Cancel;
|
|
}
|
|
}
|
|
|
|
pub fn skip_to_last(&mut self) {
|
|
if !self.ignore_normal_keybinds() {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Proc => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state.table_state.current_scroll_position =
|
|
proc_widget_state.table_data.data.len().saturating_sub(1);
|
|
proc_widget_state.table_state.scroll_direction = ScrollDirection::Down;
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSort => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 2)
|
|
{
|
|
proc_widget_state.sort_table_state.current_scroll_position =
|
|
proc_widget_state.num_enabled_columns() - 1;
|
|
proc_widget_state.sort_table_state.scroll_direction = ScrollDirection::Down;
|
|
}
|
|
}
|
|
BottomWidgetType::Temp => {
|
|
if let Some(temp_widget_state) = self
|
|
.temp_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if !self.converted_data.temp_sensor_data.data.is_empty() {
|
|
temp_widget_state.table_state.current_scroll_position =
|
|
self.converted_data.temp_sensor_data.data.len() - 1;
|
|
temp_widget_state.table_state.scroll_direction = ScrollDirection::Down;
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Disk => {
|
|
if let Some(disk_widget_state) = self
|
|
.disk_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if !self.converted_data.disk_data.data.is_empty() {
|
|
disk_widget_state.table_state.current_scroll_position =
|
|
self.converted_data.disk_data.data.len() - 1;
|
|
disk_widget_state.table_state.scroll_direction = ScrollDirection::Down;
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::CpuLegend => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 1)
|
|
{
|
|
let cap = self.converted_data.cpu_data.len();
|
|
if cap > 0 {
|
|
cpu_widget_state.table_state.current_scroll_position = cap - 1;
|
|
cpu_widget_state.table_state.scroll_direction = ScrollDirection::Down;
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
self.reset_multi_tap_keys();
|
|
} else if self.help_dialog_state.is_showing_help {
|
|
self.help_dialog_state.scroll_state.current_scroll_index = self
|
|
.help_dialog_state
|
|
.scroll_state
|
|
.max_scroll_index
|
|
.saturating_sub(1);
|
|
} else if self.delete_dialog_state.is_showing_dd {
|
|
self.delete_dialog_state.selected_signal = KillSignal::Kill(MAX_SIGNAL);
|
|
}
|
|
}
|
|
|
|
pub fn decrement_position_count(&mut self) {
|
|
self.change_position_count(-1);
|
|
}
|
|
|
|
pub fn increment_position_count(&mut self) {
|
|
self.change_position_count(1);
|
|
}
|
|
|
|
fn change_position_count(&mut self, amount: i64) {
|
|
if !self.ignore_normal_keybinds() {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Proc => {
|
|
self.change_process_position(amount);
|
|
}
|
|
BottomWidgetType::ProcSort => self.change_process_sort_position(amount),
|
|
BottomWidgetType::Temp => self.change_temp_position(amount),
|
|
BottomWidgetType::Disk => self.change_disk_position(amount),
|
|
BottomWidgetType::CpuLegend => self.change_cpu_legend_position(amount),
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn change_process_sort_position(&mut self, num_to_change_by: i64) {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id - 2)
|
|
{
|
|
let num_entries = proc_widget_state.num_enabled_columns();
|
|
proc_widget_state
|
|
.sort_table_state
|
|
.update_position(num_to_change_by, num_entries);
|
|
}
|
|
}
|
|
|
|
fn change_cpu_legend_position(&mut self, num_to_change_by: i64) {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get_mut(&(self.current_widget.widget_id - 1))
|
|
{
|
|
cpu_widget_state
|
|
.table_state
|
|
.update_position(num_to_change_by, self.converted_data.cpu_data.len());
|
|
}
|
|
}
|
|
|
|
/// Returns the new position.
|
|
fn change_process_position(&mut self, num_to_change_by: i64) -> Option<usize> {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
proc_widget_state
|
|
.table_state
|
|
.update_position(num_to_change_by, proc_widget_state.table_data.data.len())
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn change_temp_position(&mut self, num_to_change_by: i64) {
|
|
if let Some(temp_widget_state) = self
|
|
.temp_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
temp_widget_state.table_state.update_position(
|
|
num_to_change_by,
|
|
self.converted_data.temp_sensor_data.data.len(),
|
|
);
|
|
}
|
|
}
|
|
|
|
fn change_disk_position(&mut self, num_to_change_by: i64) {
|
|
if let Some(disk_widget_state) = self
|
|
.disk_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
disk_widget_state
|
|
.table_state
|
|
.update_position(num_to_change_by, self.converted_data.disk_data.data.len());
|
|
}
|
|
}
|
|
|
|
fn help_scroll_up(&mut self) {
|
|
if self.help_dialog_state.scroll_state.current_scroll_index > 0 {
|
|
self.help_dialog_state.scroll_state.current_scroll_index -= 1;
|
|
}
|
|
}
|
|
|
|
fn help_scroll_down(&mut self) {
|
|
if self.help_dialog_state.scroll_state.current_scroll_index + 1
|
|
< self.help_dialog_state.scroll_state.max_scroll_index
|
|
{
|
|
self.help_dialog_state.scroll_state.current_scroll_index += 1;
|
|
}
|
|
}
|
|
|
|
fn help_scroll_to_or_max(&mut self, new_position: u16) {
|
|
if new_position < self.help_dialog_state.scroll_state.max_scroll_index {
|
|
self.help_dialog_state.scroll_state.current_scroll_index = new_position;
|
|
} else {
|
|
self.help_dialog_state.scroll_state.current_scroll_index =
|
|
self.help_dialog_state.scroll_state.max_scroll_index - 1;
|
|
}
|
|
}
|
|
|
|
pub fn handle_scroll_up(&mut self) {
|
|
if self.delete_dialog_state.is_showing_dd {
|
|
#[cfg(target_family = "unix")]
|
|
{
|
|
self.on_up_key();
|
|
return;
|
|
}
|
|
}
|
|
if self.help_dialog_state.is_showing_help {
|
|
self.help_scroll_up();
|
|
} else if self.current_widget.widget_type.is_widget_graph() {
|
|
self.zoom_in();
|
|
} else if self.current_widget.widget_type.is_widget_table() {
|
|
self.decrement_position_count();
|
|
}
|
|
}
|
|
|
|
pub fn handle_scroll_down(&mut self) {
|
|
if self.delete_dialog_state.is_showing_dd {
|
|
#[cfg(target_family = "unix")]
|
|
{
|
|
self.on_down_key();
|
|
return;
|
|
}
|
|
}
|
|
if self.help_dialog_state.is_showing_help {
|
|
self.help_scroll_down();
|
|
} else if self.current_widget.widget_type.is_widget_graph() {
|
|
self.zoom_out();
|
|
} else if self.current_widget.widget_type.is_widget_table() {
|
|
self.increment_position_count();
|
|
}
|
|
}
|
|
|
|
fn on_plus(&mut self) {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
// Toggle collapsing if tree
|
|
self.toggle_collapsing_process_branch();
|
|
} else {
|
|
self.zoom_in();
|
|
}
|
|
}
|
|
|
|
fn on_minus(&mut self) {
|
|
if let BottomWidgetType::Proc = self.current_widget.widget_type {
|
|
// Toggle collapsing if tree
|
|
self.toggle_collapsing_process_branch();
|
|
} else {
|
|
self.zoom_out();
|
|
}
|
|
}
|
|
|
|
fn toggle_collapsing_process_branch(&mut self) {
|
|
if let Some(pws) = self
|
|
.proc_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
pws.toggle_tree_branch();
|
|
}
|
|
}
|
|
|
|
fn zoom_out(&mut self) {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Cpu => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
let new_time = cpu_widget_state.current_display_time
|
|
+ self.app_config_fields.time_interval;
|
|
if new_time <= constants::STALE_MAX_MILLISECONDS {
|
|
cpu_widget_state.current_display_time = new_time;
|
|
self.cpu_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
cpu_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
} else if cpu_widget_state.current_display_time
|
|
!= constants::STALE_MAX_MILLISECONDS
|
|
{
|
|
cpu_widget_state.current_display_time = constants::STALE_MAX_MILLISECONDS;
|
|
self.cpu_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
cpu_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Mem => {
|
|
if let Some(mem_widget_state) = self
|
|
.mem_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
let new_time = mem_widget_state.current_display_time
|
|
+ self.app_config_fields.time_interval;
|
|
if new_time <= constants::STALE_MAX_MILLISECONDS {
|
|
mem_widget_state.current_display_time = new_time;
|
|
self.mem_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
mem_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
} else if mem_widget_state.current_display_time
|
|
!= constants::STALE_MAX_MILLISECONDS
|
|
{
|
|
mem_widget_state.current_display_time = constants::STALE_MAX_MILLISECONDS;
|
|
self.mem_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
mem_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Net => {
|
|
if let Some(net_widget_state) = self
|
|
.net_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
let new_time = net_widget_state.current_display_time
|
|
+ self.app_config_fields.time_interval;
|
|
if new_time <= constants::STALE_MAX_MILLISECONDS {
|
|
net_widget_state.current_display_time = new_time;
|
|
self.net_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
net_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
} else if net_widget_state.current_display_time
|
|
!= constants::STALE_MAX_MILLISECONDS
|
|
{
|
|
net_widget_state.current_display_time = constants::STALE_MAX_MILLISECONDS;
|
|
self.net_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
net_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
fn zoom_in(&mut self) {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Cpu => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
let new_time = cpu_widget_state.current_display_time
|
|
- self.app_config_fields.time_interval;
|
|
if new_time >= constants::STALE_MIN_MILLISECONDS {
|
|
cpu_widget_state.current_display_time = new_time;
|
|
self.cpu_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
cpu_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
} else if cpu_widget_state.current_display_time
|
|
!= constants::STALE_MIN_MILLISECONDS
|
|
{
|
|
cpu_widget_state.current_display_time = constants::STALE_MIN_MILLISECONDS;
|
|
self.cpu_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
cpu_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Mem => {
|
|
if let Some(mem_widget_state) = self
|
|
.mem_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
let new_time = mem_widget_state.current_display_time
|
|
- self.app_config_fields.time_interval;
|
|
if new_time >= constants::STALE_MIN_MILLISECONDS {
|
|
mem_widget_state.current_display_time = new_time;
|
|
self.mem_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
mem_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
} else if mem_widget_state.current_display_time
|
|
!= constants::STALE_MIN_MILLISECONDS
|
|
{
|
|
mem_widget_state.current_display_time = constants::STALE_MIN_MILLISECONDS;
|
|
self.mem_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
mem_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Net => {
|
|
if let Some(net_widget_state) = self
|
|
.net_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
let new_time = net_widget_state.current_display_time
|
|
- self.app_config_fields.time_interval;
|
|
if new_time >= constants::STALE_MIN_MILLISECONDS {
|
|
net_widget_state.current_display_time = new_time;
|
|
self.net_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
net_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
} else if net_widget_state.current_display_time
|
|
!= constants::STALE_MIN_MILLISECONDS
|
|
{
|
|
net_widget_state.current_display_time = constants::STALE_MIN_MILLISECONDS;
|
|
self.net_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
net_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
fn reset_cpu_zoom(&mut self) {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
cpu_widget_state.current_display_time = self.app_config_fields.default_time_value;
|
|
self.cpu_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
cpu_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
|
|
fn reset_mem_zoom(&mut self) {
|
|
if let Some(mem_widget_state) = self
|
|
.mem_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
mem_widget_state.current_display_time = self.app_config_fields.default_time_value;
|
|
self.mem_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
mem_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
|
|
fn reset_net_zoom(&mut self) {
|
|
if let Some(net_widget_state) = self
|
|
.net_state
|
|
.widget_states
|
|
.get_mut(&self.current_widget.widget_id)
|
|
{
|
|
net_widget_state.current_display_time = self.app_config_fields.default_time_value;
|
|
self.net_state.force_update = Some(self.current_widget.widget_id);
|
|
if self.app_config_fields.autohide_time {
|
|
net_widget_state.autohide_timer = Some(Instant::now());
|
|
}
|
|
}
|
|
}
|
|
|
|
fn reset_zoom(&mut self) {
|
|
match self.current_widget.widget_type {
|
|
BottomWidgetType::Cpu => self.reset_cpu_zoom(),
|
|
BottomWidgetType::Mem => self.reset_mem_zoom(),
|
|
BottomWidgetType::Net => self.reset_net_zoom(),
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
/// Moves the mouse to the widget that was clicked on, then propagates the click down to be
|
|
/// handled by the widget specifically.
|
|
pub fn on_left_mouse_up(&mut self, x: u16, y: u16) {
|
|
// Pretty dead simple - iterate through the widget map and go to the widget where the click
|
|
// is within.
|
|
|
|
// TODO: [REFACTOR] might want to refactor this, it's really ugly.
|
|
// TODO: [REFACTOR] Might wanna refactor ALL state things in general, currently everything
|
|
// is grouped up as an app state. We should separate stuff like event state and gui state and etc.
|
|
|
|
// TODO: [MOUSE] double click functionality...? We would do this above all other actions and SC if needed.
|
|
|
|
// Short circuit if we're in basic table... we might have to handle the basic table arrow
|
|
// case here...
|
|
|
|
if let Some(bt) = &mut self.basic_table_widget_state {
|
|
if let (
|
|
Some((left_tlc_x, left_tlc_y)),
|
|
Some((left_brc_x, left_brc_y)),
|
|
Some((right_tlc_x, right_tlc_y)),
|
|
Some((right_brc_x, right_brc_y)),
|
|
) = (bt.left_tlc, bt.left_brc, bt.right_tlc, bt.right_brc)
|
|
{
|
|
if (x >= left_tlc_x && y >= left_tlc_y) && (x < left_brc_x && y < left_brc_y) {
|
|
// Case for the left "button" in the simple arrow.
|
|
if let Some(new_widget) =
|
|
self.widget_map.get(&(bt.currently_displayed_widget_id))
|
|
{
|
|
// We have to move to the current table widget first...
|
|
self.current_widget = new_widget.clone();
|
|
|
|
if let BottomWidgetType::Proc = &new_widget.widget_type {
|
|
if let Some(proc_widget_state) =
|
|
self.proc_state.get_widget_state(new_widget.widget_id)
|
|
{
|
|
if proc_widget_state.is_sort_open {
|
|
self.move_widget_selection(&WidgetDirection::Left);
|
|
}
|
|
}
|
|
}
|
|
self.move_widget_selection(&WidgetDirection::Left);
|
|
return;
|
|
}
|
|
} else if (x >= right_tlc_x && y >= right_tlc_y)
|
|
&& (x < right_brc_x && y < right_brc_y)
|
|
{
|
|
// Case for the right "button" in the simple arrow.
|
|
if let Some(new_widget) =
|
|
self.widget_map.get(&(bt.currently_displayed_widget_id))
|
|
{
|
|
// We have to move to the current table widget first...
|
|
self.current_widget = new_widget.clone();
|
|
|
|
if let BottomWidgetType::ProcSort = &new_widget.widget_type {
|
|
if let Some(proc_widget_state) =
|
|
self.proc_state.get_widget_state(new_widget.widget_id - 2)
|
|
{
|
|
if proc_widget_state.is_sort_open {
|
|
self.move_widget_selection(&WidgetDirection::Right);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
self.move_widget_selection(&WidgetDirection::Right);
|
|
// Bit extra logic to ensure you always land on a proc widget, not the sort
|
|
if let BottomWidgetType::ProcSort = &self.current_widget.widget_type {
|
|
self.move_widget_selection(&WidgetDirection::Right);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Second short circuit --- are we in the dd dialog state? If so, only check yes/no/signals
|
|
// and bail after.
|
|
if self.is_in_dialog() {
|
|
match self.delete_dialog_state.button_positions.iter().find(
|
|
|(tl_x, tl_y, br_x, br_y, _idx)| {
|
|
(x >= *tl_x && y >= *tl_y) && (x <= *br_x && y <= *br_y)
|
|
},
|
|
) {
|
|
Some((_, _, _, _, 0)) => {
|
|
self.delete_dialog_state.selected_signal = KillSignal::Cancel
|
|
}
|
|
Some((_, _, _, _, idx)) => {
|
|
if *idx > 31 {
|
|
self.delete_dialog_state.selected_signal = KillSignal::Kill(*idx + 2)
|
|
} else {
|
|
self.delete_dialog_state.selected_signal = KillSignal::Kill(*idx)
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
return;
|
|
}
|
|
|
|
let mut failed_to_get = true;
|
|
for (new_widget_id, widget) in &self.widget_map {
|
|
if let (Some((tlc_x, tlc_y)), Some((brc_x, brc_y))) =
|
|
(widget.top_left_corner, widget.bottom_right_corner)
|
|
{
|
|
if (x >= tlc_x && y >= tlc_y) && (x < brc_x && y < brc_y) {
|
|
if let Some(new_widget) = self.widget_map.get(new_widget_id) {
|
|
self.current_widget = new_widget.clone();
|
|
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::Temp
|
|
| BottomWidgetType::Proc
|
|
| BottomWidgetType::ProcSort
|
|
| BottomWidgetType::Disk
|
|
| BottomWidgetType::Battery => {
|
|
if let Some(basic_table_widget_state) =
|
|
&mut self.basic_table_widget_state
|
|
{
|
|
basic_table_widget_state.currently_displayed_widget_id =
|
|
self.current_widget.widget_id;
|
|
basic_table_widget_state.currently_displayed_widget_type =
|
|
self.current_widget.widget_type.clone();
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
|
|
failed_to_get = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if failed_to_get {
|
|
return;
|
|
}
|
|
|
|
// Now handle click propagation down to widget.
|
|
if let (Some((_tlc_x, tlc_y)), Some((_brc_x, brc_y))) = (
|
|
&self.current_widget.top_left_corner,
|
|
&self.current_widget.bottom_right_corner,
|
|
) {
|
|
let border_offset = if self.is_drawing_border() { 1 } else { 0 };
|
|
|
|
// This check ensures the click isn't actually just clicking on the bottom border.
|
|
if y < (brc_y - border_offset) {
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::Proc
|
|
| BottomWidgetType::ProcSort
|
|
| BottomWidgetType::CpuLegend
|
|
| BottomWidgetType::Temp
|
|
| BottomWidgetType::Disk => {
|
|
// Get our index...
|
|
let clicked_entry = y - *tlc_y;
|
|
let header_offset = self.header_offset(&self.current_widget);
|
|
let offset = border_offset + header_offset;
|
|
if clicked_entry >= offset {
|
|
let offset_clicked_entry = clicked_entry - offset;
|
|
match &self.current_widget.widget_type {
|
|
BottomWidgetType::Proc => {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if let Some(visual_index) =
|
|
proc_widget_state.table_state.table_state.selected()
|
|
{
|
|
// If in tree mode, also check to see if this click is on
|
|
// the same entry as the already selected one - if it is,
|
|
// then we minimize.
|
|
|
|
let is_tree_mode = matches!(
|
|
proc_widget_state.mode,
|
|
ProcWidgetMode::Tree { .. }
|
|
);
|
|
|
|
let previous_scroll_position = proc_widget_state
|
|
.table_state
|
|
.current_scroll_position;
|
|
|
|
let new_position = self.change_process_position(
|
|
offset_clicked_entry as i64 - visual_index as i64,
|
|
);
|
|
|
|
if is_tree_mode {
|
|
if let Some(new_position) = new_position {
|
|
if previous_scroll_position == new_position {
|
|
self.toggle_collapsing_process_branch();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::ProcSort => {
|
|
// TODO: [Feature] This could sort if you double click!
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_widget_state(self.current_widget.widget_id - 2)
|
|
{
|
|
if let Some(visual_index) = proc_widget_state
|
|
.sort_table_state
|
|
.table_state
|
|
.selected()
|
|
{
|
|
self.change_process_sort_position(
|
|
offset_clicked_entry as i64 - visual_index as i64,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::CpuLegend => {
|
|
if let Some(cpu_widget_state) = self
|
|
.cpu_state
|
|
.get_widget_state(self.current_widget.widget_id - 1)
|
|
{
|
|
if let Some(visual_index) =
|
|
cpu_widget_state.table_state.table_state.selected()
|
|
{
|
|
self.change_cpu_legend_position(
|
|
offset_clicked_entry as i64 - visual_index as i64,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Temp => {
|
|
if let Some(temp_widget_state) = self
|
|
.temp_state
|
|
.get_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if let Some(visual_index) =
|
|
temp_widget_state.table_state.table_state.selected()
|
|
{
|
|
self.change_temp_position(
|
|
offset_clicked_entry as i64 - visual_index as i64,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Disk => {
|
|
if let Some(disk_widget_state) = self
|
|
.disk_state
|
|
.get_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if let Some(visual_index) =
|
|
disk_widget_state.table_state.table_state.selected()
|
|
{
|
|
self.change_disk_position(
|
|
offset_clicked_entry as i64 - visual_index as i64,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
} else {
|
|
// We might have clicked on a header! Check if we only exceeded the table + border offset, and
|
|
// it's implied we exceeded the gap offset.
|
|
if clicked_entry == border_offset {
|
|
if let BottomWidgetType::Proc = &self.current_widget.widget_type {
|
|
if let Some(proc_widget_state) = self
|
|
.proc_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if let SortState::Sortable(st) =
|
|
&mut proc_widget_state.table_state.sort_state
|
|
{
|
|
if st.try_select_location(x, y).is_some() {
|
|
proc_widget_state.force_update = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BottomWidgetType::Battery => {
|
|
if let Some(battery_widget_state) = self
|
|
.battery_state
|
|
.get_mut_widget_state(self.current_widget.widget_id)
|
|
{
|
|
if let Some(tab_spacing) = &battery_widget_state.tab_click_locs {
|
|
for (itx, ((tlc_x, tlc_y), (brc_x, brc_y))) in
|
|
tab_spacing.iter().enumerate()
|
|
{
|
|
if (x >= *tlc_x && y >= *tlc_y) && (x <= *brc_x && y <= *brc_y)
|
|
{
|
|
battery_widget_state.currently_selected_battery_index = itx;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn is_drawing_border(&self) -> bool {
|
|
self.is_expanded || !self.app_config_fields.use_basic_mode
|
|
}
|
|
|
|
fn header_offset(&self, widget: &BottomWidget) -> u16 {
|
|
if let (Some((_tlc_x, tlc_y)), Some((_brc_x, brc_y))) =
|
|
(widget.top_left_corner, widget.bottom_right_corner)
|
|
{
|
|
let height_diff = brc_y - tlc_y;
|
|
if height_diff >= constants::TABLE_GAP_HEIGHT_LIMIT {
|
|
1 + self.app_config_fields.table_gap
|
|
} else {
|
|
let min_height_for_header = if self.is_drawing_border() { 3 } else { 1 };
|
|
if height_diff > min_height_for_header {
|
|
1
|
|
} else {
|
|
0
|
|
}
|
|
}
|
|
} else {
|
|
1 + self.app_config_fields.table_gap
|
|
}
|
|
}
|
|
}
|