mirror of
https://github.com/RGBCube/serenity
synced 2025-05-31 16:28:11 +00:00

This means all three window titles are visible, regardless of what the TitleHeight and BorderThickness values are. :^) The one exception is when TitleHeight is less than the height of the title text. WindowManager ensures that the real title height is at least enough to fit the text, so if the value is set to less than that, the window titles will start to overlap. So, don't set values that are impossibly small!
158 lines
5 KiB
C++
158 lines
5 KiB
C++
/*
|
|
* Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
|
|
* Copyright (c) 2021-2022, Sam Atkins <atkinssj@serenityos.org>
|
|
* Copyright (c) 2021, Antonio Di Stefano <tonio9681@gmail.com>
|
|
* Copyright (c) 2022, the SerenityOS developers.
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "PreviewWidget.h"
|
|
#include <AK/LexicalPath.h>
|
|
#include <AK/StringView.h>
|
|
#include <Applications/ThemeEditor/WindowPreviewGML.h>
|
|
#include <LibCore/MimeData.h>
|
|
#include <LibFileSystemAccessClient/Client.h>
|
|
#include <LibGUI/MessageBox.h>
|
|
#include <LibGUI/Painter.h>
|
|
#include <LibGfx/Bitmap.h>
|
|
#include <LibGfx/WindowTheme.h>
|
|
|
|
namespace ThemeEditor {
|
|
|
|
class MiniWidgetGallery final : public GUI::Widget {
|
|
C_OBJECT(MiniWidgetGallery);
|
|
|
|
public:
|
|
void set_preview_palette(Gfx::Palette const& palette)
|
|
{
|
|
set_palette(palette);
|
|
Function<void(GUI::Widget&)> recurse = [&](GUI::Widget& parent_widget) {
|
|
parent_widget.for_each_child_widget([&](auto& widget) {
|
|
widget.set_palette(palette);
|
|
recurse(widget);
|
|
return IterationDecision::Continue;
|
|
});
|
|
};
|
|
recurse(*this);
|
|
}
|
|
|
|
private:
|
|
MiniWidgetGallery()
|
|
{
|
|
load_from_gml(window_preview_gml);
|
|
|
|
for_each_child_widget([](auto& child) {
|
|
child.set_focus_policy(GUI::FocusPolicy::NoFocus);
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
};
|
|
|
|
PreviewWidget::PreviewWidget(Gfx::Palette const& initial_preview_palette)
|
|
: GUI::AbstractThemePreview(initial_preview_palette)
|
|
{
|
|
m_gallery = add<MiniWidgetGallery>();
|
|
set_greedy_for_hits(true);
|
|
}
|
|
|
|
void PreviewWidget::palette_changed()
|
|
{
|
|
m_gallery->set_preview_palette(preview_palette());
|
|
update_preview_window_locations();
|
|
}
|
|
|
|
void PreviewWidget::set_color_filter(OwnPtr<Gfx::ColorBlindnessFilter> color_filter)
|
|
{
|
|
m_color_filter = move(color_filter);
|
|
repaint();
|
|
}
|
|
|
|
void PreviewWidget::update_preview_window_locations()
|
|
{
|
|
auto& palette = preview_palette();
|
|
int window_title_height = palette.metric(Gfx::MetricRole::TitleHeight)
|
|
+ palette.metric(Gfx::MetricRole::BorderThickness);
|
|
|
|
constexpr int inactive_offset_x = -20;
|
|
int inactive_offset_y = -(window_title_height + 4);
|
|
constexpr int hightlight_offset_x = 140;
|
|
int hightlight_offset_y = window_title_height + 40;
|
|
|
|
m_active_window_rect = Gfx::IntRect(0, 0, 320, 220);
|
|
m_inactive_window_rect = m_active_window_rect.translated(inactive_offset_x, inactive_offset_y);
|
|
m_highlight_window_rect = Gfx::IntRect(m_active_window_rect.location(), { 160, 70 }).translated(hightlight_offset_x, hightlight_offset_y);
|
|
|
|
auto window_group = Array {
|
|
Window { m_active_window_rect },
|
|
Window { m_inactive_window_rect },
|
|
Window { m_highlight_window_rect },
|
|
};
|
|
|
|
center_window_group_within(window_group, frame_inner_rect());
|
|
|
|
m_gallery->set_relative_rect(m_active_window_rect);
|
|
}
|
|
|
|
void PreviewWidget::paint_preview(GUI::PaintEvent&)
|
|
{
|
|
paint_window("Inactive window", m_inactive_window_rect, Gfx::WindowTheme::WindowState::Inactive, active_window_icon());
|
|
paint_window("Active window", m_active_window_rect, Gfx::WindowTheme::WindowState::Active, inactive_window_icon());
|
|
}
|
|
|
|
void PreviewWidget::paint_hightlight_window()
|
|
{
|
|
GUI::Painter painter(*this);
|
|
paint_window("Highlight window", m_highlight_window_rect, Gfx::WindowTheme::WindowState::Highlighted, inactive_window_icon(), 1);
|
|
auto button_rect = Gfx::IntRect(0, 0, 80, 22).centered_within(m_highlight_window_rect);
|
|
Gfx::StylePainter::paint_button(painter, button_rect, preview_palette(), Gfx::ButtonStyle::Normal, false, false, false, true, false, false);
|
|
painter.draw_text(button_rect, ":^)", Gfx::TextAlignment::Center, preview_palette().color(foreground_role()), Gfx::TextElision::Right, Gfx::TextWrapping::DontWrap);
|
|
}
|
|
|
|
void PreviewWidget::second_paint_event(GUI::PaintEvent&)
|
|
{
|
|
GUI::Painter painter(*this);
|
|
|
|
paint_hightlight_window();
|
|
|
|
if (!m_color_filter)
|
|
return;
|
|
|
|
auto target = painter.target();
|
|
auto bitmap_clone_or_error = target->clone();
|
|
if (bitmap_clone_or_error.is_error())
|
|
return;
|
|
|
|
auto clone = bitmap_clone_or_error.release_value();
|
|
auto rect = target->rect();
|
|
|
|
m_color_filter->apply(*target, rect, *clone, rect);
|
|
}
|
|
|
|
void PreviewWidget::resize_event(GUI::ResizeEvent&)
|
|
{
|
|
update_preview_window_locations();
|
|
}
|
|
|
|
void PreviewWidget::drop_event(GUI::DropEvent& event)
|
|
{
|
|
event.accept();
|
|
window()->move_to_front();
|
|
|
|
if (event.mime_data().has_urls()) {
|
|
auto urls = event.mime_data().urls();
|
|
if (urls.is_empty())
|
|
return;
|
|
if (urls.size() > 1) {
|
|
GUI::MessageBox::show(window(), "ThemeEditor can only open one file at a time!", "One at a time please!", GUI::MessageBox::Type::Error);
|
|
return;
|
|
}
|
|
|
|
auto response = FileSystemAccessClient::Client::the().try_request_file(window(), urls.first().path(), Core::OpenMode::ReadOnly);
|
|
if (response.is_error())
|
|
return;
|
|
set_theme_from_file(*response.value());
|
|
}
|
|
}
|
|
|
|
}
|