mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-10 03:40:25 +00:00
1011 lines
31 KiB
C++
1011 lines
31 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 "gui/predictivedialog.h"
|
|
#include "gui/widget.h"
|
|
#include "gui/widgets/edittext.h"
|
|
#include "gui/gui-manager.h"
|
|
#include "gui/ThemeEval.h"
|
|
|
|
#include "common/config-manager.h"
|
|
#include "common/translation.h"
|
|
#include "common/events.h"
|
|
#include "common/debug.h"
|
|
#include "common/system.h"
|
|
#include "common/keyboard.h"
|
|
#include "common/file.h"
|
|
#include "common/savefile.h"
|
|
|
|
namespace GUI {
|
|
|
|
enum {
|
|
kCancelCmd = 'CNCL',
|
|
kOkCmd = '__OK',
|
|
kBut1Cmd = 'BUT1',
|
|
kBut2Cmd = 'BUT2',
|
|
kBut3Cmd = 'BUT3',
|
|
kBut4Cmd = 'BUT4',
|
|
kBut5Cmd = 'BUT5',
|
|
kBut6Cmd = 'BUT6',
|
|
kBut7Cmd = 'BUT7',
|
|
kBut8Cmd = 'BUT8',
|
|
kBut9Cmd = 'BUT9',
|
|
kBut0Cmd = 'BUT0',
|
|
kNextCmd = 'NEXT',
|
|
kAddCmd = '_ADD',
|
|
kModeCmd = 'MODE',
|
|
kDelCmd = '_DEL',
|
|
kTestCmd = 'TEST'
|
|
};
|
|
|
|
enum {
|
|
kModePre = 0,
|
|
kModeNum = 1,
|
|
kModeAbc = 2
|
|
};
|
|
|
|
PredictiveDialog::PredictiveDialog() : Dialog("Predictive") {
|
|
new StaticTextWidget(this, "Predictive.Headline", _("Enter Text"));
|
|
|
|
_button[kCancelAct] = new ButtonWidget(this, "Predictive.Cancel", _("Cancel") , Common::U32String(), kCancelCmd);
|
|
_button[kOkAct] = new ButtonWidget(this, "Predictive.OK", _("OK") , Common::U32String(), kOkCmd);
|
|
|
|
if (g_gui.useRTL()) {
|
|
/** If using RTL, swap the internal name of odd columns, to be flipped again when drawing.
|
|
We flip them back to original, because the keyboard layout stays the same in LTR & RTL.
|
|
The rest, like okButton, cancel, etc are all flipped.
|
|
*/
|
|
|
|
_button[kButton3Act] = new ButtonWidget(this, "Predictive.Button1", Common::U32String("3 def" ), Common::U32String(), kBut3Cmd);
|
|
_button[kButton2Act] = new ButtonWidget(this, "Predictive.Button2", Common::U32String("2 abc" ), Common::U32String(), kBut2Cmd);
|
|
_button[kButton1Act] = new ButtonWidget(this, "Predictive.Button3", Common::U32String("1 `-.&" ), Common::U32String(), kBut1Cmd);
|
|
_button[kButton6Act] = new ButtonWidget(this, "Predictive.Button4", Common::U32String("6 mno" ), Common::U32String(), kBut6Cmd);
|
|
_button[kButton5Act] = new ButtonWidget(this, "Predictive.Button5", Common::U32String("5 jkl" ), Common::U32String(), kBut5Cmd);
|
|
_button[kButton4Act] = new ButtonWidget(this, "Predictive.Button6", Common::U32String("4 ghi" ), Common::U32String(), kBut4Cmd);
|
|
_button[kButton9Act] = new ButtonWidget(this, "Predictive.Button7", Common::U32String("9 wxyz" ), Common::U32String(), kBut9Cmd);
|
|
_button[kButton8Act] = new ButtonWidget(this, "Predictive.Button8", Common::U32String("8 tuv" ), Common::U32String(), kBut8Cmd);
|
|
_button[kButton7Act] = new ButtonWidget(this, "Predictive.Button9", Common::U32String("7 pqrs" ), Common::U32String(), kBut7Cmd);
|
|
_button[kButton0Act] = new ButtonWidget(this, "Predictive.Button0", Common::U32String("0" ), Common::U32String(), kBut0Cmd);
|
|
} else {
|
|
_button[kButton1Act] = new ButtonWidget(this, "Predictive.Button1", Common::U32String("1 `-.&" ), Common::U32String(), kBut1Cmd);
|
|
_button[kButton2Act] = new ButtonWidget(this, "Predictive.Button2", Common::U32String("2 abc" ), Common::U32String(), kBut2Cmd);
|
|
_button[kButton3Act] = new ButtonWidget(this, "Predictive.Button3", Common::U32String("3 def" ), Common::U32String(), kBut3Cmd);
|
|
_button[kButton4Act] = new ButtonWidget(this, "Predictive.Button4", Common::U32String("4 ghi" ), Common::U32String(), kBut4Cmd);
|
|
_button[kButton5Act] = new ButtonWidget(this, "Predictive.Button5", Common::U32String("5 jkl" ), Common::U32String(), kBut5Cmd);
|
|
_button[kButton6Act] = new ButtonWidget(this, "Predictive.Button6", Common::U32String("6 mno" ), Common::U32String(), kBut6Cmd);
|
|
_button[kButton7Act] = new ButtonWidget(this, "Predictive.Button7", Common::U32String("7 pqrs" ), Common::U32String(), kBut7Cmd);
|
|
_button[kButton8Act] = new ButtonWidget(this, "Predictive.Button8", Common::U32String("8 tuv" ), Common::U32String(), kBut8Cmd);
|
|
_button[kButton9Act] = new ButtonWidget(this, "Predictive.Button9", Common::U32String("9 wxyz" ), Common::U32String(), kBut9Cmd);
|
|
_button[kButton0Act] = new ButtonWidget(this, "Predictive.Button0", Common::U32String("0" ), Common::U32String(), kBut0Cmd);
|
|
}
|
|
|
|
// I18N: You must leave "#" as is, only word 'next' is translatable
|
|
_button[kNextAct] = new ButtonWidget(this, "Predictive.Next", _("# next") , Common::U32String(), kNextCmd);
|
|
_button[kAddAct] = new ButtonWidget(this, "Predictive.Add", _("add") , Common::U32String(), kAddCmd);
|
|
_button[kAddAct]->setEnabled(false);
|
|
|
|
#ifndef DISABLE_FANCY_THEMES
|
|
if (g_gui.xmlEval()->getVar("Globals.Predictive.ShowDeletePic") == 1 && g_gui.theme()->supportsImages()) {
|
|
_button[kDelAct] = new PicButtonWidget(this, "Predictive.Delete", _("Delete char"), kDelCmd);
|
|
((PicButtonWidget *)_button[kDelAct])->setGfxFromTheme(ThemeEngine::kImageDelButton);
|
|
} else
|
|
#endif
|
|
_button[kDelAct] = new ButtonWidget(this, "Predictive.Delete" , _("<") , Common::U32String(), kDelCmd);
|
|
// I18N: Pre means 'Predictive', leave '*' as is
|
|
_button[kModeAct] = new ButtonWidget(this, "Predictive.Pre", _("* Pre"), Common::U32String(), kModeCmd);
|
|
_editText = new EditTextWidget(this, "Predictive.Word", _search, Common::U32String(), 0, 0);
|
|
|
|
_userDictHasChanged = false;
|
|
|
|
_predictiveDict.nameDict = "predictive_dictionary";
|
|
_predictiveDict.defaultFilename = "pred.dic";
|
|
|
|
_userDict.nameDict = "user_dictionary";
|
|
_userDict.defaultFilename = "user.dic";
|
|
|
|
if (!_predictiveDict.dictText) {
|
|
loadAllDictionary(_predictiveDict);
|
|
if (!_predictiveDict.dictText)
|
|
debug(5, "Predictive Dialog: pred.dic not loaded");
|
|
}
|
|
|
|
if (!_userDict.dictText) {
|
|
loadAllDictionary(_userDict);
|
|
if (!_userDict.dictText)
|
|
debug(5, "Predictive Dialog: user.dic not loaded");
|
|
}
|
|
|
|
mergeDicts();
|
|
|
|
memset(_repeatcount, 0, sizeof(_repeatcount));
|
|
|
|
_prefix.clear();
|
|
_currentCode.clear();
|
|
_currentWord.clear();
|
|
_wordNumber = 0;
|
|
_numMatchingWords = 0;
|
|
memset(_predictiveResult, 0, sizeof(_predictiveResult));
|
|
|
|
_lastButton = kNoAct;
|
|
_mode = kModePre;
|
|
|
|
_lastTime = 0;
|
|
_curTime = 0;
|
|
_lastPressedButton = kNoAct;
|
|
|
|
_memoryList[0] = _predictiveDict.dictText;
|
|
_memoryList[1] = _userDict.dictText;
|
|
_numMemory = 0;
|
|
|
|
_navigationWithKeys = false;
|
|
|
|
_curPressedButton = kNoAct;
|
|
_needRefresh = true;
|
|
_isPressed = false;
|
|
|
|
}
|
|
|
|
PredictiveDialog::~PredictiveDialog() {
|
|
for (int i = 0; i < _numMemory; i++) {
|
|
free(_memoryList[i]);
|
|
}
|
|
free(_userDict.dictLine);
|
|
free(_predictiveDict.dictLine);
|
|
free(_unitedDict.dictLine);
|
|
}
|
|
|
|
void PredictiveDialog::reflowLayout() {
|
|
#ifndef DISABLE_FANCY_THEMES
|
|
removeWidget(_button[kDelAct]);
|
|
_button[kDelAct]->setNext(nullptr);
|
|
delete _button[kDelAct];
|
|
_button[kDelAct] = nullptr;
|
|
|
|
if (g_gui.xmlEval()->getVar("Globals.Predictive.ShowDeletePic") == 1 && g_gui.theme()->supportsImages()) {
|
|
_button[kDelAct] = new PicButtonWidget(this, "Predictive.Delete", _("Delete char"), kDelCmd);
|
|
((PicButtonWidget *)_button[kDelAct])->setGfxFromTheme(ThemeEngine::kImageDelButton);
|
|
} else {
|
|
_button[kDelAct] = new ButtonWidget(this, "Predictive.Delete" , _("<") , Common::U32String(), kDelCmd);
|
|
}
|
|
#endif
|
|
|
|
Dialog::reflowLayout();
|
|
}
|
|
|
|
void PredictiveDialog::saveUserDictToFile() {
|
|
if (_userDictHasChanged) {
|
|
ConfMan.registerDefault("user_dictionary", "user.dic");
|
|
|
|
Common::OutSaveFile *file = g_system->getSavefileManager()->openForSaving(ConfMan.get("user_dictionary"));
|
|
|
|
for (int i = 0; i < _userDict.dictLineCount; i++) {
|
|
file->writeString(_userDict.dictLine[i]);
|
|
file->writeString("\n");
|
|
}
|
|
|
|
file->finalize();
|
|
delete file;
|
|
}
|
|
}
|
|
|
|
void PredictiveDialog::handleKeyUp(Common::KeyState state) {
|
|
if (_curPressedButton != kNoAct && !_needRefresh) {
|
|
_button[_curPressedButton]->setUnpressedState();
|
|
processButton(_curPressedButton);
|
|
}
|
|
|
|
_isPressed = false;
|
|
}
|
|
|
|
void PredictiveDialog::handleKeyDown(Common::KeyState state) {
|
|
if (_isPressed) {
|
|
return;
|
|
}
|
|
|
|
_isPressed = true;
|
|
_curPressedButton = kNoAct;
|
|
_needRefresh = false;
|
|
|
|
if (getFocusWidget() == _editText) {
|
|
setFocusWidget(_button[kAddAct]);
|
|
}
|
|
|
|
if (_lastButton == kNoAct) {
|
|
_lastButton = kButton5Act;
|
|
}
|
|
|
|
switch (state.keycode) {
|
|
case Common::KEYCODE_ESCAPE:
|
|
saveUserDictToFile();
|
|
close();
|
|
return;
|
|
case Common::KEYCODE_LEFT:
|
|
_navigationWithKeys = true;
|
|
if (_lastButton == kButton1Act || _lastButton == kButton4Act || _lastButton == kButton7Act)
|
|
_curPressedButton = ButtonId(_lastButton + 2);
|
|
else if (_lastButton == kDelAct)
|
|
_curPressedButton = kButton1Act;
|
|
else if (_lastButton == kModeAct)
|
|
_curPressedButton = kNextAct;
|
|
else if (_lastButton == kNextAct)
|
|
_curPressedButton = kButton0Act;
|
|
else if (_lastButton == kAddAct)
|
|
_curPressedButton = kOkAct;
|
|
else if (_lastButton == kCancelAct)
|
|
_curPressedButton = kAddAct;
|
|
else
|
|
_curPressedButton = ButtonId(_lastButton - 1);
|
|
|
|
|
|
if (_mode != kModeAbc && _lastButton == kCancelAct)
|
|
_curPressedButton = kOkAct;
|
|
|
|
_needRefresh = true;
|
|
break;
|
|
case Common::KEYCODE_RIGHT:
|
|
_navigationWithKeys = true;
|
|
if (_lastButton == kButton3Act || _lastButton == kButton6Act || _lastButton == kButton9Act || _lastButton == kOkAct)
|
|
_curPressedButton = ButtonId(_lastButton - 2);
|
|
else if (_lastButton == kDelAct)
|
|
_curPressedButton = kButton3Act;
|
|
else if (_lastButton == kButton0Act)
|
|
_curPressedButton = kNextAct;
|
|
else if (_lastButton == kNextAct)
|
|
_curPressedButton = kModeAct;
|
|
else if (_lastButton == kAddAct)
|
|
_curPressedButton = kCancelAct;
|
|
else if (_lastButton == kOkAct)
|
|
_curPressedButton = kAddAct;
|
|
else
|
|
_curPressedButton = ButtonId(_lastButton + 1);
|
|
|
|
if (_mode != kModeAbc && _lastButton == kOkAct)
|
|
_curPressedButton = kCancelAct;
|
|
_needRefresh = true;
|
|
break;
|
|
case Common::KEYCODE_UP:
|
|
_navigationWithKeys = true;
|
|
if (_lastButton <= kButton3Act)
|
|
_curPressedButton = kDelAct;
|
|
else if (_lastButton == kDelAct)
|
|
_curPressedButton = kOkAct;
|
|
else if (_lastButton == kModeAct)
|
|
_curPressedButton = kButton7Act;
|
|
else if (_lastButton == kButton0Act)
|
|
_curPressedButton = kButton8Act;
|
|
else if (_lastButton == kNextAct)
|
|
_curPressedButton = kButton9Act;
|
|
else if (_lastButton == kAddAct)
|
|
_curPressedButton = kModeAct;
|
|
else if (_lastButton == kCancelAct)
|
|
_curPressedButton = kButton0Act;
|
|
else if (_lastButton == kOkAct)
|
|
_curPressedButton = kNextAct;
|
|
else
|
|
_curPressedButton = ButtonId(_lastButton - 3);
|
|
_needRefresh = true;
|
|
break;
|
|
case Common::KEYCODE_DOWN:
|
|
_navigationWithKeys = true;
|
|
if (_lastButton == kDelAct)
|
|
_curPressedButton = kButton3Act;
|
|
else if (_lastButton == kButton7Act)
|
|
_curPressedButton = kModeAct;
|
|
else if (_lastButton == kButton8Act)
|
|
_curPressedButton = kButton0Act;
|
|
else if (_lastButton == kButton9Act)
|
|
_curPressedButton = kNextAct;
|
|
else if (_lastButton == kModeAct)
|
|
_curPressedButton = kAddAct;
|
|
else if (_lastButton == kButton0Act)
|
|
_curPressedButton = kCancelAct;
|
|
else if (_lastButton == kNextAct)
|
|
_curPressedButton = kOkAct;
|
|
else if (_lastButton == kAddAct || _lastButton == kCancelAct || _lastButton == kOkAct)
|
|
_curPressedButton = kDelAct;
|
|
else
|
|
_curPressedButton = ButtonId(_lastButton + 3);
|
|
|
|
if (_mode != kModeAbc && _lastButton == kModeAct)
|
|
_curPressedButton = kCancelAct;
|
|
|
|
_needRefresh = true;
|
|
break;
|
|
case Common::KEYCODE_KP_ENTER:
|
|
case Common::KEYCODE_RETURN:
|
|
if (state.flags & Common::KBD_CTRL) {
|
|
_curPressedButton = kOkAct;
|
|
break;
|
|
}
|
|
if (_navigationWithKeys) {
|
|
// when the user has utilized arrow key navigation,
|
|
// interpret enter as 'click' on the _curPressedButton button
|
|
_curPressedButton = _lastButton;
|
|
_needRefresh = false;
|
|
} else {
|
|
// else it is a shortcut for 'Ok'
|
|
_curPressedButton = kOkAct;
|
|
}
|
|
break;
|
|
case Common::KEYCODE_KP_PLUS:
|
|
_curPressedButton = kAddAct;
|
|
break;
|
|
case Common::KEYCODE_BACKSPACE:
|
|
case Common::KEYCODE_KP_MINUS:
|
|
_curPressedButton = kDelAct;
|
|
break;
|
|
case Common::KEYCODE_KP_DIVIDE:
|
|
_curPressedButton = kNextAct;
|
|
break;
|
|
case Common::KEYCODE_KP_MULTIPLY:
|
|
_curPressedButton = kModeAct;
|
|
break;
|
|
case Common::KEYCODE_KP0:
|
|
_curPressedButton = kButton0Act;
|
|
break;
|
|
case Common::KEYCODE_KP1:
|
|
case Common::KEYCODE_KP2:
|
|
case Common::KEYCODE_KP3:
|
|
case Common::KEYCODE_KP4:
|
|
case Common::KEYCODE_KP5:
|
|
case Common::KEYCODE_KP6:
|
|
case Common::KEYCODE_KP7:
|
|
case Common::KEYCODE_KP8:
|
|
case Common::KEYCODE_KP9:
|
|
_curPressedButton = ButtonId(state.keycode - Common::KEYCODE_KP1);
|
|
break;
|
|
default:
|
|
Dialog::handleKeyDown(state);
|
|
}
|
|
|
|
if (_lastButton != _curPressedButton)
|
|
_button[_lastButton]->setUnpressedState();
|
|
|
|
if (_curPressedButton != kNoAct && !_needRefresh)
|
|
_button[_curPressedButton]->setPressedState();
|
|
else
|
|
updateHighLightedButton(_curPressedButton);
|
|
}
|
|
|
|
void PredictiveDialog::updateHighLightedButton(ButtonId act) {
|
|
if (_curPressedButton != kNoAct) {
|
|
_button[_lastButton]->setHighLighted(false);
|
|
_lastButton = act;
|
|
_button[_lastButton]->setHighLighted(true);
|
|
}
|
|
}
|
|
|
|
void PredictiveDialog::handleCommand(CommandSender *sender, uint32 cmd, uint32 data) {
|
|
_curPressedButton = kNoAct;
|
|
|
|
_navigationWithKeys = false;
|
|
|
|
if (_lastButton != kNoAct)
|
|
_button[_lastButton]->setHighLighted(false);
|
|
|
|
switch (cmd) {
|
|
case kDelCmd:
|
|
_curPressedButton = kDelAct;
|
|
break;
|
|
case kNextCmd:
|
|
_curPressedButton = kNextAct;
|
|
break;
|
|
case kAddCmd:
|
|
_curPressedButton = kAddAct;
|
|
break;
|
|
case kModeCmd:
|
|
_curPressedButton = kModeAct;
|
|
break;
|
|
case kBut1Cmd:
|
|
_curPressedButton = kButton1Act;
|
|
break;
|
|
case kBut2Cmd:
|
|
_curPressedButton = kButton2Act;
|
|
break;
|
|
case kBut3Cmd:
|
|
_curPressedButton = kButton3Act;
|
|
break;
|
|
case kBut4Cmd:
|
|
_curPressedButton = kButton4Act;
|
|
break;
|
|
case kBut5Cmd:
|
|
_curPressedButton = kButton5Act;
|
|
break;
|
|
case kBut6Cmd:
|
|
_curPressedButton = kButton6Act;
|
|
break;
|
|
case kBut7Cmd:
|
|
_curPressedButton = kButton7Act;
|
|
break;
|
|
case kBut8Cmd:
|
|
_curPressedButton = kButton8Act;
|
|
break;
|
|
case kBut9Cmd:
|
|
_curPressedButton = kButton9Act;
|
|
break;
|
|
case kBut0Cmd:
|
|
_curPressedButton = kButton0Act;
|
|
break;
|
|
case kCancelCmd:
|
|
saveUserDictToFile();
|
|
close();
|
|
// When we cancel the dialog no result should be returned. Thus, we
|
|
// will invalidate any result here.
|
|
_predictiveResult[0] = 0;
|
|
return;
|
|
case kOkCmd:
|
|
_curPressedButton = kOkAct;
|
|
break;
|
|
default:
|
|
Dialog::handleCommand(sender, cmd, data);
|
|
}
|
|
|
|
if (_curPressedButton != kNoAct) {
|
|
processButton(_curPressedButton);
|
|
}
|
|
}
|
|
|
|
void PredictiveDialog::processButton(ButtonId button) {
|
|
static const char *const buttonStr[] = {
|
|
"1", "2", "3",
|
|
"4", "5", "6",
|
|
"7", "8", "9",
|
|
"0"
|
|
};
|
|
|
|
static const char *const buttons[] = {
|
|
"'-.&", "abc", "def",
|
|
"ghi", "jkl", "mno",
|
|
"pqrs", "tuv", "wxyz",
|
|
"next", "add",
|
|
"<",
|
|
"Cancel", "OK",
|
|
"Pre", "(0) ", nullptr
|
|
};
|
|
|
|
if (_mode == kModeAbc) {
|
|
if (button >= kButton1Act && button <= kButton9Act) {
|
|
if (!_lastTime)
|
|
_lastTime = g_system->getMillis();
|
|
if (_lastPressedButton == button) {
|
|
_curTime = g_system->getMillis();
|
|
if ((_curTime - _lastTime) < kRepeatDelay) {
|
|
button = kNextAct;
|
|
_lastTime = _curTime;
|
|
} else {
|
|
_lastTime = 0;
|
|
}
|
|
} else {
|
|
_lastPressedButton = button;
|
|
_lastTime = g_system->getMillis();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (button >= kButton1Act) {
|
|
_lastButton = button;
|
|
if (button == kButton0Act && _mode != kModeNum) { // Space
|
|
// bring MRU word at the top of the list when changing words
|
|
if (_mode == kModePre && _unitedDict.dictActLine && _numMatchingWords > 1 && _wordNumber != 0)
|
|
bringWordtoTop(_unitedDict.dictActLine, _wordNumber);
|
|
|
|
Common::strcpy_s(_temp, _currentWord.c_str());
|
|
_prefix += _temp;
|
|
_prefix += " ";
|
|
_currentCode.clear();
|
|
_currentWord.clear();
|
|
_numMatchingWords = 0;
|
|
memset(_repeatcount, 0, sizeof(_repeatcount));
|
|
_lastTime = 0;
|
|
_lastPressedButton = kNoAct;
|
|
_curTime = 0;
|
|
} else if (button < kNextAct || button == kDelAct || button == kButton0Act) { // number or backspace
|
|
if (button == kDelAct) { // backspace
|
|
if (_currentCode.size()) {
|
|
_repeatcount[_currentCode.size() - 1] = 0;
|
|
_currentCode.deleteLastChar();
|
|
if (_currentCode.empty())
|
|
_currentWord.clear();
|
|
} else {
|
|
if (_prefix.size())
|
|
_prefix.deleteLastChar();
|
|
}
|
|
} else if (_prefix.size() + _currentCode.size() < kMaxWordLen - 1) { // don't overflow the dialog line
|
|
if (button == kButton0Act) { // zero
|
|
_currentCode += buttonStr[9];
|
|
} else {
|
|
_currentCode += buttonStr[button];
|
|
}
|
|
}
|
|
|
|
switch (_mode) {
|
|
case kModeNum:
|
|
_currentWord = _currentCode;
|
|
break;
|
|
case kModePre:
|
|
if (!matchWord() && _currentCode.size()) {
|
|
_currentCode.deleteLastChar();
|
|
matchWord();
|
|
}
|
|
_numMatchingWords = countWordsInString(_unitedDict.dictActLine);
|
|
break;
|
|
case kModeAbc:
|
|
for (uint x = 0; x < _currentCode.size(); x++)
|
|
if (_currentCode[x] >= '1')
|
|
_temp[x] = buttons[_currentCode[x] - '1'][_repeatcount[x]];
|
|
_temp[_currentCode.size()] = 0;
|
|
_currentWord = _temp;
|
|
default:
|
|
break;
|
|
}
|
|
} else if (button == kNextAct) { // next
|
|
if (_mode == kModePre) {
|
|
if (_unitedDict.dictActLine && _numMatchingWords > 1) {
|
|
_wordNumber = (_wordNumber + 1) % _numMatchingWords;
|
|
char tmp[kMaxLineLen];
|
|
Common::strlcpy(tmp, _unitedDict.dictActLine, kMaxLineLen);
|
|
char *tok = strtok(tmp, " ");
|
|
for (uint8 i = 0; i <= _wordNumber; i++)
|
|
tok = strtok(nullptr, " ");
|
|
_currentWord = Common::String(tok, _currentCode.size());
|
|
}
|
|
} else if (_mode == kModeAbc) {
|
|
uint x = _currentCode.size();
|
|
if (x) {
|
|
if (_currentCode.lastChar() == '1' || _currentCode.lastChar() == '7' || _currentCode.lastChar() == '9')
|
|
_repeatcount[x - 1] = (_repeatcount[x - 1] + 1) % 4;
|
|
else
|
|
_repeatcount[x - 1] = (_repeatcount[x - 1] + 1) % 3;
|
|
|
|
if (_currentCode.lastChar() >= '1')
|
|
_currentWord.setChar(buttons[_currentCode[x - 1] - '1'][_repeatcount[x - 1]], x - 1);
|
|
}
|
|
}
|
|
} else if (button == kAddAct) { // add
|
|
if (_mode == kModeAbc)
|
|
addWordToDict();
|
|
else
|
|
debug(5, "Predictive Dialog: button Add doesn't work in this mode");
|
|
} else if (button == kOkAct) { // Ok
|
|
// bring MRU word at the top of the list when ok'ed out of the dialog
|
|
if (_mode == kModePre && _unitedDict.dictActLine && _numMatchingWords > 1 && _wordNumber != 0)
|
|
bringWordtoTop(_unitedDict.dictActLine, _wordNumber);
|
|
} else if (button == kModeAct) { // Mode
|
|
_mode++;
|
|
_button[kAddAct]->setEnabled(false);
|
|
if (_mode > kModeAbc) {
|
|
_mode = kModePre;
|
|
// I18N: Pre means 'Predictive', leave '*' as is
|
|
_button[kModeAct]->setLabel(_("* Pre"));
|
|
} else if (_mode == kModeNum) {
|
|
// I18N: 'Num' means Numbers
|
|
_button[kModeAct]->setLabel(_("* Num"));
|
|
} else {
|
|
// I18N: 'Abc' means Latin alphabet input
|
|
_button[kModeAct]->setLabel(_("* Abc"));
|
|
_button[kAddAct]->setEnabled(true);
|
|
}
|
|
|
|
// truncate current input at mode change
|
|
strncpy(_temp, _currentWord.c_str(), _currentCode.size());
|
|
_temp[_currentCode.size()] = 0;
|
|
_prefix += _temp;
|
|
_currentCode.clear();
|
|
_currentWord.clear();
|
|
memset(_repeatcount, 0, sizeof(_repeatcount));
|
|
|
|
_lastTime = 0;
|
|
_lastPressedButton = kNoAct;
|
|
_curTime = 0;
|
|
}
|
|
}
|
|
|
|
pressEditText();
|
|
|
|
if (button == kOkAct)
|
|
close();
|
|
|
|
if (button == kCancelAct) {
|
|
saveUserDictToFile();
|
|
close();
|
|
}
|
|
}
|
|
|
|
void PredictiveDialog::mergeDicts() {
|
|
_unitedDict.dictLineCount = _predictiveDict.dictLineCount + _userDict.dictLineCount;
|
|
_unitedDict.dictLine = (char **)calloc(_unitedDict.dictLineCount, sizeof(char *));
|
|
|
|
if (!_unitedDict.dictLine) {
|
|
debug(5, "Predictive Dialog: cannot allocate memory for united dic");
|
|
return;
|
|
}
|
|
|
|
int lenUserDictCode, lenPredictiveDictCode, lenCode;
|
|
int i, j, k;
|
|
i = j = k = 0;
|
|
|
|
while ((i < _userDict.dictLineCount) && (j < _predictiveDict.dictLineCount)) {
|
|
lenUserDictCode = strchr(_userDict.dictLine[i], ' ') - _userDict.dictLine[i];
|
|
lenPredictiveDictCode = strchr(_predictiveDict.dictLine[j], ' ') - _predictiveDict.dictLine[j];
|
|
lenCode = (lenUserDictCode >= lenPredictiveDictCode) ? lenUserDictCode : lenPredictiveDictCode;
|
|
if (strncmp(_userDict.dictLine[i], _predictiveDict.dictLine[j], lenCode) >= 0) {
|
|
_unitedDict.dictLine[k++] = _predictiveDict.dictLine[j++];
|
|
} else {
|
|
_unitedDict.dictLine[k++] = _userDict.dictLine[i++];
|
|
}
|
|
}
|
|
|
|
while (i < _userDict.dictLineCount) {
|
|
_unitedDict.dictLine[k++] = _userDict.dictLine[i++];
|
|
}
|
|
|
|
while (j < _predictiveDict.dictLineCount) {
|
|
_unitedDict.dictLine[k++] = _predictiveDict.dictLine[j++];
|
|
}
|
|
}
|
|
|
|
uint8 PredictiveDialog::countWordsInString(const char *const str) {
|
|
// Count the number of (space separated) words in the given string.
|
|
const char *ptr;
|
|
|
|
if (!str)
|
|
return 0;
|
|
|
|
ptr = strchr(str, ' ');
|
|
if (!ptr) {
|
|
debug(5, "Predictive Dialog: Invalid dictionary line");
|
|
return 0;
|
|
}
|
|
|
|
uint8 num = 1;
|
|
ptr++;
|
|
while ((ptr = strchr(ptr, ' '))) {
|
|
ptr++;
|
|
num++;
|
|
}
|
|
return num;
|
|
}
|
|
|
|
void PredictiveDialog::bringWordtoTop(char *str, int wordnum) {
|
|
// This function reorders the words on the given pred.dic line
|
|
// by moving the word at position 'wordnum' to the front (that is, right behind
|
|
// right behind the numerical code word at the start of the line).
|
|
Common::Array<Common::String> words;
|
|
char buf[kMaxLineLen];
|
|
|
|
if (!str)
|
|
return;
|
|
Common::strlcpy(buf, str, kMaxLineLen);
|
|
char *word = strtok(buf, " ");
|
|
if (!word) {
|
|
debug(5, "Predictive Dialog: Invalid dictionary line");
|
|
return;
|
|
}
|
|
|
|
words.push_back(word);
|
|
while ((word = strtok(nullptr, " ")) != nullptr)
|
|
words.push_back(word);
|
|
words.insert_at(1, words.remove_at(wordnum + 1));
|
|
|
|
Common::String tmp;
|
|
for (uint8 i = 0; i < words.size(); i++)
|
|
tmp += words[i] + " ";
|
|
tmp.deleteLastChar();
|
|
memcpy(str, tmp.c_str(), strlen(str));
|
|
}
|
|
|
|
int PredictiveDialog::binarySearch(const char *const *const dictLine, const Common::String &code, const int dictLineCount) {
|
|
int hi = dictLineCount - 1;
|
|
int lo = 0;
|
|
int line = 0;
|
|
while (lo <= hi) {
|
|
line = (lo + hi) / 2;
|
|
int cmpVal = strncmp(dictLine[line], code.c_str(), code.size());
|
|
if (cmpVal > 0)
|
|
hi = line - 1;
|
|
else if (cmpVal < 0)
|
|
lo = line + 1;
|
|
else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (hi < lo) {
|
|
return -(lo + 1);
|
|
} else {
|
|
return line;
|
|
}
|
|
}
|
|
|
|
bool PredictiveDialog::matchWord() {
|
|
// If there is no dictionary, then there is no match.
|
|
if (_unitedDict.dictLineCount <= 0)
|
|
return false;
|
|
|
|
// If no text has been entered, then there is no match.
|
|
if (_currentCode.empty())
|
|
return false;
|
|
|
|
// If the currently entered text is too long, it cannot match anything.
|
|
if (_currentCode.size() > kMaxWordLen)
|
|
return false;
|
|
|
|
// The entries in the dictionary consist of a code, a space, and then
|
|
// a space-separated list of words matching this code.
|
|
// To exactly match a code, we therefore match the code plus the trailing
|
|
// space in the dictionary.
|
|
Common::String code = _currentCode + " ";
|
|
|
|
int line = binarySearch(_unitedDict.dictLine, code, _unitedDict.dictLineCount);
|
|
if (line < 0) {
|
|
line = -(line + 1);
|
|
_unitedDict.dictActLine = nullptr;
|
|
} else {
|
|
_unitedDict.dictActLine = _unitedDict.dictLine[line];
|
|
}
|
|
|
|
_currentWord.clear();
|
|
_wordNumber = 0;
|
|
if (0 == strncmp(_unitedDict.dictLine[line], _currentCode.c_str(), _currentCode.size())) {
|
|
char tmp[kMaxLineLen];
|
|
Common::strlcpy(tmp, _unitedDict.dictLine[line], kMaxLineLen);
|
|
char *tok;
|
|
strtok(tmp, " ");
|
|
tok = strtok(nullptr, " ");
|
|
_currentWord = Common::String(tok, _currentCode.size());
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool PredictiveDialog::searchWord(const char *const where, const Common::String &whatCode) {
|
|
const char *ptr = where;
|
|
ptr += whatCode.size();
|
|
|
|
const char *newPtr;
|
|
bool is = false;
|
|
while ((newPtr = strchr(ptr, ' '))) {
|
|
if (0 == strncmp(ptr, _currentWord.c_str(), newPtr - ptr)) {
|
|
is = true;
|
|
break;
|
|
}
|
|
ptr = newPtr + 1;
|
|
}
|
|
if (!is) {
|
|
if (0 == strcmp(ptr, _currentWord.c_str())) {
|
|
is = true;
|
|
}
|
|
}
|
|
return is;
|
|
}
|
|
|
|
void PredictiveDialog::addWord(Dict &dict, const Common::String &word, const Common::String &code) {
|
|
char *newLine = nullptr;
|
|
Common::String tmpCode = code + ' ';
|
|
int line = binarySearch(dict.dictLine, tmpCode, dict.dictLineCount);
|
|
if (line >= 0) {
|
|
if (searchWord(dict.dictLine[line], tmpCode)) {
|
|
// if we found code and word, we should not insert/expands any word
|
|
return;
|
|
} else {
|
|
// if we found the code, but did not find a word, we must
|
|
// EXPANDS the currnent line with new word
|
|
int oldLineSize = strlen(dict.dictLine[line]);
|
|
int newLineSize = oldLineSize + word.size() + 1;
|
|
|
|
newLine = (char *)malloc(newLineSize + 1);
|
|
|
|
char *ptr = newLine;
|
|
Common::strcpy_s(ptr, newLineSize + 1, dict.dictLine[line]);
|
|
ptr += oldLineSize;
|
|
Common::String tmp = ' ' + word;
|
|
Common::strcpy_s(ptr, newLineSize + 1 - oldLineSize, tmp.c_str());
|
|
|
|
dict.dictLine[line] = newLine;
|
|
_memoryList[_numMemory++] = newLine;
|
|
|
|
if (dict.nameDict == "user_dictionary")
|
|
_userDictHasChanged = true;
|
|
|
|
return;
|
|
}
|
|
} else { // if we didn't find the code, we need to INSERT new line with code and word
|
|
if (dict.nameDict == "user_dictionary") {
|
|
// if we must INSERT new line(code and word) to user_dictionary, we need to
|
|
// check if there is a line that we want to INSERT in predictive dictionay
|
|
int predictLine = binarySearch(_predictiveDict.dictLine, tmpCode, _predictiveDict.dictLineCount);
|
|
if (predictLine >= 0) {
|
|
if (searchWord(_predictiveDict.dictLine[predictLine], tmpCode)) {
|
|
// if code and word is in predictive dictionary, we need to copy
|
|
// this line to user dictionary
|
|
int len = (predictLine == _predictiveDict.dictLineCount - 1) ? &_predictiveDict.dictText[_predictiveDict.dictTextSize] - _predictiveDict.dictLine[predictLine] :
|
|
_predictiveDict.dictLine[predictLine + 1] - _predictiveDict.dictLine[predictLine];
|
|
newLine = (char *)malloc(len);
|
|
Common::strlcpy(newLine, _predictiveDict.dictLine[predictLine], len);
|
|
} else {
|
|
// if there is no word in predictive dictionary, we need to copy to
|
|
// user dictionary mathed line + new word.
|
|
int len = (predictLine == _predictiveDict.dictLineCount - 1) ? &_predictiveDict.dictText[_predictiveDict.dictTextSize] - _predictiveDict.dictLine[predictLine] :
|
|
_predictiveDict.dictLine[predictLine + 1] - _predictiveDict.dictLine[predictLine];
|
|
newLine = (char *)malloc(len + word.size() + 1);
|
|
char *ptr = newLine;
|
|
Common::strlcpy(ptr, _predictiveDict.dictLine[predictLine], len);
|
|
ptr[len - 1] = ' ';
|
|
ptr += len;
|
|
Common::strlcpy(ptr, word.c_str(), word.size() + 1);
|
|
}
|
|
} else {
|
|
// if we didn't find line in predictive dialog, we should copy to user dictionary
|
|
// code + word
|
|
Common::String tmp;
|
|
tmp = tmpCode + word;
|
|
newLine = (char *)malloc(tmp.size() + 1);
|
|
Common::strcpy_s(newLine, tmp.size() + 1, tmp.c_str());
|
|
}
|
|
} else {
|
|
// if want to insert line to different from user dictionary, we should copy to this
|
|
// dictionary code + word
|
|
Common::String tmp;
|
|
tmp = tmpCode + word;
|
|
newLine = (char *)malloc(tmp.size() + 1);
|
|
Common::strcpy_s(newLine, tmp.size() + 1, tmp.c_str());
|
|
}
|
|
}
|
|
|
|
// start from here are INSERTING new line to dictionaty ( dict )
|
|
char **newDictLine = (char **)calloc(dict.dictLineCount + 1, sizeof(char *));
|
|
if (!newDictLine) {
|
|
warning("Predictive Dialog: cannot allocate memory for index buffer");
|
|
|
|
free(newLine);
|
|
|
|
return;
|
|
}
|
|
|
|
int k = 0;
|
|
bool inserted = false;
|
|
for (int i = 0; i < dict.dictLineCount; i++) {
|
|
uint lenPredictiveDictCode = strchr(dict.dictLine[i], ' ') - dict.dictLine[i];
|
|
uint lenCode = (lenPredictiveDictCode >= (code.size() - 1)) ? lenPredictiveDictCode : code.size() - 1;
|
|
if ((strncmp(dict.dictLine[i], code.c_str(), lenCode) > 0) && !inserted) {
|
|
newDictLine[k++] = newLine;
|
|
inserted = true;
|
|
}
|
|
if (k != (dict.dictLineCount + 1)) {
|
|
newDictLine[k++] = dict.dictLine[i];
|
|
}
|
|
}
|
|
if (!inserted)
|
|
newDictLine[k] = newLine;
|
|
|
|
_memoryList[_numMemory++] = newLine;
|
|
|
|
free(dict.dictLine);
|
|
dict.dictLineCount += 1;
|
|
dict.dictLine = (char **)calloc(dict.dictLineCount, sizeof(char *));
|
|
if (!dict.dictLine) {
|
|
warning("Predictive Dialog: cannot allocate memory for index buffer");
|
|
free(newDictLine);
|
|
return;
|
|
}
|
|
|
|
for (int i = 0; i < dict.dictLineCount; i++) {
|
|
dict.dictLine[i] = newDictLine[i];
|
|
}
|
|
|
|
if (dict.nameDict == "user_dictionary")
|
|
_userDictHasChanged = true;
|
|
|
|
free(newDictLine);
|
|
}
|
|
|
|
void PredictiveDialog::addWordToDict() {
|
|
if (_numMemory < kMaxWord) {
|
|
addWord(_unitedDict, _currentWord, _currentCode);
|
|
addWord(_userDict, _currentWord, _currentCode);
|
|
} else {
|
|
warning("Predictive Dialog: You cannot add word to user dictionary...");
|
|
}
|
|
}
|
|
|
|
void PredictiveDialog::loadDictionary(Common::SeekableReadStream *in, Dict &dict) {
|
|
int lines = 0;
|
|
|
|
uint32 time1 = g_system->getMillis();
|
|
|
|
dict.dictTextSize = in->size();
|
|
dict.dictText = (char *)malloc(dict.dictTextSize + 1);
|
|
|
|
if (!dict.dictText) {
|
|
warning("Predictive Dialog: Not enough memory to load the file user.dic");
|
|
return;
|
|
}
|
|
|
|
in->read(dict.dictText, dict.dictTextSize);
|
|
dict.dictText[dict.dictTextSize] = 0;
|
|
uint32 time2 = g_system->getMillis();
|
|
debug(5, "Predictive Dialog: Time to read %s: %d bytes, %d ms", ConfMan.get(dict.nameDict).c_str(), dict.dictTextSize, time2 - time1);
|
|
delete in;
|
|
|
|
char *ptr = dict.dictText;
|
|
lines = 1;
|
|
while ((ptr = strchr(ptr, '\n'))) {
|
|
lines++;
|
|
ptr++;
|
|
}
|
|
|
|
dict.dictLine = (char **)calloc(lines, sizeof(char *));
|
|
if (dict.dictLine == nullptr) {
|
|
warning("Predictive Dialog: Cannot allocate memory for line index buffer");
|
|
return;
|
|
}
|
|
dict.dictLine[0] = dict.dictText;
|
|
ptr = dict.dictText;
|
|
int i = 1;
|
|
while ((ptr = strchr(ptr, '\n'))) {
|
|
*ptr = 0;
|
|
ptr++;
|
|
dict.dictLine[i++] = ptr;
|
|
}
|
|
if (dict.dictLine[lines - 1][0] == 0)
|
|
lines--;
|
|
|
|
dict.dictLineCount = lines;
|
|
debug(5, "Predictive Dialog: Loaded %d lines", dict.dictLineCount);
|
|
|
|
// FIXME: We use binary search on _predictiveDict.dictLine, yet we make no at_tempt
|
|
// to ever sort this array. That seems risky, doesn't it?
|
|
|
|
uint32 time3 = g_system->getMillis();
|
|
debug(5, "Predictive Dialog: Time to parse %s: %d, total: %d", ConfMan.get(dict.nameDict).c_str(), time3 - time2, time3 - time1);
|
|
}
|
|
|
|
void PredictiveDialog::loadAllDictionary(Dict &dict) {
|
|
ConfMan.registerDefault(dict.nameDict, dict.defaultFilename);
|
|
|
|
if (dict.nameDict == "predictive_dictionary") {
|
|
Common::File *inFile = new Common::File();
|
|
if (!inFile->open(ConfMan.getPath(dict.nameDict))) {
|
|
warning("Predictive Dialog: cannot read file: %s", dict.defaultFilename.c_str());
|
|
delete inFile;
|
|
return;
|
|
}
|
|
loadDictionary(inFile, dict);
|
|
} else {
|
|
Common::InSaveFile *inFile = g_system->getSavefileManager()->openForLoading(ConfMan.get(dict.nameDict));
|
|
if (!inFile) {
|
|
warning("Predictive Dialog: cannot read file: %s", dict.defaultFilename.c_str());
|
|
return;
|
|
}
|
|
loadDictionary(inFile, dict);
|
|
}
|
|
}
|
|
|
|
void PredictiveDialog::pressEditText() {
|
|
Common::strlcpy(_predictiveResult, _prefix.c_str(), sizeof(_predictiveResult));
|
|
Common::strlcat(_predictiveResult, _currentWord.c_str(), sizeof(_predictiveResult));
|
|
_editText->setEditString(Common::convertToU32String(_predictiveResult));
|
|
//_editText->setCaretPos(_prefix.size() + _currentWord.size());
|
|
}
|
|
|
|
} // namespace GUI
|