mirror of https://github.com/bsnes-emu/bsnes.git
370 lines
10 KiB
C++
370 lines
10 KiB
C++
#if defined(Hiro_Window)
|
|
|
|
namespace hiro {
|
|
|
|
auto pWindow::construct() -> void {
|
|
qtWindow = new QtWindow(*this);
|
|
qtWindow->setWindowTitle(" ");
|
|
|
|
//if program was given a name, try and set the window taskbar icon to a matching pixmap image
|
|
if(auto name = Application::state().name) {
|
|
if(file::exists({Path::user(), ".local/share/icons/", name, ".png"})) {
|
|
qtWindow->setWindowIcon(QIcon(QString::fromUtf8(string{Path::user(), ".local/share/icons/", name, ".png"})));
|
|
} else if(file::exists({"/usr/local/share/pixmaps/", name, ".png"})) {
|
|
qtWindow->setWindowIcon(QIcon(QString::fromUtf8(string{"/usr/local/share/pixmaps/", name, ".png"})));
|
|
} else if(file::exists({"/usr/share/pixmaps/", name, ".png"})) {
|
|
qtWindow->setWindowIcon(QIcon(QString::fromUtf8(string{"/usr/share/pixmaps/", name, ".png"})));
|
|
}
|
|
}
|
|
|
|
qtLayout = new QVBoxLayout(qtWindow);
|
|
qtLayout->setMargin(0);
|
|
qtLayout->setSpacing(0);
|
|
qtWindow->setLayout(qtLayout);
|
|
|
|
qtMenuBar = new QMenuBar(qtWindow);
|
|
qtMenuBar->setVisible(false);
|
|
qtLayout->addWidget(qtMenuBar);
|
|
|
|
qtContainer = new QWidget(qtWindow);
|
|
qtContainer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
|
|
qtContainer->setVisible(true);
|
|
qtLayout->addWidget(qtContainer);
|
|
|
|
qtStatusBar = new QStatusBar(qtWindow);
|
|
qtStatusBar->setSizeGripEnabled(true);
|
|
qtStatusBar->setVisible(false);
|
|
qtLayout->addWidget(qtStatusBar);
|
|
|
|
setBackgroundColor(state().backgroundColor);
|
|
setDroppable(state().droppable);
|
|
setGeometry(state().geometry);
|
|
setResizable(state().resizable);
|
|
setTitle(state().title);
|
|
}
|
|
|
|
auto pWindow::destruct() -> void {
|
|
if(Application::state().quit) return; //TODO: hack
|
|
delete qtStatusBar;
|
|
delete qtContainer;
|
|
delete qtMenuBar;
|
|
delete qtLayout;
|
|
delete qtWindow;
|
|
}
|
|
|
|
auto pWindow::append(sMenuBar menuBar) -> void {
|
|
}
|
|
|
|
auto pWindow::append(sSizable sizable) -> void {
|
|
}
|
|
|
|
auto pWindow::append(sStatusBar statusBar) -> void {
|
|
}
|
|
|
|
auto pWindow::focused() const -> bool {
|
|
return qtWindow->isActiveWindow() && !qtWindow->isMinimized();
|
|
}
|
|
|
|
auto pWindow::frameMargin() const -> Geometry {
|
|
if(state().fullScreen) return {
|
|
0, _menuHeight(),
|
|
0, _menuHeight() + _statusHeight()
|
|
};
|
|
return {
|
|
settings.geometry.frameX,
|
|
settings.geometry.frameY + _menuHeight(),
|
|
settings.geometry.frameWidth,
|
|
settings.geometry.frameHeight + _menuHeight() + _statusHeight()
|
|
};
|
|
}
|
|
|
|
auto pWindow::handle() const -> uintptr_t {
|
|
return (uintptr_t)qtWindow->winId();
|
|
}
|
|
|
|
auto pWindow::monitor() const -> uint {
|
|
//TODO
|
|
return 0;
|
|
}
|
|
|
|
auto pWindow::remove(sMenuBar menuBar) -> void {
|
|
//QMenuBar::removeMenu() does not exist
|
|
//qtMenu->clear();
|
|
//for(auto& menu : window.state.menu) append(menu);
|
|
}
|
|
|
|
auto pWindow::remove(sSizable sizable) -> void {
|
|
}
|
|
|
|
auto pWindow::remove(sStatusBar statusBar) -> void {
|
|
}
|
|
|
|
auto pWindow::setBackgroundColor(Color color) -> void {
|
|
static auto defaultColor = qtContainer->palette().color(QPalette::Background);
|
|
|
|
auto palette = qtContainer->palette();
|
|
palette.setColor(QPalette::Background, CreateColor(color, defaultColor));
|
|
qtContainer->setPalette(palette);
|
|
qtContainer->setAutoFillBackground((bool)color);
|
|
//translucency results are very unpleasant without a compositor; so disable for now
|
|
//qtWindow->setAttribute(Qt::WA_TranslucentBackground, color && color.alpha() != 255);
|
|
}
|
|
|
|
auto pWindow::setDismissable(bool dismissable) -> void {
|
|
}
|
|
|
|
auto pWindow::setDroppable(bool droppable) -> void {
|
|
qtWindow->setAcceptDrops(droppable);
|
|
}
|
|
|
|
auto pWindow::setEnabled(bool enabled) -> void {
|
|
}
|
|
|
|
auto pWindow::setFocused() -> void {
|
|
qtWindow->raise();
|
|
qtWindow->activateWindow();
|
|
}
|
|
|
|
auto pWindow::setFullScreen(bool fullScreen) -> void {
|
|
lock();
|
|
if(fullScreen) {
|
|
windowedGeometry = state().geometry;
|
|
qtLayout->setSizeConstraint(QLayout::SetDefaultConstraint);
|
|
qtContainer->setFixedSize(Desktop::size().width() - frameMargin().width(), Desktop::size().height() - frameMargin().height());
|
|
qtWindow->showFullScreen();
|
|
state().geometry = Monitor::geometry(Monitor::primary());
|
|
} else {
|
|
setResizable(state().resizable);
|
|
qtWindow->showNormal();
|
|
qtWindow->adjustSize();
|
|
self().setGeometry(windowedGeometry);
|
|
}
|
|
unlock();
|
|
}
|
|
|
|
auto pWindow::setGeometry(Geometry geometry) -> void {
|
|
auto lock = acquire();
|
|
Application::processEvents();
|
|
#if HIRO_QT==4
|
|
QApplication::syncX();
|
|
#elif HIRO_QT==5
|
|
QApplication::sync();
|
|
#endif
|
|
|
|
setResizable(state().resizable);
|
|
qtWindow->move(geometry.x() - frameMargin().x(), geometry.y() - frameMargin().y());
|
|
//qtWindow->adjustSize() fails if larger than 2/3rds screen size
|
|
qtWindow->resize(qtWindow->sizeHint());
|
|
qtContainer->setMinimumSize(1, 1);
|
|
if(state().resizable) {
|
|
setMaximumSize(state().maximumSize);
|
|
setMinimumSize(state().minimumSize);
|
|
} else {
|
|
setMaximumSize(geometry.size());
|
|
setMinimumSize(geometry.size());
|
|
}
|
|
}
|
|
|
|
auto pWindow::setMaximized(bool maximized) -> void {
|
|
//todo
|
|
}
|
|
|
|
auto pWindow::setMaximumSize(Size size) -> void {
|
|
static auto maximumSize = qtWindow->maximumSize();
|
|
|
|
if(!state().resizable) size = state().geometry.size();
|
|
if(size) {
|
|
qtWindow->setMaximumSize(size.width(), size.height() + _menuHeight() + _statusHeight());
|
|
} else {
|
|
qtWindow->setMaximumSize(maximumSize);
|
|
}
|
|
}
|
|
|
|
auto pWindow::setMinimized(bool minimized) -> void {
|
|
//todo
|
|
}
|
|
|
|
auto pWindow::setMinimumSize(Size size) -> void {
|
|
if(!state().resizable) size = state().geometry.size();
|
|
qtWindow->setMinimumSize(size.width(), size.height() + _menuHeight() + _statusHeight());
|
|
}
|
|
|
|
auto pWindow::setModal(bool modal) -> void {
|
|
if(modal) {
|
|
//windowModality can only be enabled while window is invisible
|
|
setVisible(false);
|
|
qtWindow->setWindowModality(Qt::ApplicationModal);
|
|
setVisible(true);
|
|
while(!Application::state().quit && state().modal) {
|
|
if(Application::state().onMain) {
|
|
Application::doMain();
|
|
} else {
|
|
usleep(20 * 1000);
|
|
}
|
|
Application::processEvents();
|
|
}
|
|
qtWindow->setWindowModality(Qt::NonModal);
|
|
}
|
|
}
|
|
|
|
auto pWindow::setResizable(bool resizable) -> void {
|
|
if(resizable) {
|
|
qtLayout->setSizeConstraint(QLayout::SetDefaultConstraint);
|
|
//qtContainer->setMinimumSize(state().geometry.width(), state().geometry.height());
|
|
} else {
|
|
qtLayout->setSizeConstraint(QLayout::SetFixedSize);
|
|
//qtContainer->setFixedSize(state().geometry.width(), state().geometry.height());
|
|
}
|
|
qtStatusBar->setSizeGripEnabled(resizable);
|
|
|
|
setMaximumSize(state().maximumSize);
|
|
setMinimumSize(state().minimumSize);
|
|
}
|
|
|
|
auto pWindow::setTitle(const string& text) -> void {
|
|
qtWindow->setWindowTitle(text ? QString::fromUtf8(text) : " ");
|
|
}
|
|
|
|
auto pWindow::setVisible(bool visible) -> void {
|
|
auto lock = acquire();
|
|
qtWindow->setVisible(visible);
|
|
if(visible) {
|
|
_updateFrameGeometry();
|
|
setGeometry(state().geometry);
|
|
}
|
|
}
|
|
|
|
auto pWindow::_append(mWidget& widget) -> void {
|
|
if(auto self = widget.self()) {
|
|
self->qtWidget->setParent(qtContainer);
|
|
}
|
|
}
|
|
|
|
auto pWindow::_menuHeight() const -> uint {
|
|
if(auto& menuBar = state().menuBar) {
|
|
if(menuBar->visible()) return settings.geometry.menuHeight + _menuTextHeight();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
auto pWindow::_menuTextHeight() const -> uint {
|
|
uint height = 0;
|
|
if(auto& menuBar = state().menuBar) {
|
|
for(auto& menu : menuBar->state.menus) {
|
|
height = max(height, menu->font(true).size(menu->text()).height());
|
|
}
|
|
}
|
|
return height;
|
|
}
|
|
|
|
auto pWindow::_statusHeight() const -> uint {
|
|
if(auto& statusBar = state().statusBar) {
|
|
if(statusBar->visible()) return settings.geometry.statusHeight + _statusTextHeight();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
auto pWindow::_statusTextHeight() const -> uint {
|
|
uint height = 0;
|
|
if(auto& statusBar = state().statusBar) {
|
|
height = statusBar->font(true).size(statusBar->text()).height();
|
|
}
|
|
return height;
|
|
}
|
|
|
|
auto pWindow::_updateFrameGeometry() -> void {
|
|
pApplication::synchronize();
|
|
QRect border = qtWindow->frameGeometry();
|
|
QRect client = qtWindow->geometry();
|
|
|
|
settings.geometry.frameX = client.x() - border.x();
|
|
settings.geometry.frameY = client.y() - border.y();
|
|
settings.geometry.frameWidth = border.width() - client.width();
|
|
settings.geometry.frameHeight = border.height() - client.height();
|
|
|
|
if(qtMenuBar->isVisible()) {
|
|
pApplication::synchronize();
|
|
settings.geometry.menuHeight = qtMenuBar->height() - _menuTextHeight();
|
|
}
|
|
|
|
if(qtStatusBar->isVisible()) {
|
|
pApplication::synchronize();
|
|
settings.geometry.statusHeight = qtStatusBar->height() - _statusTextHeight();
|
|
}
|
|
}
|
|
|
|
auto QtWindow::closeEvent(QCloseEvent* event) -> void {
|
|
event->ignore();
|
|
if(p.state().onClose) p.self().doClose();
|
|
else p.self().setVisible(false);
|
|
if(p.state().modal && !p.self().visible()) p.self().setModal(false);
|
|
}
|
|
|
|
auto QtWindow::moveEvent(QMoveEvent* event) -> void {
|
|
if(!p.locked() && !p.state().fullScreen && p.qtWindow->isVisible()) {
|
|
p.state().geometry.setPosition({
|
|
p.state().geometry.x() + event->pos().x() - event->oldPos().x(),
|
|
p.state().geometry.y() + event->pos().y() - event->oldPos().y()
|
|
});
|
|
}
|
|
|
|
if(!p.locked()) {
|
|
p.self().doMove();
|
|
}
|
|
}
|
|
|
|
auto QtWindow::dragEnterEvent(QDragEnterEvent* event) -> void {
|
|
if(event->mimeData()->hasUrls()) {
|
|
event->acceptProposedAction();
|
|
}
|
|
}
|
|
|
|
auto QtWindow::dropEvent(QDropEvent* event) -> void {
|
|
if(auto paths = DropPaths(event)) p.self().doDrop(paths);
|
|
}
|
|
|
|
auto QtWindow::keyPressEvent(QKeyEvent* event) -> void {
|
|
//Keyboard::Keycode sym = Keysym(event->nativeVirtualKey());
|
|
//if(sym != Keyboard::Keycode::None && self.window.onKeyPress) self.window.onKeyPress(sym);
|
|
|
|
if(p.state().dismissable && event->key() == Qt::Key_Escape) {
|
|
if(p.state().onClose) p.self().doClose();
|
|
else p.self().setVisible(false);
|
|
if(p.state().modal && !p.self().visible()) p.self().setModal(false);
|
|
}
|
|
}
|
|
|
|
auto QtWindow::keyReleaseEvent(QKeyEvent* event) -> void {
|
|
//Keyboard::Keycode sym = Keysym(event->nativeVirtualKey());
|
|
//if(sym != Keyboard::Keycode::None && self.window.onKeyRelease) self.window.onKeyRelease(sym);
|
|
}
|
|
|
|
auto QtWindow::resizeEvent(QResizeEvent* event) -> void {
|
|
if(!p.locked() && !p.state().fullScreen && p.qtWindow->isVisible()) {
|
|
p.state().geometry.setSize({
|
|
p.qtContainer->geometry().width(),
|
|
p.qtContainer->geometry().height()
|
|
});
|
|
}
|
|
|
|
if(auto& sizable = p.state().sizable) {
|
|
sizable->setGeometry(p.self().geometry().setPosition());
|
|
}
|
|
|
|
if(!p.locked()) {
|
|
p.self().doSize();
|
|
}
|
|
}
|
|
|
|
auto QtWindow::sizeHint() const -> QSize {
|
|
uint width = p.state().geometry.width();
|
|
uint height = p.state().geometry.height();
|
|
height += p._menuHeight();
|
|
height += p._statusHeight();
|
|
return QSize(width, height);
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|