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 {
|
|
|
|
CPU,
|
|
|
|
MEM,
|
|
|
|
DISK,
|
|
|
|
TEMP,
|
|
|
|
NETWORK,
|
|
|
|
PROCESS,
|
|
|
|
}
|
|
|
|
|
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-06 05:57:04 +00:00
|
|
|
pub process_sorting_type: processes::ProcessSorting,
|
|
|
|
pub process_sorting_reverse: bool,
|
|
|
|
pub to_be_resorted: bool,
|
|
|
|
pub currently_selected_process_position: i64,
|
|
|
|
pub currently_selected_disk_position: i64,
|
|
|
|
pub currently_selected_temperature_position: i64,
|
|
|
|
pub temperature_type: temperature::TemperatureType,
|
|
|
|
pub update_rate_in_milliseconds: u64,
|
|
|
|
pub show_average_cpu: bool,
|
|
|
|
pub current_application_position: ApplicationPosition,
|
|
|
|
pub current_zoom_level_percent: f64, // Make at most 200, least 50?
|
|
|
|
pub data: data_collection::Data,
|
|
|
|
pub scroll_direction: ScrollDirection,
|
|
|
|
pub previous_disk_position: i64,
|
|
|
|
pub previous_temp_position: i64,
|
|
|
|
pub previous_process_position: i64,
|
|
|
|
awaiting_second_d: bool,
|
|
|
|
pub use_dot: bool,
|
|
|
|
pub show_help: bool,
|
|
|
|
pub is_frozen: 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-06 05:57:04 +00:00
|
|
|
pub fn new(show_average_cpu: bool, temperature_type: temperature::TemperatureType, update_rate_in_milliseconds: u64, use_dot: bool) -> 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,
|
|
|
|
currently_selected_process_position: 0,
|
|
|
|
currently_selected_disk_position: 0,
|
|
|
|
currently_selected_temperature_position: 0,
|
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-06 05:57:04 +00:00
|
|
|
current_application_position: ApplicationPosition::PROCESS,
|
|
|
|
current_zoom_level_percent: 100.0,
|
|
|
|
data: data_collection::Data::default(),
|
|
|
|
scroll_direction: ScrollDirection::DOWN,
|
|
|
|
previous_process_position: 0,
|
|
|
|
previous_disk_position: 0,
|
|
|
|
previous_temp_position: 0,
|
|
|
|
awaiting_second_d: false,
|
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-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) {
|
|
|
|
self.awaiting_second_d = false;
|
|
|
|
}
|
|
|
|
|
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-16 07:21:44 +00:00
|
|
|
self.awaiting_second_d = false;
|
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-06 05:57:04 +00:00
|
|
|
pub fn on_key(&mut self, c: 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-10-10 02:00:10 +00:00
|
|
|
match c {
|
|
|
|
'd' => {
|
|
|
|
if self.awaiting_second_d {
|
|
|
|
self.awaiting_second_d = false;
|
2019-10-10 02:34:09 +00:00
|
|
|
self.kill_highlighted_process().unwrap_or(());
|
2019-12-06 05:57:04 +00:00
|
|
|
} else {
|
2019-10-10 02:00:10 +00:00
|
|
|
self.awaiting_second_d = true;
|
|
|
|
}
|
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-10-10 02:00:10 +00:00
|
|
|
if self.awaiting_second_d && c != 'd' {
|
|
|
|
self.awaiting_second_d = false;
|
|
|
|
}
|
2019-09-17 01:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn kill_highlighted_process(&self) -> crate::utils::error::Result<()> {
|
|
|
|
let current_pid = u64::from(self.data.list_of_processes[self.currently_selected_process_position as usize].pid);
|
|
|
|
process_killer::kill_process_given_pid(current_pid)?;
|
|
|
|
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 {
|
|
|
|
ApplicationPosition::PROCESS => ApplicationPosition::NETWORK,
|
|
|
|
ApplicationPosition::DISK => ApplicationPosition::MEM,
|
|
|
|
ApplicationPosition::TEMP => ApplicationPosition::MEM,
|
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-16 07:21:44 +00:00
|
|
|
self.awaiting_second_d = false;
|
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 {
|
|
|
|
ApplicationPosition::MEM => ApplicationPosition::TEMP,
|
|
|
|
ApplicationPosition::NETWORK => ApplicationPosition::PROCESS,
|
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-16 07:21:44 +00:00
|
|
|
self.awaiting_second_d = false;
|
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 {
|
|
|
|
ApplicationPosition::MEM => ApplicationPosition::CPU,
|
|
|
|
ApplicationPosition::NETWORK => ApplicationPosition::MEM,
|
|
|
|
ApplicationPosition::PROCESS => ApplicationPosition::DISK,
|
|
|
|
ApplicationPosition::TEMP => ApplicationPosition::CPU,
|
|
|
|
ApplicationPosition::DISK => ApplicationPosition::TEMP,
|
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-16 07:21:44 +00:00
|
|
|
self.awaiting_second_d = false;
|
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 {
|
|
|
|
ApplicationPosition::CPU => ApplicationPosition::MEM,
|
|
|
|
ApplicationPosition::MEM => ApplicationPosition::NETWORK,
|
|
|
|
ApplicationPosition::TEMP => ApplicationPosition::DISK,
|
|
|
|
ApplicationPosition::DISK => ApplicationPosition::PROCESS,
|
|
|
|
_ => self.current_application_position,
|
|
|
|
};
|
2019-12-16 07:21:44 +00:00
|
|
|
self.awaiting_second_d = false;
|
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-09-16 20:18:42 +00:00
|
|
|
ApplicationPosition::PROCESS => self.change_process_position(-1),
|
|
|
|
ApplicationPosition::TEMP => self.change_temp_position(-1),
|
|
|
|
ApplicationPosition::DISK => self.change_disk_position(-1),
|
2019-09-15 18:16:18 +00:00
|
|
|
_ => {}
|
|
|
|
}
|
2019-09-16 20:18:42 +00:00
|
|
|
self.scroll_direction = ScrollDirection::UP;
|
2019-12-16 07:21:44 +00:00
|
|
|
self.awaiting_second_d = false;
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn increment_position_count(&mut self) {
|
|
|
|
match self.current_application_position {
|
2019-09-16 20:18:42 +00:00
|
|
|
ApplicationPosition::PROCESS => self.change_process_position(1),
|
|
|
|
ApplicationPosition::TEMP => self.change_temp_position(1),
|
|
|
|
ApplicationPosition::DISK => self.change_disk_position(1),
|
2019-09-15 18:16:18 +00:00
|
|
|
_ => {}
|
|
|
|
}
|
2019-09-16 20:18:42 +00:00
|
|
|
self.scroll_direction = ScrollDirection::DOWN;
|
2019-12-16 07:21:44 +00:00
|
|
|
self.awaiting_second_d = false;
|
2019-09-15 18:16:18 +00:00
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|