scummvm/graphics/macgui/mactextwindow.cpp

669 lines
17 KiB
C++

/* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/timer.h"
#include "common/system.h"
#include "graphics/macgui/macwindowmanager.h"
#include "graphics/macgui/macfontmanager.h"
#include "graphics/macgui/mactextwindow.h"
#include "graphics/macgui/macmenu.h"
namespace Graphics {
enum {
kConWOverlap = 20,
kConHOverlap = 20,
kConWPadding = 3,
kConHPadding = 4,
kConOverscan = 3,
kConScrollStep = 12,
kCursorHeight = 12
};
static void cursorTimerHandler(void *refCon);
MacTextWindow::MacTextWindow(MacWindowManager *wm, const MacFont *font, int fgcolor, int bgcolor, int maxWidth, TextAlign textAlignment, MacMenu *menu, bool cursorHandler) :
MacWindow(wm->getLastId(), true, true, true, wm), _bgcolor(bgcolor), _maxWidth(maxWidth), _menu(menu) {
_font = font;
_mactext = new MacText(this, 0, 0, 0, 0, _wm, Common::U32String(""), font, fgcolor, bgcolor, maxWidth, textAlignment);
_fontRef = wm->_fontMan->getFont(*font);
init(cursorHandler);
}
MacTextWindow::MacTextWindow(MacWindowManager *wm, const Font *font, int fgcolor, int bgcolor, int maxWidth, TextAlign textAlignment, MacMenu *menu, bool cursorHandler) :
MacWindow(wm->getLastId(), true, true, true, wm), _bgcolor(bgcolor), _maxWidth(maxWidth), _menu(menu) {
_font = nullptr;
_mactext = new MacText(Common::U32String(""), _wm, font, fgcolor, bgcolor, maxWidth, textAlignment);
_fontRef = font;
init(cursorHandler);
}
void MacTextWindow::init(bool cursorHandler) {
_inputTextHeight = 0;
_inputIsDirty = true;
_inTextSelection = false;
_scrollPos = 0;
_editable = true;
_selectable = true;
_cursorX = 0;
_cursorY = 0;
_cursorState = false;
_cursorOff = false;
_cursorDirty = true;
_cursorRect = new Common::Rect(0, 0, 1, kCursorHeight);
_cursorSurface = new ManagedSurface(1, kCursorHeight);
_cursorSurface->fillRect(*_cursorRect, _wm->_colorBlack);
_textColorRGB = 0;
if (cursorHandler)
g_system->getTimerManager()->installTimerProc(&cursorTimerHandler, 200000, this, "textWindowCursor");
if (_wm->_mode & kWMModeWin95) {
// in win95 mode, we set scrollbar as default
_hasScrollBar = true;
setBorderType(kWinBorderWin95Scrollbar);
loadInternalBorder(kWindowBorderScrollbar | kWindowBorderActive);
}
}
void MacTextWindow::resize(int w, int h) {
if (_composeSurface->w == w && _composeSurface->h == h)
return;
undrawInput();
MacWindow::resize(w, h);
_maxWidth = getInnerDimensions().width();
_mactext->resize(_maxWidth, h);
}
void MacTextWindow::setDimensions(const Common::Rect &r) {
resize(r.width(), r.height());
_dims.moveTo(r.left, r.top);
updateInnerDims();
_contentIsDirty = true;
_wm->setFullRefresh(true);
_mactext->setDimensions(Common::Rect(_innerDims.width(), _innerDims.height()));
}
void MacTextWindow::appendText(const Common::U32String &str, const MacFont *macFont, bool skipAdd) {
// the reason we put undrawInput here before appendText, is we don't want the appended text affect our input
// thus, we first delete all of out input, and we append new text, and we redraw the input
undrawInput();
// we calc the rgb though fgcolor
uint16 red = (_textColorRGB >> 16) & 0xFF;
uint16 green = (_textColorRGB >> 8) & 0xFF;
uint16 blue = (_textColorRGB) & 0xFF;
if (macFont)
_mactext->appendText(str, macFont->getId(), macFont->getSize(), macFont->getSlant(), red, green, blue, skipAdd);
else {
_mactext->appendText(str, _fontRef, red, green, blue, skipAdd);
}
_contentIsDirty = true;
_inputIsDirty = true; //force it to redraw input
if (_editable) {
_scrollPos = MAX<int>(0, _mactext->getTextHeight() - getInnerDimensions().height());
updateCursorPos();
}
if (_wm->_mode & kWMModeWin95)
calcScrollBar();
// if we enable the dynamic scrollbar, and the text height is smaller than window height, then we disable the border
// if the window is editable, then we don't disable the border, because in editable window, the area you scroll is always bigger
// than the window
if (!_editable && (_mode & kWindowModeDynamicScrollbar) && _mactext->getTextHeight() < getInnerDimensions().height()) {
int w = getDimensions().width();
int h = getDimensions().height();
enableScrollbar(false);
disableBorder();
resize(w, h);
_mactext->_fullRefresh = true;
}
// if the text height is bigger than the window, then we enable the scrollbar again
if (!_editable && (_mode & kWindowModeDynamicScrollbar) && _mactext->getTextHeight() > getInnerDimensions().height()) {
enableScrollbar(true);
int w = getDimensions().width();
int h = getDimensions().height();
resize(w, h);
_mactext->_fullRefresh = true;
}
}
void MacTextWindow::appendText(const Common::String &str, const MacFont *macFont, bool skipAdd) {
appendText(Common::U32String(str), macFont, skipAdd);
}
void MacTextWindow::setMarkdownText(const Common::U32String &str) {
_mactext->setMarkdownText(str);
_contentIsDirty = true;
}
void MacTextWindow::clearText() {
_mactext->clearText();
_contentIsDirty = true;
_borderIsDirty = true;
updateCursorPos();
}
MacTextWindow::~MacTextWindow() {
delete _cursorRect;
delete _cursorSurface;
delete _mactext;
g_system->getTimerManager()->removeTimerProc(&cursorTimerHandler);
}
void MacTextWindow::setTextWindowFont(const MacFont *font) {
_font = font;
_fontRef = _wm->_fontMan->getFont(*font);
_mactext->setDefaultFormatting(font->getId(), font->getSlant(), font->getSize(), 0, 0, 0);
}
const MacFont *MacTextWindow::getTextWindowFont() {
return _font;
}
bool MacTextWindow::draw(bool forceRedraw) {
if (!_borderIsDirty && !_contentIsDirty && !_cursorDirty && !_inputIsDirty && !forceRedraw)
return false;
if (_borderIsDirty || forceRedraw) {
drawBorder();
if (_wm->_mode & kWMModeWin95) {
_composeSurface->clear(_bgcolor);
} else {
_composeSurface->clear(_wm->_colorWhite);
}
}
if (_inputIsDirty || forceRedraw) {
drawInput();
_inputIsDirty = false;
}
_contentIsDirty = false;
_cursorDirty = false;
// Compose
_mactext->draw(_composeSurface, true);
if (_cursorState)
_composeSurface->blitFrom(*_cursorSurface, *_cursorRect, Common::Point(_cursorX, _cursorY));
return true;
}
bool MacTextWindow::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 MacTextWindow::blit(ManagedSurface *g, Common::Rect &dest) {
g->transBlitFrom(*_composeSurface, _composeSurface->getBounds(), dest, _wm->_colorGreen2);
}
void MacTextWindow::drawSelection() {
if (_selectedText.endY == -1)
return;
SelectedText s = _selectedText;
if (s.startY > s.endY || (s.startY == s.endY && s.startX > s.endX)) {
SWAP(s.startX, s.endX);
SWAP(s.startY, s.endY);
SWAP(s.startRow, s.endRow);
SWAP(s.startCol, s.endCol);
}
int lastLineStart = s.endY;
s.endY += _mactext->getLineHeight(s.endRow);
int start = s.startY - _scrollPos;
start = MAX(0, start);
if (start > getInnerDimensions().height())
return;
int end = s.endY - _scrollPos;
if (end < 0)
return;
end = MIN((int)getInnerDimensions().height(), end);
int numLines = 0;
int x1 = 0, x2 = 0;
for (int y = start; y < end; y++) {
if (!numLines) {
x1 = 0;
x2 = getInnerDimensions().width() - 1;
if (y + _scrollPos == s.startY && s.startX > 0) {
numLines = _mactext->getLineHeight(s.startRow);
x1 = s.startX;
}
if (y + _scrollPos >= lastLineStart) {
numLines = _mactext->getLineHeight(s.endRow);
x2 = s.endX;
}
} else {
numLines--;
}
byte *ptr = (byte *)_composeSurface->getBasePtr(x1 + 1, y + 1);
for (int x = x1; x < x2; x++, ptr++)
if (*ptr == _wm->_colorBlack)
*ptr = _wm->_colorWhite;
else
*ptr = _wm->_colorBlack;
}
}
Common::U32String MacTextWindow::getSelection(bool formatted, bool newlines) {
if (_selectedText.endY == -1)
return Common::U32String();
SelectedText s = _selectedText;
if (s.startY > s.endY || (s.startY == s.endY && s.startX > s.endX)) {
SWAP(s.startRow, s.endRow);
SWAP(s.startCol, s.endCol);
}
return _mactext->getTextChunk(s.startRow, s.startCol, s.endRow, s.endCol, formatted, newlines);
}
void MacTextWindow::clearSelection() {
_selectedText.endY = _selectedText.startY = -1;
}
bool MacTextWindow::isCutAllowed() {
if (_selectedText.startRow >= (int)(_mactext->getLineCount() - _inputTextHeight) &&
_selectedText.endRow >= (int)(_mactext->getLineCount() - _inputTextHeight))
return true;
return false;
}
Common::U32String MacTextWindow::cutSelection() {
if (!isCutAllowed())
return Common::U32String();
SelectedText s = _selectedText;
if (s.startY > s.endY || (s.startY == s.endY && s.startX > s.endX)) {
SWAP(s.startRow, s.endRow);
SWAP(s.startCol, s.endCol);
}
Common::U32String selection = _mactext->getTextChunk(s.startRow, s.startCol, s.endRow, s.endCol, false, false);
uint32 selPos = _inputText.find(selection);
if (selPos == Common::U32String::npos) {
//warning("Cannot find substring '%s' in '%s'", selection.c_str(), _inputText.c_str()); // Needed encode method
return Common::U32String();
}
Common::U32String newInput = _inputText.substr(0, selPos) + _inputText.substr(selPos + selection.size());
clearSelection();
clearInput();
appendInput(newInput);
return selection;
}
int MacTextWindow::getMouseLine(int x, int y) {
// TODO: Improve the algorithm here, since after long scrolling there is
// sometimes error of +2 rows
x -= getInnerDimensions().left;
y -= getInnerDimensions().top + kConScrollStep;
return _mactext->getMouseLine(x, y);
}
void MacTextWindow::calcScrollBar() {
// since this function only able for the window which has scroll bar
// thus, if it doesn't has scrollbar, then we don't have to calc it
if (!_hasScrollBar)
return;
int maxText = 0, maxScrollbar = 0, displayHeight = 0;
displayHeight = getInnerDimensions().height();
maxScrollbar = getDimensions().height() - getBorderOffsets().upperScrollHeight - getBorderOffsets().lowerScrollHeight;
// if we enable the win95 mode but the text height is smaller than window height, then we don't draw the scrollbar
if (_wm->_mode & kWMModeWin95 && displayHeight > _mactext->getTextHeight() && !_editable)
return;
if (_editable)
maxText = _mactext->getTextHeight() + getInnerDimensions().height();
else
maxText = MAX<int>(_mactext->getTextHeight(), displayHeight);
float scrollSize = (float)maxScrollbar * (float)displayHeight / (float)maxText;
float scrollPos = (float)_scrollPos * (float)maxScrollbar / (float)maxText;
setScroll(scrollPos, scrollSize);
}
bool MacTextWindow::processEvent(Common::Event &event) {
WindowClick click = isInBorder(event.mouse.x, event.mouse.y);
if (event.type == Common::EVENT_KEYDOWN) {
if (!_editable)
return false;
_wm->setActiveWindow(getId());
if (event.kbd.flags & (Common::KBD_ALT | Common::KBD_CTRL | Common::KBD_META)) {
switch (event.kbd.keycode) {
case Common::KEYCODE_x:
_wm->setTextInClipboard(cutSelection());
return true;
case Common::KEYCODE_c:
_wm->setTextInClipboard(getSelection(true, false));
return true;
case Common::KEYCODE_v:
if (g_system->hasTextInClipboard()) {
if (_selectedText.endY != -1)
cutSelection();
appendInput(_wm->getTextFromClipboard());
_inputIsDirty = true;
}
return true;
default:
break;
}
return false;
}
switch (event.kbd.keycode) {
case Common::KEYCODE_BACKSPACE:
if (!_inputText.empty()) {
if (_selectedText.endY != -1) {
cutSelection();
} else {
_inputText.deleteLastChar();
}
_inputIsDirty = true;
}
return true;
case Common::KEYCODE_RETURN:
undrawInput();
_inputIsDirty = true; // we force it to redraw input
return false; // Pass it to the higher level for processing
default:
if (event.kbd.ascii == '~')
return false;
if (event.kbd.ascii >= 0x20 && event.kbd.ascii <= 0x7f) {
if (_selectedText.endY != -1)
cutSelection();
_inputText += (char)event.kbd.ascii;
_inputIsDirty = true;
return true;
}
break;
}
}
if (hasAllFocus())
return MacWindow::processEvent(event); // Pass it to upstream
if (event.type == Common::EVENT_WHEELUP) {
setHighlight(kBorderScrollUp);
scroll(-2);
calcScrollBar();
return true;
}
if (event.type == Common::EVENT_WHEELDOWN) {
setHighlight(kBorderScrollDown);
scroll(2);
calcScrollBar();
return true;
}
if (click == kBorderScrollUp || click == kBorderScrollDown) {
if (event.type == Common::EVENT_LBUTTONDOWN) {
setHighlight(click);
calcScrollBar();
return true;
} else if (event.type == Common::EVENT_LBUTTONUP) {
switch (click) {
case kBorderScrollUp:
scroll(-1);
break;
case kBorderScrollDown:
scroll(1);
break;
default:
return false;
}
return true;
}
return false;
}
if (click == kBorderInner) {
// Call callback for processing any events
if (_callback)
(*_callback)(click, event, _dataPtr);
if (!_selectable)
return false;
if (event.type == Common::EVENT_LBUTTONDOWN) {
startMarking(event.mouse.x, event.mouse.y);
return true;
} else if (event.type == Common::EVENT_LBUTTONUP && _menu) {
if (_inTextSelection) {
_inTextSelection = false;
if (_selectedText.endY == -1 ||
(_selectedText.endX == _selectedText.startX && _selectedText.endY == _selectedText.startY)) {
_selectedText.startY = _selectedText.endY = -1;
_contentIsDirty = true;
_menu->enableCommand("Edit", "Copy", false);
} else {
_menu->enableCommand("Edit", "Copy", true);
bool cutAllowed = isCutAllowed();
_menu->enableCommand("Edit", "Cut", cutAllowed);
_menu->enableCommand("Edit", "Clear", cutAllowed);
}
}
return true;
} else if (event.type == Common::EVENT_MOUSEMOVE) {
if (_inTextSelection) {
updateTextSelection(event.mouse.x, event.mouse.y);
return true;
}
}
return false;
}
return MacWindow::processEvent(event);
}
void MacTextWindow::scroll(int delta) {
int oldScrollPos = _scrollPos;
_scrollPos += delta * kConScrollStep;
if (_editable)
_scrollPos = CLIP<int>(_scrollPos, 0, _mactext->getTextHeight() - kConScrollStep);
else
_scrollPos = CLIP<int>(_scrollPos, 0, MAX<int>(0, _mactext->getTextHeight() - getInnerDimensions().height()));
undrawCursor();
_cursorY -= (_scrollPos - oldScrollPos);
_contentIsDirty = true;
_borderIsDirty = true;
_mactext->scroll(delta);
}
void MacTextWindow::startMarking(int x, int y) {
x -= getInnerDimensions().left - 2;
y -= getInnerDimensions().top;
y += _scrollPos;
_mactext->getRowCol(x, y, &_selectedText.startX, &_selectedText.startY, &_selectedText.startRow, &_selectedText.startCol);
_selectedText.endY = -1;
_inTextSelection = true;
}
void MacTextWindow::updateTextSelection(int x, int y) {
x -= getInnerDimensions().left - 2;
y -= getInnerDimensions().top;
y += _scrollPos;
_mactext->getRowCol(x, y, &_selectedText.endX, &_selectedText.endY, &_selectedText.endRow, &_selectedText.endCol);
debug(3, "s: %d,%d (%d, %d) e: %d,%d (%d, %d)", _selectedText.startX, _selectedText.startY,
_selectedText.startRow, _selectedText.startCol, _selectedText.endX,
_selectedText.endY, _selectedText.endRow, _selectedText.endCol);
_contentIsDirty = true;
}
void MacTextWindow::undrawInput() {
for (uint i = 0; i < _inputTextHeight; i++)
_mactext->removeLastLine();
_inputTextHeight = 0;
}
void MacTextWindow::drawInput() {
if (!_editable)
return;
int oldLen = _mactext->getLineCount() - _inputTextHeight;
// add new input line to the text
appendText(_inputText, _font, true);
// Now recalc new text height
int newLen = _mactext->getLineCount();
_inputTextHeight = newLen - oldLen;
_cursorX = _inputText.empty() ? 0 : _mactext->getLastLineWidth();
updateCursorPos();
_contentIsDirty = true;
}
void MacTextWindow::clearInput() {
undrawCursor();
_cursorX = 0;
_inputText.clear();
}
void MacTextWindow::appendInput(const Common::U32String &str) {
_inputText += str;
drawInput();
}
void MacTextWindow::appendInput(const Common::String &str) {
appendInput(Common::U32String(str));
}
//////////////////
// Cursor stuff
static void cursorTimerHandler(void *refCon) {
MacTextWindow *w = (MacTextWindow *)refCon;
if (!w->_cursorOff)
w->_cursorState = !w->_cursorState;
w->_cursorDirty = true;
}
void MacTextWindow::updateCursorPos() {
_cursorY = _mactext->getTextHeight() - _scrollPos - kCursorHeight;
_cursorY += _inputText.empty() ? 3 : 0;
_cursorDirty = true;
}
void MacTextWindow::undrawCursor() {
_cursorState = false;
_cursorDirty = true;
}
} // End of namespace Graphics