mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 09:52:07 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			163 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			163 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | |
|  * Copyright (c) 2022, the SerenityOS developers.
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include "GraphWidget.h"
 | |
| #include <LibCore/EventReceiver.h>
 | |
| #include <LibGUI/Application.h>
 | |
| #include <LibGUI/Painter.h>
 | |
| #include <LibGfx/Font/Font.h>
 | |
| #include <LibGfx/Palette.h>
 | |
| #include <LibGfx/Path.h>
 | |
| #include <LibGfx/SystemTheme.h>
 | |
| 
 | |
| REGISTER_WIDGET(SystemMonitor, GraphWidget)
 | |
| 
 | |
| namespace SystemMonitor {
 | |
| 
 | |
| GraphWidget::GraphWidget()
 | |
| {
 | |
|     REGISTER_BOOL_PROPERTY("stack_values", stack_values, set_stack_values);
 | |
| }
 | |
| 
 | |
| void GraphWidget::set_stack_values(bool stack_values)
 | |
| {
 | |
|     m_stack_values = stack_values;
 | |
|     update();
 | |
| }
 | |
| 
 | |
| void GraphWidget::add_value(Vector<u64, 1>&& value)
 | |
| {
 | |
|     m_values.enqueue(move(value));
 | |
|     update();
 | |
| }
 | |
| 
 | |
| void GraphWidget::paint_event(GUI::PaintEvent& event)
 | |
| {
 | |
|     auto const& system_palette = GUI::Application::the()->palette();
 | |
| 
 | |
|     GUI::Frame::paint_event(event);
 | |
|     GUI::Painter painter(*this);
 | |
|     painter.add_clip_rect(event.rect());
 | |
|     painter.add_clip_rect(frame_inner_rect());
 | |
|     painter.fill_rect(event.rect(), palette().base());
 | |
| 
 | |
|     auto inner_rect = frame_inner_rect();
 | |
|     float scale = (float)inner_rect.height() / (float)m_max;
 | |
| 
 | |
|     if (!m_values.is_empty()) {
 | |
|         // Draw one set of values at a time
 | |
|         for (size_t k = 0; k < m_value_format.size(); k++) {
 | |
|             auto const& format = m_value_format[k];
 | |
|             if (format.graph_color_role == ColorRole::Base) {
 | |
|                 continue;
 | |
|             }
 | |
|             auto const& line_color = system_palette.color(format.graph_color_role);
 | |
|             auto const& background_color = line_color.with_alpha(0x7f);
 | |
|             m_calculated_points.clear_with_capacity();
 | |
|             for (size_t i = 0; i < m_values.size(); i++) {
 | |
|                 int x = inner_rect.right() - i * 2;
 | |
|                 if (x < 0)
 | |
|                     break;
 | |
|                 auto const& current_values = m_values.at(m_values.size() - i - 1);
 | |
|                 if (current_values.size() <= k) {
 | |
|                     // Don't have a data point
 | |
|                     m_calculated_points.append({ -1, -1 });
 | |
|                     continue;
 | |
|                 }
 | |
|                 float value = current_values[k];
 | |
|                 if (m_stack_values) {
 | |
|                     for (size_t l = k + 1; l < current_values.size(); l++)
 | |
|                         value += current_values[l];
 | |
|                 }
 | |
|                 float scaled_value = value * scale;
 | |
|                 Gfx::IntPoint current_point { x, inner_rect.bottom() - 1 - (int)scaled_value };
 | |
|                 m_calculated_points.append(current_point);
 | |
|             }
 | |
|             VERIFY(m_calculated_points.size() <= m_values.size());
 | |
|             if (format.graph_color_role != ColorRole::Base) {
 | |
|                 // Fill the background for the area we have values for
 | |
|                 Gfx::Path path;
 | |
|                 size_t points_in_path = 0;
 | |
|                 bool started_path = false;
 | |
|                 Gfx::IntPoint const* current_point = nullptr;
 | |
|                 Gfx::IntPoint const* first_point = nullptr;
 | |
|                 auto check_fill_area = [&]() {
 | |
|                     if (!started_path)
 | |
|                         return;
 | |
|                     if (points_in_path > 1) {
 | |
|                         VERIFY(current_point);
 | |
|                         VERIFY(first_point);
 | |
|                         path.line_to({ current_point->x() - 1, inner_rect.bottom() });
 | |
|                         path.line_to({ first_point->x() + 1, inner_rect.bottom() });
 | |
|                         path.close();
 | |
|                         painter.fill_path(path, background_color, Gfx::Painter::WindingRule::EvenOdd);
 | |
|                     } else if (points_in_path == 1 && current_point) {
 | |
|                         // Can't fill any area, we only have one data point.
 | |
|                         // Just draw a vertical line as a "fill"...
 | |
|                         painter.draw_line(*current_point, { current_point->x(), inner_rect.bottom() - 1 }, background_color);
 | |
|                     }
 | |
|                     path = {};
 | |
|                     points_in_path = 0;
 | |
|                     first_point = nullptr;
 | |
|                     started_path = false;
 | |
|                 };
 | |
|                 for (size_t i = 0; i < m_calculated_points.size(); i++) {
 | |
|                     current_point = &m_calculated_points[i];
 | |
|                     if (current_point->x() < 0) {
 | |
|                         check_fill_area();
 | |
|                         continue;
 | |
|                     }
 | |
|                     if (!started_path) {
 | |
|                         path.move_to({ current_point->x() + 1, current_point->y() });
 | |
|                         points_in_path = 1;
 | |
|                         first_point = current_point;
 | |
|                         started_path = true;
 | |
|                     } else {
 | |
|                         path.line_to({ current_point->x(), current_point->y() });
 | |
|                         points_in_path++;
 | |
|                     }
 | |
|                 }
 | |
|                 check_fill_area();
 | |
|             }
 | |
|             if (format.graph_color_role != ColorRole::Base) {
 | |
|                 // Draw the line for the data points we have
 | |
|                 Gfx::IntPoint const* previous_point = nullptr;
 | |
|                 for (size_t i = 0; i < m_calculated_points.size(); i++) {
 | |
|                     auto const& current_point = m_calculated_points[i];
 | |
|                     if (current_point.x() < 0) {
 | |
|                         previous_point = nullptr;
 | |
|                         continue;
 | |
|                     }
 | |
|                     if (previous_point)
 | |
|                         painter.draw_line(*previous_point, current_point, line_color);
 | |
|                     previous_point = ¤t_point;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!m_values.is_empty() && !m_value_format.is_empty()) {
 | |
|         auto const& current_values = m_values.last();
 | |
|         int y = 0;
 | |
|         for (size_t i = 0; i < min(m_value_format.size(), current_values.size()); i++) {
 | |
|             auto const& format = m_value_format[i];
 | |
|             auto const& graph_color = system_palette.color(format.graph_color_role);
 | |
|             if (!format.text_formatter)
 | |
|                 continue;
 | |
|             auto constrain_rect = inner_rect.shrunken(8, 8);
 | |
|             auto text_rect = constrain_rect.translated(0, y).intersected(constrain_rect);
 | |
|             text_rect.set_height(font().pixel_size_rounded_up());
 | |
|             auto text = format.text_formatter(current_values[i]);
 | |
|             if (format.text_shadow_color != Color::Transparent)
 | |
|                 painter.draw_text(text_rect.translated(1, 1), text, Gfx::TextAlignment::CenterRight, format.text_shadow_color);
 | |
|             painter.draw_text(text_rect, text, Gfx::TextAlignment::CenterRight, graph_color);
 | |
|             y += text_rect.height() + 4;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| }
 | 
