123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- /*
- * 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/Object.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;
- }
- }
- }
- }
|