/* * Copyright (c) 2022, kleines Filmröllchen * * SPDX-License-Identifier: BSD-2-Clause */ #include "VectorscopeWidget.h" #include "Layer.h" #include #include #include #include #include #include #include #include #include #include #include REGISTER_WIDGET(PixelPaint, VectorscopeWidget); namespace PixelPaint { void VectorscopeWidget::image_changed() { (void)rebuild_vectorscope_data(); rebuild_vectorscope_image(); update(); } ErrorOr VectorscopeWidget::rebuild_vectorscope_data() { if (!m_image) return {}; m_vectorscope_data.fill({}); VERIFY(AK::abs(m_vectorscope_data[0][0]) < 0.01f); auto full_bitmap = TRY(m_image->try_compose_bitmap(Gfx::BitmapFormat::BGRA8888)); for (size_t x = 0; x < static_cast(full_bitmap->width()); ++x) { for (size_t y = 0; y < static_cast(full_bitmap->height()); ++y) { auto yuv = full_bitmap->get_pixel(x, y).to_yuv(); auto u_index = u_v_to_index(yuv.u); auto v_index = u_v_to_index(yuv.v); m_vectorscope_data[u_index][v_index]++; } } auto maximum = full_bitmap->width() * full_bitmap->height() * pixel_percentage_for_max_brightness * pixel_percentage_for_max_brightness; for (size_t i = 0; i < m_vectorscope_data.size(); ++i) { for (size_t j = 0; j < m_vectorscope_data[i].size(); ++j) { m_vectorscope_data[i][j] = AK::sqrt(m_vectorscope_data[i][j]) / maximum; } } return {}; } void VectorscopeWidget::rebuild_vectorscope_image() { m_vectorscope_image = MUST(Gfx::Bitmap::try_create(Gfx::BitmapFormat::BGRA8888, size())); m_vectorscope_image->fill(Color::Transparent); Gfx::Painter base_painter(*m_vectorscope_image); Gfx::AntiAliasingPainter painter(base_painter); auto const scope_size = min(height(), width()); auto const min_scope_size = parent_widget()->min_height().as_int(); auto const color_vector_scale = scope_size / static_cast(min_scope_size); auto const size_1x1 = Gfx::FloatSize { 2.5f, 2.5f } * static_cast(color_vector_scale); base_painter.translate(width() / 2, height() / 2); painter.translate(static_cast(width()) / 2.0f, static_cast(height()) / 2.0f); for (size_t u_index = 0; u_index < u_v_steps; ++u_index) { for (size_t v_index = 0; v_index < u_v_steps; ++v_index) { auto const color_vector = ColorVector::from_indices(u_index, v_index); auto const brightness = m_vectorscope_data[u_index][v_index]; if (brightness < 0.0001f) continue; auto const pseudo_rect = Gfx::FloatRect::centered_at(color_vector.to_vector(scope_size) * 2.0f, size_1x1); auto color = Color::from_yuv(0.6f, color_vector.u, color_vector.v); color = color.saturated_to(1.0f - min(brightness, 1.0f)); color.set_alpha(static_cast(min(AK::sqrt(brightness), alpha_range) * NumericLimits::max() / alpha_range)); painter.fill_rect(pseudo_rect, color); } } } void VectorscopeWidget::paint_event(GUI::PaintEvent& event) { GUI::Painter base_painter(*this); Gfx::AntiAliasingPainter painter(base_painter); base_painter.add_clip_rect(event.rect()); // From this point on we're working with 0,0 as the scope center to make things easier. base_painter.translate(width() / 2, height() / 2); painter.translate(static_cast(width()) / 2.0f, static_cast(height()) / 2.0f); auto const graticule_color = Color::White; auto const scope_size = min(height(), width()); auto const graticule_size = scope_size / 6; auto const graticule_thickness = graticule_size / 12; auto const entire_scope_rect = Gfx::FloatRect::centered_at({ 0, 0 }, { scope_size, scope_size }); painter.fill_ellipse(entire_scope_rect.to_rounded().shrunken(graticule_thickness * 2, graticule_thickness * 2), Color::Black); // Main scope data if (m_image) { if (m_vectorscope_image->size() != this->size()) rebuild_vectorscope_image(); base_painter.blit({ -width() / 2, -height() / 2 }, *m_vectorscope_image, m_vectorscope_image->rect()); } // Graticule(s) painter.draw_ellipse(entire_scope_rect.to_rounded(), graticule_color, graticule_thickness); // FIXME: Translation calls to the painters don't appear to work correctly, and I figured out a combination of calls through trial and error that do what I want, but I don't know how they do that. // Translation does work correctly with things like rectangle and text drawing, so it's very strange. painter.translate(-static_cast(width()) / 2.0f, -static_cast(height()) / 2.0f); // We intentionally draw the skin tone line much further than the actual color we're using for it. painter.draw_line({ 0, 0 }, skin_tone_color.to_vector(scope_size) * 2.0, graticule_color); painter.translate(-static_cast(width()) / 2.0f, -static_cast(height()) / 2.0f); for (auto const& primary_color : primary_colors) { // FIXME: Only draw the rectangle corners for a more classical oscilloscope look (& less obscuring of color data) auto graticule_rect = Gfx::FloatRect::centered_at(primary_color.to_vector(scope_size), { graticule_size, graticule_size }).to_rounded(); base_painter.draw_rect_with_thickness(graticule_rect, graticule_color, graticule_thickness); auto text_rect = graticule_rect.translated(graticule_size / 2, graticule_size / 2); base_painter.draw_text(text_rect, StringView { &primary_color.symbol, 1 }, Gfx::TextAlignment::TopLeft, graticule_color); } if (m_color_at_mouseposition != Color::Transparent) { auto color_vector = ColorVector { m_color_at_mouseposition }; painter.draw_ellipse(Gfx::FloatRect::centered_at(color_vector.to_vector(scope_size) * 2.0, { graticule_size, graticule_size }).to_rounded(), graticule_color, graticule_thickness); } } }