2019-09-12 00:41:11 +00:00
|
|
|
pub mod data_collection;
|
2019-09-14 21:07:18 +00:00
|
|
|
use data_collection::{processes, temperature};
|
2019-12-16 07:21:44 +00:00
|
|
|
use std::time::Instant;
|
|
|
|
|
|
|
|
use crate::constants;
|
2019-09-08 23:56:23 +00:00
|
|
|
|
2019-09-16 23:05:44 +00:00
|
|
|
mod process_killer;
|
|
|
|
|
2019-09-17 02:39:57 +00:00
|
|
|
#[derive(Clone, Copy)]
|
2019-09-15 18:16:18 +00:00
|
|
|
pub enum ApplicationPosition {
|
2019-12-28 06:20:05 +00:00
|
|
|
Cpu,
|
|
|
|
Mem,
|
|
|
|
Disk,
|
|
|
|
Temp,
|
|
|
|
Network,
|
|
|
|
Process,
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
|
2019-09-16 20:18:42 +00:00
|
|
|
pub enum ScrollDirection {
|
|
|
|
/// UP means scrolling up --- this usually DECREMENTS
|
|
|
|
UP,
|
|
|
|
/// DOWN means scrolling down --- this usually INCREMENTS
|
|
|
|
DOWN,
|
|
|
|
}
|
|
|
|
|
2019-09-14 20:46:14 +00:00
|
|
|
pub struct App {
|
2019-12-28 06:20:05 +00:00
|
|
|
// Sorting
|
2019-12-06 05:57:04 +00:00
|
|
|
pub process_sorting_type: processes::ProcessSorting,
|
|
|
|
pub process_sorting_reverse: bool,
|
|
|
|
pub to_be_resorted: bool,
|
2019-12-28 06:20:05 +00:00
|
|
|
// Positioning
|
2020-01-01 22:55:15 +00:00
|
|
|
pub scroll_direction: ScrollDirection,
|
2019-12-06 05:57:04 +00:00
|
|
|
pub currently_selected_process_position: i64,
|
|
|
|
pub currently_selected_disk_position: i64,
|
|
|
|
pub currently_selected_temperature_position: i64,
|
2020-01-01 22:55:15 +00:00
|
|
|
pub currently_selected_cpu_table_position: i64,
|
2019-12-28 06:20:05 +00:00
|
|
|
pub previous_disk_position: i64,
|
|
|
|
pub previous_temp_position: i64,
|
|
|
|
pub previous_process_position: i64,
|
2020-01-01 22:55:15 +00:00
|
|
|
pub previous_cpu_table_position: i64,
|
2019-12-06 05:57:04 +00:00
|
|
|
pub temperature_type: temperature::TemperatureType,
|
|
|
|
pub update_rate_in_milliseconds: u64,
|
|
|
|
pub show_average_cpu: bool,
|
|
|
|
pub current_application_position: ApplicationPosition,
|
|
|
|
pub data: data_collection::Data,
|
2019-12-26 04:30:57 +00:00
|
|
|
awaiting_second_char: bool,
|
|
|
|
second_char: char,
|
2019-12-06 05:57:04 +00:00
|
|
|
pub use_dot: bool,
|
|
|
|
pub show_help: bool,
|
|
|
|
pub is_frozen: bool,
|
2019-12-28 03:39:25 +00:00
|
|
|
pub left_legend: bool,
|
2020-01-01 03:24:54 +00:00
|
|
|
pub use_current_cpu_total: bool,
|
2019-12-16 07:21:44 +00:00
|
|
|
last_key_press: Instant,
|
2019-09-08 23:56:23 +00:00
|
|
|
}
|
|
|
|
|
2019-09-14 20:46:14 +00:00
|
|
|
impl App {
|
2019-12-28 03:39:25 +00:00
|
|
|
pub fn new(
|
|
|
|
show_average_cpu: bool, temperature_type: temperature::TemperatureType, update_rate_in_milliseconds: u64, use_dot: bool, left_legend: bool,
|
2020-01-01 03:24:54 +00:00
|
|
|
use_current_cpu_total: bool,
|
2019-12-28 03:39:25 +00:00
|
|
|
) -> App {
|
2019-09-09 04:09:58 +00:00
|
|
|
App {
|
2019-12-06 05:57:04 +00:00
|
|
|
process_sorting_type: processes::ProcessSorting::CPU,
|
|
|
|
process_sorting_reverse: true,
|
|
|
|
to_be_resorted: false,
|
2019-09-14 20:46:14 +00:00
|
|
|
temperature_type,
|
|
|
|
update_rate_in_milliseconds,
|
2019-09-14 21:07:18 +00:00
|
|
|
show_average_cpu,
|
2019-12-28 06:20:05 +00:00
|
|
|
current_application_position: ApplicationPosition::Process,
|
2019-12-06 05:57:04 +00:00
|
|
|
scroll_direction: ScrollDirection::DOWN,
|
2020-01-01 22:55:15 +00:00
|
|
|
currently_selected_process_position: 0,
|
|
|
|
currently_selected_disk_position: 0,
|
|
|
|
currently_selected_temperature_position: 0,
|
|
|
|
currently_selected_cpu_table_position: 0,
|
2019-12-06 05:57:04 +00:00
|
|
|
previous_process_position: 0,
|
|
|
|
previous_disk_position: 0,
|
|
|
|
previous_temp_position: 0,
|
2020-01-01 22:55:15 +00:00
|
|
|
previous_cpu_table_position: 0,
|
2019-12-28 06:20:05 +00:00
|
|
|
data: data_collection::Data::default(),
|
2019-12-26 04:30:57 +00:00
|
|
|
awaiting_second_char: false,
|
|
|
|
second_char: ' ',
|
2019-09-25 04:48:53 +00:00
|
|
|
use_dot,
|
2019-12-06 05:57:04 +00:00
|
|
|
show_help: false,
|
|
|
|
is_frozen: false,
|
2019-12-28 03:39:25 +00:00
|
|
|
left_legend,
|
2020-01-01 03:24:54 +00:00
|
|
|
use_current_cpu_total,
|
2019-12-16 07:21:44 +00:00
|
|
|
last_key_press: Instant::now(),
|
2019-09-09 04:09:58 +00:00
|
|
|
}
|
2019-09-08 23:56:23 +00:00
|
|
|
}
|
|
|
|
|
2019-12-16 07:21:44 +00:00
|
|
|
pub fn reset(&mut self) {
|
|
|
|
self.reset_multi_tap_keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn reset_multi_tap_keys(&mut self) {
|
2019-12-26 04:30:57 +00:00
|
|
|
self.awaiting_second_char = false;
|
|
|
|
self.second_char = ' ';
|
2019-12-16 07:21:44 +00:00
|
|
|
}
|
|
|
|
|
2019-12-06 05:57:04 +00:00
|
|
|
pub fn on_enter(&mut self) {}
|
2019-10-10 02:00:10 +00:00
|
|
|
|
|
|
|
pub fn on_esc(&mut self) {
|
|
|
|
if self.show_help {
|
|
|
|
self.show_help = false;
|
|
|
|
}
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
2019-10-10 02:00:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: How should we make it for process panel specific hotkeys? Only if we're on process panel? Or what?
|
2019-12-26 04:30:57 +00:00
|
|
|
pub fn on_key(&mut self, caught_char: char) {
|
2019-10-10 02:00:10 +00:00
|
|
|
if !self.show_help {
|
2019-12-16 07:21:44 +00:00
|
|
|
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 {
|
|
|
|
self.reset_multi_tap_keys();
|
|
|
|
}
|
|
|
|
self.last_key_press = current_key_press_inst;
|
|
|
|
|
2019-12-26 04:30:57 +00:00
|
|
|
match caught_char {
|
2019-10-10 02:00:10 +00:00
|
|
|
'd' => {
|
2019-12-26 04:30:57 +00:00
|
|
|
if self.awaiting_second_char && self.second_char == 'd' {
|
|
|
|
self.awaiting_second_char = false;
|
|
|
|
self.second_char = ' ';
|
2020-01-01 21:32:20 +00:00
|
|
|
self.kill_highlighted_process().unwrap_or(()); // TODO: Return error to user? We have a dialog box...
|
2019-12-26 04:30:57 +00:00
|
|
|
} else {
|
|
|
|
self.awaiting_second_char = true;
|
|
|
|
self.second_char = 'd';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
'g' => {
|
|
|
|
if self.awaiting_second_char && self.second_char == 'g' {
|
|
|
|
self.awaiting_second_char = false;
|
|
|
|
self.second_char = ' ';
|
|
|
|
self.skip_to_first();
|
2019-12-06 05:57:04 +00:00
|
|
|
} else {
|
2019-12-26 04:30:57 +00:00
|
|
|
self.awaiting_second_char = true;
|
|
|
|
self.second_char = 'g';
|
2019-10-10 02:00:10 +00:00
|
|
|
}
|
2019-09-17 01:45:48 +00:00
|
|
|
}
|
2019-10-10 02:34:09 +00:00
|
|
|
'f' => {
|
|
|
|
self.is_frozen = !self.is_frozen;
|
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
'c' => {
|
|
|
|
match self.process_sorting_type {
|
|
|
|
processes::ProcessSorting::CPU => self.process_sorting_reverse = !self.process_sorting_reverse,
|
|
|
|
_ => {
|
|
|
|
self.process_sorting_type = processes::ProcessSorting::CPU;
|
|
|
|
self.process_sorting_reverse = true;
|
|
|
|
}
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
self.to_be_resorted = true;
|
|
|
|
self.currently_selected_process_position = 0;
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
'm' => {
|
|
|
|
match self.process_sorting_type {
|
|
|
|
processes::ProcessSorting::MEM => self.process_sorting_reverse = !self.process_sorting_reverse,
|
|
|
|
_ => {
|
|
|
|
self.process_sorting_type = processes::ProcessSorting::MEM;
|
|
|
|
self.process_sorting_reverse = true;
|
|
|
|
}
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
self.to_be_resorted = true;
|
|
|
|
self.currently_selected_process_position = 0;
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
'p' => {
|
|
|
|
match self.process_sorting_type {
|
|
|
|
processes::ProcessSorting::PID => self.process_sorting_reverse = !self.process_sorting_reverse,
|
|
|
|
_ => {
|
|
|
|
self.process_sorting_type = processes::ProcessSorting::PID;
|
|
|
|
self.process_sorting_reverse = false;
|
|
|
|
}
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
self.to_be_resorted = true;
|
|
|
|
self.currently_selected_process_position = 0;
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
'n' => {
|
|
|
|
match self.process_sorting_type {
|
|
|
|
processes::ProcessSorting::NAME => self.process_sorting_reverse = !self.process_sorting_reverse,
|
|
|
|
_ => {
|
|
|
|
self.process_sorting_type = processes::ProcessSorting::NAME;
|
|
|
|
self.process_sorting_reverse = false;
|
|
|
|
}
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
self.to_be_resorted = true;
|
|
|
|
self.currently_selected_process_position = 0;
|
2019-09-10 22:22:34 +00:00
|
|
|
}
|
2019-10-10 02:00:10 +00:00
|
|
|
'?' => {
|
|
|
|
self.show_help = true;
|
|
|
|
}
|
|
|
|
_ => {}
|
2019-09-09 04:09:58 +00:00
|
|
|
}
|
2019-09-17 01:45:48 +00:00
|
|
|
|
2019-12-26 04:30:57 +00:00
|
|
|
if self.awaiting_second_char && caught_char != self.second_char {
|
|
|
|
self.awaiting_second_char = false;
|
2019-10-10 02:00:10 +00:00
|
|
|
}
|
2019-09-17 01:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-30 02:40:22 +00:00
|
|
|
fn kill_highlighted_process(&self) -> crate::utils::error::Result<()> {
|
2019-09-17 01:45:48 +00:00
|
|
|
let current_pid = u64::from(self.data.list_of_processes[self.currently_selected_process_position as usize].pid);
|
2019-12-30 02:40:22 +00:00
|
|
|
process_killer::kill_process_given_pid(current_pid)?;
|
2019-09-17 01:45:48 +00:00
|
|
|
Ok(())
|
2019-09-08 23:56:23 +00:00
|
|
|
}
|
|
|
|
|
2019-09-17 02:39:57 +00:00
|
|
|
// For now, these are hard coded --- in the future, they shouldn't be!
|
|
|
|
//
|
|
|
|
// General idea for now:
|
|
|
|
// CPU -(down)> MEM
|
|
|
|
// MEM -(down)> Network, -(right)> TEMP
|
|
|
|
// TEMP -(down)> Disk, -(left)> MEM, -(up)> CPU
|
|
|
|
// Disk -(down)> Processes, -(left)> MEM, -(up)> TEMP
|
|
|
|
// Network -(up)> MEM, -(right)> PROC
|
|
|
|
// PROC -(up)> Disk, -(left)> Network
|
2019-09-09 22:34:13 +00:00
|
|
|
pub fn on_left(&mut self) {
|
2019-09-17 02:39:57 +00:00
|
|
|
self.current_application_position = match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Process => ApplicationPosition::Network,
|
|
|
|
ApplicationPosition::Disk => ApplicationPosition::Mem,
|
|
|
|
ApplicationPosition::Temp => ApplicationPosition::Mem,
|
2019-09-17 02:39:57 +00:00
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
2019-09-09 22:34:13 +00:00
|
|
|
}
|
2019-09-08 23:56:23 +00:00
|
|
|
|
2019-09-09 22:34:13 +00:00
|
|
|
pub fn on_right(&mut self) {
|
2019-09-17 02:39:57 +00:00
|
|
|
self.current_application_position = match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Mem => ApplicationPosition::Temp,
|
|
|
|
ApplicationPosition::Network => ApplicationPosition::Process,
|
2019-09-17 02:39:57 +00:00
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
2019-09-09 22:34:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn on_up(&mut self) {
|
2019-09-17 02:39:57 +00:00
|
|
|
self.current_application_position = match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Mem => ApplicationPosition::Cpu,
|
|
|
|
ApplicationPosition::Network => ApplicationPosition::Mem,
|
|
|
|
ApplicationPosition::Process => ApplicationPosition::Disk,
|
|
|
|
ApplicationPosition::Temp => ApplicationPosition::Cpu,
|
|
|
|
ApplicationPosition::Disk => ApplicationPosition::Temp,
|
2019-09-17 02:39:57 +00:00
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
2019-09-09 22:34:13 +00:00
|
|
|
}
|
2019-09-08 23:56:23 +00:00
|
|
|
|
2019-09-09 22:34:13 +00:00
|
|
|
pub fn on_down(&mut self) {
|
2019-09-17 02:39:57 +00:00
|
|
|
self.current_application_position = match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Cpu => ApplicationPosition::Mem,
|
|
|
|
ApplicationPosition::Mem => ApplicationPosition::Network,
|
|
|
|
ApplicationPosition::Temp => ApplicationPosition::Disk,
|
|
|
|
ApplicationPosition::Disk => ApplicationPosition::Process,
|
2019-09-17 02:39:57 +00:00
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn skip_to_first(&mut self) {
|
|
|
|
match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Process => self.currently_selected_process_position = 0,
|
|
|
|
ApplicationPosition::Temp => self.currently_selected_temperature_position = 0,
|
|
|
|
ApplicationPosition::Disk => self.currently_selected_disk_position = 0,
|
2019-12-26 04:30:57 +00:00
|
|
|
_ => {}
|
|
|
|
}
|
2019-12-28 06:20:05 +00:00
|
|
|
self.scroll_direction = ScrollDirection::UP;
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn skip_to_last(&mut self) {
|
|
|
|
match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Process => self.currently_selected_process_position = self.data.list_of_processes.len() as i64 - 1,
|
|
|
|
ApplicationPosition::Temp => self.currently_selected_temperature_position = self.data.list_of_temperature_sensor.len() as i64 - 1,
|
|
|
|
ApplicationPosition::Disk => self.currently_selected_disk_position = self.data.list_of_disks.len() as i64 - 1,
|
2019-12-26 04:30:57 +00:00
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
self.scroll_direction = ScrollDirection::DOWN;
|
|
|
|
self.reset_multi_tap_keys();
|
2019-09-08 23:56:23 +00:00
|
|
|
}
|
2019-09-15 18:16:18 +00:00
|
|
|
|
|
|
|
pub fn decrement_position_count(&mut self) {
|
|
|
|
match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Process => self.change_process_position(-1),
|
|
|
|
ApplicationPosition::Temp => self.change_temp_position(-1),
|
|
|
|
ApplicationPosition::Disk => self.change_disk_position(-1),
|
2020-01-01 22:55:15 +00:00
|
|
|
ApplicationPosition::Cpu => self.change_cpu_table_position(-1), // TODO: Temporary
|
2019-09-15 18:16:18 +00:00
|
|
|
_ => {}
|
|
|
|
}
|
2019-09-16 20:18:42 +00:00
|
|
|
self.scroll_direction = ScrollDirection::UP;
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn increment_position_count(&mut self) {
|
|
|
|
match self.current_application_position {
|
2019-12-28 06:20:05 +00:00
|
|
|
ApplicationPosition::Process => self.change_process_position(1),
|
|
|
|
ApplicationPosition::Temp => self.change_temp_position(1),
|
|
|
|
ApplicationPosition::Disk => self.change_disk_position(1),
|
2020-01-01 22:55:15 +00:00
|
|
|
ApplicationPosition::Cpu => self.change_cpu_table_position(1), // TODO: Temporary
|
2019-09-15 18:16:18 +00:00
|
|
|
_ => {}
|
|
|
|
}
|
2019-09-16 20:18:42 +00:00
|
|
|
self.scroll_direction = ScrollDirection::DOWN;
|
2019-12-26 04:30:57 +00:00
|
|
|
self.reset_multi_tap_keys();
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
|
2020-01-01 22:55:15 +00:00
|
|
|
fn change_cpu_table_position(&mut self, num_to_change_by: i64) {
|
|
|
|
if let Some(cpu_package) = self.data.list_of_cpu_packages.last() {
|
|
|
|
if self.currently_selected_cpu_table_position + num_to_change_by >= 0
|
|
|
|
&& self.currently_selected_cpu_table_position + num_to_change_by
|
|
|
|
< if self.show_average_cpu {
|
|
|
|
cpu_package.cpu_vec.len()
|
|
|
|
} else {
|
|
|
|
cpu_package.cpu_vec.len() - 1
|
|
|
|
} as i64
|
|
|
|
{
|
|
|
|
self.currently_selected_cpu_table_position += num_to_change_by;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-06 05:57:04 +00:00
|
|
|
fn change_process_position(&mut self, num_to_change_by: i64) {
|
2019-09-16 20:18:42 +00:00
|
|
|
if self.currently_selected_process_position + num_to_change_by >= 0
|
|
|
|
&& self.currently_selected_process_position + num_to_change_by < self.data.list_of_processes.len() as i64
|
|
|
|
{
|
|
|
|
self.currently_selected_process_position += num_to_change_by;
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-06 05:57:04 +00:00
|
|
|
fn change_temp_position(&mut self, num_to_change_by: i64) {
|
2019-09-25 20:43:13 +00:00
|
|
|
if self.currently_selected_temperature_position + num_to_change_by >= 0
|
|
|
|
&& self.currently_selected_temperature_position + num_to_change_by < self.data.list_of_temperature_sensor.len() as i64
|
|
|
|
{
|
2019-09-16 20:18:42 +00:00
|
|
|
self.currently_selected_temperature_position += num_to_change_by;
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-06 05:57:04 +00:00
|
|
|
fn change_disk_position(&mut self, num_to_change_by: i64) {
|
2019-09-25 20:43:13 +00:00
|
|
|
if self.currently_selected_disk_position + num_to_change_by >= 0
|
|
|
|
&& self.currently_selected_disk_position + num_to_change_by < self.data.list_of_disks.len() as i64
|
|
|
|
{
|
2019-09-16 20:18:42 +00:00
|
|
|
self.currently_selected_disk_position += num_to_change_by;
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
}
|
2019-09-08 23:56:23 +00:00
|
|
|
}
|