mirror of
https://github.com/aseprite/aseprite.git
synced 2025-04-16 05:42:32 +00:00
245 lines
5.7 KiB
C++
245 lines
5.7 KiB
C++
// Aseprite UI Library
|
|
// Copyright (C) 2001-2013, 2015 David Capello
|
|
//
|
|
// This file is released under the terms of the MIT license.
|
|
// Read LICENSE.txt for more information.
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "gfx/size.h"
|
|
#include "ui/graphics.h"
|
|
#include "ui/intern.h"
|
|
#include "ui/size_hint_event.h"
|
|
#include "ui/theme.h"
|
|
#include "ui/ui.h"
|
|
|
|
namespace ui {
|
|
|
|
using namespace gfx;
|
|
|
|
PopupWindow::PopupWindow(const std::string& text,
|
|
ClickBehavior clickBehavior,
|
|
EnterBehavior enterBehavior)
|
|
: Window(text.empty() ? WithoutTitleBar: WithTitleBar, text)
|
|
, m_clickBehavior(clickBehavior)
|
|
, m_enterBehavior(enterBehavior)
|
|
, m_filtering(false)
|
|
{
|
|
setSizeable(false);
|
|
setMoveable(false);
|
|
setWantFocus(false);
|
|
setAlign(LEFT | TOP);
|
|
|
|
removeDecorativeWidgets();
|
|
|
|
initTheme();
|
|
noBorderNoChildSpacing();
|
|
}
|
|
|
|
PopupWindow::~PopupWindow()
|
|
{
|
|
stopFilteringMessages();
|
|
}
|
|
|
|
void PopupWindow::setHotRegion(const gfx::Region& region)
|
|
{
|
|
startFilteringMessages();
|
|
|
|
m_hotRegion = region;
|
|
}
|
|
|
|
void PopupWindow::setClickBehavior(ClickBehavior behavior)
|
|
{
|
|
m_clickBehavior = behavior;
|
|
}
|
|
|
|
void PopupWindow::setEnterBehavior(EnterBehavior behavior)
|
|
{
|
|
m_enterBehavior = behavior;
|
|
}
|
|
|
|
void PopupWindow::makeFloating()
|
|
{
|
|
stopFilteringMessages();
|
|
setMoveable(true);
|
|
}
|
|
|
|
void PopupWindow::makeFixed()
|
|
{
|
|
startFilteringMessages();
|
|
setMoveable(false);
|
|
}
|
|
|
|
bool PopupWindow::onProcessMessage(Message* msg)
|
|
{
|
|
switch (msg->type()) {
|
|
|
|
// There are cases where startFilteringMessages() is called when a
|
|
// kCloseMessage for this same PopupWindow is enqueued. Processing
|
|
// the kOpenMessage we ensure that the popup will be filtering
|
|
// messages if it's needed when it's visible (as kCloseMessage and
|
|
// kOpenMessage must be enqueued in the correct order).
|
|
case kOpenMessage:
|
|
if (!isMoveable())
|
|
startFilteringMessages();
|
|
break;
|
|
|
|
case kCloseMessage:
|
|
stopFilteringMessages();
|
|
break;
|
|
|
|
case kMouseLeaveMessage:
|
|
if (m_hotRegion.isEmpty() && !isMoveable())
|
|
closeWindow(nullptr);
|
|
break;
|
|
|
|
case kKeyDownMessage:
|
|
if (m_filtering) {
|
|
KeyMessage* keymsg = static_cast<KeyMessage*>(msg);
|
|
KeyScancode scancode = keymsg->scancode();
|
|
|
|
if (scancode == kKeyEsc)
|
|
closeWindow(nullptr);
|
|
|
|
if (m_enterBehavior == EnterBehavior::CloseOnEnter &&
|
|
(scancode == kKeyEnter ||
|
|
scancode == kKeyEnterPad)) {
|
|
closeWindow(this);
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case kMouseDownMessage:
|
|
if (m_filtering &&
|
|
manager()->getTopWindow() == this) {
|
|
gfx::Point mousePos = static_cast<MouseMessage*>(msg)->position();
|
|
|
|
switch (m_clickBehavior) {
|
|
|
|
// If the user click outside the window, we have to close
|
|
// the tooltip window.
|
|
case ClickBehavior::CloseOnClickInOtherWindow: {
|
|
Widget* picked = pick(mousePos);
|
|
if (!picked || picked->window() != this) {
|
|
closeWindow(NULL);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ClickBehavior::CloseOnClickOutsideHotRegion:
|
|
if (!m_hotRegion.contains(mousePos)) {
|
|
closeWindow(NULL);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case kMouseMoveMessage:
|
|
if (!isMoveable() &&
|
|
!m_hotRegion.isEmpty() &&
|
|
manager()->getCapture() == NULL) {
|
|
gfx::Point mousePos = static_cast<MouseMessage*>(msg)->position();
|
|
|
|
// If the mouse is outside the hot-region we have to close the
|
|
// window.
|
|
if (!m_hotRegion.contains(mousePos))
|
|
closeWindow(NULL);
|
|
}
|
|
break;
|
|
|
|
}
|
|
|
|
return Window::onProcessMessage(msg);
|
|
}
|
|
|
|
void PopupWindow::onSizeHint(SizeHintEvent& ev)
|
|
{
|
|
ScreenGraphics g;
|
|
g.setFont(font());
|
|
Size resultSize(0, 0);
|
|
|
|
if (hasText())
|
|
resultSize = g.fitString(text(),
|
|
(clientBounds() - border()).w,
|
|
align());
|
|
|
|
resultSize.w += border().width();
|
|
resultSize.h += border().height();
|
|
|
|
if (!children().empty()) {
|
|
Size maxSize(0, 0);
|
|
Size reqSize;
|
|
|
|
for (auto child : children()) {
|
|
reqSize = child->sizeHint();
|
|
|
|
maxSize.w = MAX(maxSize.w, reqSize.w);
|
|
maxSize.h = MAX(maxSize.h, reqSize.h);
|
|
}
|
|
|
|
resultSize.w = MAX(resultSize.w, maxSize.w + border().width());
|
|
resultSize.h += maxSize.h;
|
|
}
|
|
|
|
ev.setSizeHint(resultSize);
|
|
}
|
|
|
|
void PopupWindow::onPaint(PaintEvent& ev)
|
|
{
|
|
theme()->paintPopupWindow(ev);
|
|
}
|
|
|
|
void PopupWindow::onInitTheme(InitThemeEvent& ev)
|
|
{
|
|
Widget::onInitTheme(ev);
|
|
|
|
setBorder(gfx::Border(3 * guiscale()));
|
|
}
|
|
|
|
void PopupWindow::onHitTest(HitTestEvent& ev)
|
|
{
|
|
Widget* picked = manager()->pick(ev.point());
|
|
if (picked) {
|
|
WidgetType type = picked->type();
|
|
if ((type == kWindowWidget && picked == this) ||
|
|
type == kBoxWidget ||
|
|
type == kLabelWidget ||
|
|
type == kGridWidget ||
|
|
type == kSeparatorWidget) {
|
|
ev.setHit(HitTestCaption);
|
|
return;
|
|
}
|
|
}
|
|
Window::onHitTest(ev);
|
|
}
|
|
|
|
void PopupWindow::startFilteringMessages()
|
|
{
|
|
if (!m_filtering) {
|
|
m_filtering = true;
|
|
|
|
Manager* manager = Manager::getDefault();
|
|
manager->addMessageFilter(kMouseMoveMessage, this);
|
|
manager->addMessageFilter(kMouseDownMessage, this);
|
|
manager->addMessageFilter(kKeyDownMessage, this);
|
|
}
|
|
}
|
|
|
|
void PopupWindow::stopFilteringMessages()
|
|
{
|
|
if (m_filtering) {
|
|
m_filtering = false;
|
|
|
|
Manager* manager = Manager::getDefault();
|
|
manager->removeMessageFilter(kMouseMoveMessage, this);
|
|
manager->removeMessageFilter(kMouseDownMessage, this);
|
|
manager->removeMessageFilter(kKeyDownMessage, this);
|
|
}
|
|
}
|
|
|
|
} // namespace ui
|