mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2024-11-22 23:50:19 +00:00
Userland: Piano: Optimize repaints
The Piano application used to perform very poorly due to unnecessary draw calls. This is solved with two optimziations: 1. Don't draw the widgets as often as possible. The widgets are instead at least updated every 150ms, except for other events. 2. Don't re-draw the entire piano roll sheet. The piano roll background, excluding in-motion objects (notes, the play cursor), is only re-drawn when its "viewport" changes. A minor drawback of this change is that notes will appear on top of the pitch labels if placed at the left edge of the roll. This is IMO acceptable or may be changed by moving the text to the "foreground".
This commit is contained in:
parent
511ffa8d68
commit
c1345bda3e
Notes:
sideshowbarker
2024-07-18 19:11:00 +09:00
Author: https://github.com/kleinesfilmroellchen 🔰 Commit: https://github.com/SerenityOS/serenity/commit/c1345bda3ed Pull-request: https://github.com/SerenityOS/serenity/pull/6495 Reviewed-by: https://github.com/awesomekling
3 changed files with 79 additions and 19 deletions
|
@ -1,6 +1,7 @@
|
|||
/*
|
||||
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
||||
* Copyright (c) 2019-2020, William McPherson <willmcpherson2@gmail.com>
|
||||
* Copyright (c) 2021, kleines Filmröllchen <malu.bertsch@gmail.com>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
@ -68,6 +69,62 @@ void RollWidget::paint_event(GUI::PaintEvent& event)
|
|||
int horizontal_notes_to_paint = horizontal_paint_area / m_note_width;
|
||||
|
||||
GUI::Painter painter(*this);
|
||||
|
||||
// Draw the background, if necessary.
|
||||
if (viewport_changed() || paint_area != m_background->height()) {
|
||||
m_background = Gfx::Bitmap::create(Gfx::BitmapFormat::BGRx8888, Gfx::IntSize(m_roll_width, paint_area));
|
||||
Gfx::Painter background_painter(*m_background);
|
||||
|
||||
background_painter.translate(frame_thickness(), frame_thickness());
|
||||
background_painter.translate(-horizontal_note_offset_remainder, -note_offset_remainder);
|
||||
|
||||
for (int y = 0; y < notes_to_paint; ++y) {
|
||||
int y_pos = y * note_height;
|
||||
|
||||
for (int x = 0; x < horizontal_notes_to_paint; ++x) {
|
||||
// This is needed to avoid rounding errors. You can't just use
|
||||
// m_note_width as the width.
|
||||
int x_pos = x * m_note_width;
|
||||
int next_x_pos = (x + 1) * m_note_width;
|
||||
int distance_to_next_x = next_x_pos - x_pos;
|
||||
Gfx::IntRect rect(x_pos, y_pos, distance_to_next_x, note_height);
|
||||
|
||||
if (key_pattern[key_pattern_index] == Black)
|
||||
background_painter.fill_rect(rect, Color::LightGray);
|
||||
else
|
||||
background_painter.fill_rect(rect, Color::White);
|
||||
|
||||
background_painter.draw_line(rect.top_right(), rect.bottom_right(), Color::Black);
|
||||
background_painter.draw_line(rect.bottom_left(), rect.bottom_right(), Color::Black);
|
||||
}
|
||||
|
||||
if (--key_pattern_index == -1)
|
||||
key_pattern_index = notes_per_octave - 1;
|
||||
}
|
||||
|
||||
background_painter.translate(-x_offset, -y_offset);
|
||||
background_painter.translate(horizontal_note_offset_remainder, note_offset_remainder);
|
||||
|
||||
for (int note = note_count - (note_offset + notes_to_paint); note <= (note_count - 1) - note_offset; ++note) {
|
||||
int y = ((note_count - 1) - note) * note_height;
|
||||
|
||||
Gfx::IntRect note_name_rect(3, y, 1, note_height);
|
||||
const char* note_name = note_names[note % notes_per_octave];
|
||||
|
||||
background_painter.draw_text(note_name_rect, note_name, Gfx::TextAlignment::CenterLeft);
|
||||
note_name_rect.move_by(Gfx::FontDatabase::default_font().width(note_name) + 2, 0);
|
||||
if (note % notes_per_octave == 0)
|
||||
background_painter.draw_text(note_name_rect, String::formatted("{}", note / notes_per_octave + 1), Gfx::TextAlignment::CenterLeft);
|
||||
}
|
||||
|
||||
m_prev_zoom_level = m_zoom_level;
|
||||
m_prev_scroll_x = horizontal_scrollbar().value();
|
||||
m_prev_scroll_y = vertical_scrollbar().value();
|
||||
}
|
||||
|
||||
painter.blit(Gfx::IntPoint(0, 0), *m_background, m_background->rect());
|
||||
|
||||
// Draw the notes, mouse interaction, and time position.
|
||||
painter.translate(frame_thickness(), frame_thickness());
|
||||
painter.add_clip_rect(event.rect());
|
||||
painter.translate(-horizontal_note_offset_remainder, -note_offset_remainder);
|
||||
|
@ -84,20 +141,9 @@ void RollWidget::paint_event(GUI::PaintEvent& event)
|
|||
int distance_to_next_x = next_x_pos - x_pos;
|
||||
Gfx::IntRect rect(x_pos, y_pos, distance_to_next_x, note_height);
|
||||
|
||||
if (key_pattern[key_pattern_index] == Black)
|
||||
painter.fill_rect(rect, Color::LightGray);
|
||||
else
|
||||
painter.fill_rect(rect, Color::White);
|
||||
|
||||
if (keys_widget() && keys_widget()->note_is_set(note))
|
||||
painter.fill_rect(rect, note_pressed_color.with_alpha(128));
|
||||
|
||||
painter.draw_line(rect.top_right(), rect.bottom_right(), Color::Black);
|
||||
painter.draw_line(rect.bottom_left(), rect.bottom_right(), Color::Black);
|
||||
}
|
||||
|
||||
if (--key_pattern_index == -1)
|
||||
key_pattern_index = notes_per_octave - 1;
|
||||
}
|
||||
|
||||
painter.translate(-x_offset, -y_offset);
|
||||
|
@ -119,13 +165,6 @@ void RollWidget::paint_event(GUI::PaintEvent& event)
|
|||
painter.fill_rect(rect, note_pressed_color);
|
||||
painter.draw_rect(rect, Color::Black);
|
||||
}
|
||||
Gfx::IntRect note_name_rect(3, y, 1, note_height);
|
||||
const char* note_name = note_names[note % notes_per_octave];
|
||||
|
||||
painter.draw_text(note_name_rect, note_name, Gfx::TextAlignment::CenterLeft);
|
||||
note_name_rect.move_by(Gfx::FontDatabase::default_font().width(note_name) + 2, 0);
|
||||
if (note % notes_per_octave == 0)
|
||||
painter.draw_text(note_name_rect, String::formatted("{}", note / notes_per_octave + 1), Gfx::TextAlignment::CenterLeft);
|
||||
}
|
||||
|
||||
int x = m_roll_width * (static_cast<double>(m_track_manager.time()) / roll_length);
|
||||
|
@ -135,6 +174,15 @@ void RollWidget::paint_event(GUI::PaintEvent& event)
|
|||
GUI::Frame::paint_event(event);
|
||||
}
|
||||
|
||||
bool RollWidget::viewport_changed() const
|
||||
{
|
||||
// height is complicated to check, will be done in paint_event
|
||||
return m_background.is_null()
|
||||
|| m_roll_width != m_background->width()
|
||||
|| m_prev_scroll_x != horizontal_scrollbar().value() || m_prev_scroll_y != vertical_scrollbar().value()
|
||||
|| m_prev_zoom_level != m_zoom_level;
|
||||
}
|
||||
|
||||
void RollWidget::mousedown_event(GUI::MouseEvent& event)
|
||||
{
|
||||
if (!widget_inner_rect().contains(event.x(), event.y()))
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
/*
|
||||
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
||||
* Copyright (c) 2019-2020, William McPherson <willmcpherson2@gmail.com>
|
||||
* Copyright (c) 2021, kleines Filmröllchen <malu.bertsch@gmail.com>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
@ -29,6 +30,7 @@ private:
|
|||
virtual void mousemove_event(GUI::MouseEvent& event) override;
|
||||
virtual void mouseup_event(GUI::MouseEvent& event) override;
|
||||
virtual void mousewheel_event(GUI::MouseEvent&) override;
|
||||
bool viewport_changed() const;
|
||||
|
||||
TrackManager& m_track_manager;
|
||||
const KeysWidget* m_keys_widget;
|
||||
|
@ -41,4 +43,9 @@ private:
|
|||
Optional<Gfx::IntPoint> m_note_drag_start;
|
||||
Optional<RollNote> m_note_drag_location;
|
||||
int m_drag_note;
|
||||
|
||||
RefPtr<Gfx::Bitmap> m_background;
|
||||
int m_prev_zoom_level { m_zoom_level };
|
||||
int m_prev_scroll_x { horizontal_scrollbar().value() };
|
||||
int m_prev_scroll_y { vertical_scrollbar().value() };
|
||||
};
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
/*
|
||||
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
||||
* Copyright (c) 2019-2020, William McPherson <willmcpherson2@gmail.com>
|
||||
* Copyright (c) 2021, kleines Filmröllchen <malu.bertsch@gmail.com>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
@ -64,7 +65,6 @@ int main(int argc, char** argv)
|
|||
while (!Core::EventLoop::current().was_exit_requested()) {
|
||||
track_manager.fill_buffer(buffer);
|
||||
audio->write(reinterpret_cast<u8*>(buffer.data()), buffer_size);
|
||||
Core::EventLoop::current().post_event(main_widget, make<Core::CustomEvent>(0));
|
||||
Core::EventLoop::wake();
|
||||
|
||||
if (need_to_write_wav) {
|
||||
|
@ -84,6 +84,11 @@ int main(int argc, char** argv)
|
|||
});
|
||||
audio_thread->start();
|
||||
|
||||
auto main_widget_updater = Core::Timer::construct(150, [&] {
|
||||
Core::EventLoop::current().post_event(main_widget, make<Core::CustomEvent>(0));
|
||||
});
|
||||
main_widget_updater->start();
|
||||
|
||||
auto menubar = GUI::Menubar::construct();
|
||||
|
||||
auto& app_menu = menubar->add_menu("File");
|
||||
|
|
Loading…
Reference in a new issue