mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-23 09:51:57 -05:00
142 lines
3.6 KiB
C++
142 lines
3.6 KiB
C++
#include "Window.h"
|
|
#include "WindowManager.h"
|
|
#include "Event.h"
|
|
#include "EventLoop.h"
|
|
#include "Widget.h"
|
|
|
|
Window::Window(Object* parent)
|
|
: Object(parent)
|
|
{
|
|
WindowManager::the().addWindow(*this);
|
|
}
|
|
|
|
Window::~Window()
|
|
{
|
|
delete m_mainWidget;
|
|
m_mainWidget = nullptr;
|
|
if (parent())
|
|
parent()->removeChild(*this);
|
|
WindowManager::the().removeWindow(*this);
|
|
}
|
|
|
|
void Window::setMainWidget(Widget* widget)
|
|
{
|
|
if (m_mainWidget == widget)
|
|
return;
|
|
|
|
m_mainWidget = widget;
|
|
widget->setWindow(this);
|
|
}
|
|
|
|
void Window::setTitle(String&& title)
|
|
{
|
|
if (m_title == title)
|
|
return;
|
|
|
|
m_title = move(title);
|
|
WindowManager::the().notifyTitleChanged(*this);
|
|
}
|
|
|
|
void Window::setRect(const Rect& rect)
|
|
{
|
|
if (m_rect == rect)
|
|
return;
|
|
auto oldRect = m_rect;
|
|
m_rect = rect;
|
|
m_backing = GraphicsBitmap::create(m_rect.size());
|
|
WindowManager::the().notifyRectChanged(*this, oldRect, m_rect);
|
|
}
|
|
|
|
void Window::repaint(const Rect& rect)
|
|
{
|
|
event(*make<PaintEvent>(rect));
|
|
}
|
|
|
|
void Window::update(const Rect& rect)
|
|
{
|
|
EventLoop::main().postEvent(this, make<PaintEvent>(rect));
|
|
}
|
|
|
|
void Window::event(Event& event)
|
|
{
|
|
if (event.isMouseEvent()) {
|
|
auto& me = static_cast<MouseEvent&>(event);
|
|
//printf("Window{%p}: %s %d,%d\n", this, me.name(), me.x(), me.y());
|
|
if (m_mainWidget) {
|
|
auto result = m_mainWidget->hitTest(me.x(), me.y());
|
|
//printf("hit test for %d,%d found: %s{%p} %d,%d\n", me.x(), me.y(), result.widget->class_name(), result.widget, result.localX, result.localY);
|
|
// FIXME: Re-use the existing event instead of crafting a new one?
|
|
auto localEvent = make<MouseEvent>(event.type(), result.localX, result.localY, me.button());
|
|
return result.widget->event(*localEvent);
|
|
}
|
|
return Object::event(event);
|
|
}
|
|
|
|
if (event.isPaintEvent()) {
|
|
auto& pe = static_cast<PaintEvent&>(event);
|
|
printf("Window[\"%s\"]: paintEvent %d,%d %dx%d\n", title().characters(),
|
|
pe.rect().x(),
|
|
pe.rect().y(),
|
|
pe.rect().width(),
|
|
pe.rect().height());
|
|
|
|
if (isBeingDragged()) {
|
|
// Ignore paint events during window drag.
|
|
return;
|
|
}
|
|
if (m_mainWidget) {
|
|
if (pe.rect().is_empty())
|
|
m_mainWidget->event(*make<PaintEvent>(m_mainWidget->rect()));
|
|
else
|
|
m_mainWidget->event(event);
|
|
WindowManager::the().did_paint(*this);
|
|
return;
|
|
}
|
|
return Object::event(event);
|
|
}
|
|
|
|
if (event.isKeyEvent()) {
|
|
if (m_focusedWidget)
|
|
return m_focusedWidget->event(event);
|
|
return Object::event(event);
|
|
}
|
|
|
|
return Object::event(event);
|
|
}
|
|
|
|
void Window::did_paint()
|
|
{
|
|
WindowManager::the().did_paint(*this);
|
|
}
|
|
|
|
bool Window::isActive() const
|
|
{
|
|
return WindowManager::the().activeWindow() == this;
|
|
}
|
|
|
|
bool Window::isVisible() const
|
|
{
|
|
return WindowManager::the().isVisible(const_cast<Window&>(*this));
|
|
}
|
|
|
|
void Window::setFocusedWidget(Widget* widget)
|
|
{
|
|
if (m_focusedWidget.ptr() == widget)
|
|
return;
|
|
auto* previously_focused_widget = m_focusedWidget.ptr();
|
|
if (!widget)
|
|
m_focusedWidget = nullptr;
|
|
else {
|
|
m_focusedWidget = widget->makeWeakPtr();
|
|
EventLoop::main().postEvent(m_focusedWidget.ptr(), make<Event>(Event::FocusIn));
|
|
}
|
|
if (previously_focused_widget)
|
|
EventLoop::main().postEvent(previously_focused_widget, make<Event>(Event::FocusOut));
|
|
}
|
|
|
|
void Window::close()
|
|
{
|
|
WindowManager::the().removeWindow(*this);
|
|
deleteLater();
|
|
}
|
|
|