/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "graphics/font.h" #include "graphics/primitives.h" #include "common/events.h" #include "graphics/macgui/macfontmanager.h" #include "graphics/macgui/macwindowmanager.h" #include "graphics/macgui/macwindow.h" #include "graphics/macgui/macwidget.h" #include "image/bmp.h" namespace Graphics { BaseMacWindow::BaseMacWindow(int id, bool editable, MacWindowManager *wm) : MacWidget(nullptr, 0, 0, 0, 0, wm, true), _id(id), _editable(editable) { _callback = 0; _dataPtr = 0; _contentIsDirty = true; _type = kWindowUnknown; _visible = true; } void BaseMacWindow::setVisible(bool visible, bool silent) { _visible = visible; _wm->setFullRefresh(true); } bool BaseMacWindow::isVisible() { return _visible; } MacWindow::MacWindow(int id, bool scrollable, bool resizable, bool editable, MacWindowManager *wm) : BaseMacWindow(id, editable, wm), _scrollable(scrollable), _resizable(resizable) { _borderIsDirty = true; _pattern = 0; _hasPattern = false; _highlightedPart = kBorderNone; _beingDragged = false; _beingResized = false; _draggedX = _draggedY = 0; _type = kWindowWindow; _closeable = false; _borderType = -1; _borderWidth = kBorderWidth; _macBorder.setWindow(this); _hasScrollBar = false; _mode = 0; } void MacWindow::disableBorder() { _macBorder.disableBorder(); } const Font *MacWindow::getTitleFont() { return _wm->_fontMan->getFont(Graphics::MacFont(kMacFontChicago, 12)); } void MacWindow::setActive(bool active) { MacWidget::setActive(active); _borderIsDirty = true; } bool MacWindow::isActive() { return _active; } void MacWindow::resize(int w, int h, bool inner) { if (_composeSurface->w == w && _composeSurface->h == h) return; if (inner) { _innerDims.setWidth(w); _innerDims.setHeight(h); updateOuterDims(); } else { _dims.setWidth(w); _dims.setHeight(h); updateInnerDims(); } _composeSurface->free(); _composeSurface->create(_innerDims.width(), _innerDims.height(), _wm->_pixelformat); if (_hasPattern) drawPattern(); _borderSurface.free(); _borderSurface.create(_dims.width(), _dims.height(), _wm->_pixelformat); _contentIsDirty = true; _borderIsDirty = true; _wm->setFullRefresh(true); } void MacWindow::move(int x, int y) { if (_dims.left == x && _dims.top == y) return; _dims.moveTo(x, y); updateInnerDims(); _contentIsDirty = true; _wm->setFullRefresh(true); } void MacWindow::setDimensions(const Common::Rect &r) { resize(r.width(), r.height()); _dims.moveTo(r.left, r.top); updateInnerDims(); _contentIsDirty = true; _wm->setFullRefresh(true); } void MacWindow::setBackgroundPattern(int pattern) { _pattern = pattern; _hasPattern = true; drawPattern(); _contentIsDirty = true; } bool MacWindow::draw(bool forceRedraw) { if (!_borderIsDirty && !_contentIsDirty && !forceRedraw) return false; if (_borderIsDirty || forceRedraw) drawBorder(); _contentIsDirty = false; return true; } bool MacWindow::draw(ManagedSurface *g, bool forceRedraw) { if (!draw(forceRedraw)) return false; g->blitFrom(*_composeSurface, Common::Rect(0, 0, _composeSurface->w, _composeSurface->h), Common::Point(_innerDims.left, _innerDims.top)); uint32 transcolor = (_wm->_pixelformat.bytesPerPixel == 1) ? _wm->_colorGreen : 0; g->transBlitFrom(_borderSurface, Common::Rect(0, 0, _borderSurface.w, _borderSurface.h), Common::Point(_dims.left, _dims.top), transcolor); return true; } void MacWindow::blit(ManagedSurface *g, Common::Rect &dest) { // Only the inner surface is blitted here uint32 transcolor = (_wm->_pixelformat.bytesPerPixel == 1) ? _wm->_colorGreen2 : 0; g->transBlitFrom(*_composeSurface, _composeSurface->getBounds(), dest, transcolor); } uint32 MacWindow::getBorderFlags() { uint32 flags = 0; if (_active) flags |= kWindowBorderActive; if (!_title.empty()) flags |= kWindowBorderTitle; if (_hasScrollBar) flags |= kWindowBorderScrollbar; return flags; } void MacWindow::center(bool toCenter) { if (!_wm) return; Common::Rect screen = _wm->getScreenBounds(); uint32 flags = getBorderFlags(); if (toCenter) { move((screen.width() - _dims.width()) / 2, (screen.height() - _dims.height()) / 2); } else if (_macBorder.hasBorder(flags) && _macBorder.hasOffsets()) { move(_macBorder.getOffset().left, _macBorder.getOffset().top); } else { move(0, 0); } } void MacWindow::updateInnerDims() { if (_dims.isEmpty()) return; uint32 flags = getBorderFlags(); if (_macBorder.hasBorder(flags) && _macBorder.hasOffsets()) { _innerDims = Common::Rect( _dims.left + _macBorder.getOffset().left, _dims.top + _macBorder.getOffset().top, _dims.right - _macBorder.getOffset().right, _dims.bottom - _macBorder.getOffset().bottom); } else { _innerDims = _dims; _innerDims.grow(-kBorderWidth); } } void MacWindow::updateOuterDims() { if (_innerDims.isEmpty()) return; uint32 flags = getBorderFlags(); if (_macBorder.hasBorder(flags) && _macBorder.hasOffsets()) { _dims = Common::Rect( _innerDims.left - _macBorder.getOffset().left, _innerDims.top - _macBorder.getOffset().top, _innerDims.right + _macBorder.getOffset().right, _innerDims.bottom + _macBorder.getOffset().bottom); } else { _dims = _innerDims; _dims.grow(kBorderWidth); } } void MacWindow::drawBorder() { _borderIsDirty = false; ManagedSurface *g = &_borderSurface; uint32 flags = getBorderFlags(); if (_macBorder.hasBorder(flags)) { drawBorderFromSurface(g, flags); } else { warning("MacWindow: No Border Loaded"); setBorderType(0xff); return; } if (_highlightedPart == kBorderScrollUp || _highlightedPart == kBorderScrollDown) setHighlight(kBorderNone); } void MacWindow::drawBorderFromSurface(ManagedSurface *g, uint32 flags) { if (_wm->_pixelformat.bytesPerPixel == 1) { g->clear(_wm->_colorGreen); } _macBorder.blitBorderInto(*g, flags, _wm); } void MacWindow::setTitle(const Common::String &title) { _title = title; _borderIsDirty = true; _macBorder.setTitle(title, _borderSurface.w, _wm); } void MacWindow::drawPattern() { byte *pat = _wm->getPatterns()[_pattern - 1]; for (int y = 0; y < _composeSurface->h; y++) { for (int x = 0; x < _composeSurface->w; x++) { if (_wm->_pixelformat.bytesPerPixel == 1) { byte *dst = (byte *)_composeSurface->getBasePtr(x, y); if (pat[y % 8] & (1 << (7 - (x % 8)))) *dst = _wm->_colorBlack; else *dst = _wm->_colorWhite; } else { uint32 *dst = (uint32 *)_composeSurface->getBasePtr(x, y); if (pat[y % 8] & (1 << (7 - (x % 8)))) *dst = _wm->_colorBlack; else *dst = _wm->_colorWhite; } } } } void MacWindow::setHighlight(WindowClick highlightedPart) { if (_highlightedPart == highlightedPart) return; _highlightedPart = highlightedPart; _borderIsDirty = true; } void MacWindow::setScroll(float scrollPos, float scrollSize) { _macBorder.setScroll(scrollPos, scrollSize); _borderIsDirty = true; } void MacWindow::loadBorder(Common::SeekableReadStream &file, uint32 flags, int lo, int ro, int to, int bo) { _macBorder.loadBorder(file, flags, lo, ro, to, bo); } void MacWindow::loadBorder(Common::SeekableReadStream &file, uint32 flags, BorderOffsets offsets) { _macBorder.loadBorder(file, flags, offsets); } void MacWindow::setBorder(Graphics::TransparentSurface *surface, uint32 flags, BorderOffsets offsets) { _macBorder.setBorder(surface, flags, offsets); } void MacWindow::resizeBorderSurface() { updateOuterDims(); _borderSurface.free(); _borderSurface.create(_dims.width(), _dims.height(), _wm->_pixelformat); } void MacWindow::setCloseable(bool closeable) { _closeable = closeable; } void MacWindow::drawBox(ManagedSurface *g, int x, int y, int w, int h) { Common::Rect r(x, y, x + w + 1, y + h + 1); g->fillRect(r, _wm->_colorWhite); g->frameRect(r, _wm->_colorBlack); } void MacWindow::fillRect(ManagedSurface *g, int x, int y, int w, int h, int color) { Common::Rect r(x, y, x + w, y + h); g->fillRect(r, color); } WindowClick MacWindow::isInBorder(int x, int y) { if (_innerDims.contains(x, y)) return kBorderInner; if (isInCloseButton(x, y)) return kBorderCloseButton; if (_resizable) if (isInResizeButton(x, y)) return kBorderResizeButton; if (_scrollable) return isInScroll(x, y); return kBorderBorder; } bool MacWindow::isInCloseButton(int x, int y) { int bLeft = kBorderWidth; int bTop = kBorderWidth; if (_macBorder.hasOffsets()) { bLeft = _macBorder.getOffset().left; bTop = _macBorder.getOffset().top; } return (x >= _innerDims.left - bLeft && x < _innerDims.left && y >= _innerDims.top - bTop && y < _innerDims.top); } bool MacWindow::isInResizeButton(int x, int y) { int bRight = kBorderWidth; int bBottom = kBorderWidth; if (_macBorder.hasOffsets()) { bRight = _macBorder.getOffset().right; bBottom = _macBorder.getOffset().bottom; } return (x >= _innerDims.right && x < _innerDims.right + bRight && y >= _innerDims.bottom && y < _innerDims.bottom + bBottom); } WindowClick MacWindow::isInScroll(int x, int y) { int bTop = kBorderWidth; int bRight = kBorderWidth; int bBottom = kBorderWidth; if (_macBorder.hasOffsets()) { bTop = _macBorder.getOffset().top; bRight = _macBorder.getOffset().right; bBottom = _macBorder.getOffset().bottom; } if (x >= _innerDims.right && x < _innerDims.right + bRight) { if (y < _innerDims.top - bTop) return kBorderBorder; if (y >= _innerDims.bottom + bBottom) return kBorderBorder; if (y >= _innerDims.top + _innerDims.height() / 2) return kBorderScrollDown; return kBorderScrollUp; } if (y >= _innerDims.bottom && y < _innerDims.bottom + bBottom) { if (x < _innerDims.left - bTop) return kBorderBorder; if (x >= _innerDims.right + bRight) return kBorderBorder; if (x >= _innerDims.left + _innerDims.width() / 2) return kBorderScrollRight; return kBorderScrollLeft; } return kBorderBorder; } bool MacWindow::processEvent(Common::Event &event) { WindowClick click = isInBorder(event.mouse.x, event.mouse.y); switch (event.type) { case Common::EVENT_MOUSEMOVE: if (_wm->_mouseDown && _wm->_hoveredWidget && !_wm->_hoveredWidget->_dims.contains(event.mouse.x - _dims.left, event.mouse.y - _dims.top)) { _wm->_hoveredWidget->setActive(false); // since we de-active the hoveredWidget, so we need to check whether it's the activeWidget of wm if (_wm->getActiveWidget() == _wm->_hoveredWidget) _wm->setActiveWidget(nullptr); _wm->_hoveredWidget = nullptr; } if (_beingDragged) { _dims.translate(event.mouse.x - _draggedX, event.mouse.y - _draggedY); updateInnerDims(); _draggedX = event.mouse.x; _draggedY = event.mouse.y; _wm->setFullRefresh(true); } if (_beingResized) { resize(MAX(_borderWidth * 4, _dims.width() + event.mouse.x - _draggedX), MAX(_borderWidth * 4, _dims.height() + event.mouse.y - _draggedY)); _draggedX = event.mouse.x; _draggedY = event.mouse.y; _wm->setFullRefresh(true); if (_callback) (*_callback)(click, event, _dataPtr); } break; case Common::EVENT_LBUTTONDOWN: setHighlight(click); if (click == kBorderBorder) { _beingDragged = true; _draggedX = event.mouse.x; _draggedY = event.mouse.y; } if (click == kBorderResizeButton) { _beingResized = true; _draggedX = event.mouse.x; _draggedY = event.mouse.y; } if (click == kBorderCloseButton && _closeable) { _wm->removeWindow(this); } break; case Common::EVENT_LBUTTONUP: _beingDragged = false; _beingResized = false; setHighlight(kBorderNone); break; case Common::EVENT_KEYDOWN: if (!_editable && !(_wm->getActiveWidget() && _wm->getActiveWidget()->isEditable())) return false; if (_wm->getActiveWidget()) return _wm->getActiveWidget()->processEvent(event); return false; case Common::EVENT_WHEELUP: case Common::EVENT_WHEELDOWN: if (_wm->getActiveWidget() && _wm->getActiveWidget()->processEvent(event)) return true; return false; default: return false; } event.mouse.x -= _dims.left; event.mouse.y -= _dims.top; MacWidget *w = findEventHandler(event, -_dims.left + _innerDims.left, -_dims.top + _innerDims.top); if (w && w != this) { _wm->_hoveredWidget = w; if (w->processEvent(event)) return true; } if (_callback) return (*_callback)(click, event, _dataPtr); else return false; } void MacWindow::setBorderType(int borderType) { _borderType = borderType; if (borderType < 0) { _macBorder.disableBorder(); } else { _macBorder.setBorderType(borderType); } } void MacWindow::loadWin95Border(const Common::String &filename, uint32 flags) { Common::SeekableReadStream *stream = _wm->getFile(filename); if (stream) { Graphics::BorderOffsets offsets; offsets.top = 1; offsets.bottom = 1; offsets.left = 1; offsets.right = 17; offsets.lowerScrollHeight = 15; offsets.upperScrollHeight = 17; offsets.titlePos = 0; offsets.titleTop = 0; loadBorder(*stream, flags, offsets); delete stream; } } void MacWindow::addDirtyRect(const Common::Rect &r) { if (!r.isValidRect()) return; Common::Rect bounds = r; bounds.clip(Common::Rect(_innerDims.width(), _innerDims.height())); if (bounds.width() > 0 && bounds.height() > 0) _dirtyRects.push_back(bounds); } void MacWindow::markAllDirty() { _dirtyRects.clear(); _dirtyRects.push_back(Common::Rect(_composeSurface->w, _composeSurface->h)); } void MacWindow::mergeDirtyRects() { Common::List::iterator rOuter, rInner; // Process the dirty rect list to find any rects to merge for (rOuter = _dirtyRects.begin(); rOuter != _dirtyRects.end(); ++rOuter) { rInner = rOuter; while (++rInner != _dirtyRects.end()) { if ((*rOuter).intersects(*rInner)) { // These two rectangles overlap, so merge them rOuter->extend(*rInner); // remove the inner rect from the list _dirtyRects.erase(rInner); // move back to beginning of list rInner = rOuter; } } } } } // End of namespace Graphics