// Ryzom - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdpch.h" #include "group_editbox.h" #include "interface_manager.h" #include "input_handler_manager.h" #include "nel/misc/command.h" #include "view_text.h" #include "game_share/xml_auto_ptr.h" #include "interface_options.h" #include "dbctrl_sheet.h" #include "group_container.h" #include "../time_client.h" #include "lua_ihm.h" #ifdef NL_OS_WINDOWS #include #endif using namespace std; using namespace NLMISC; using namespace NL3D; ///////////// // STATICS // ///////////// sint32 CGroupEditBox::_SelectCursorPos = 0; CGroupEditBox *CGroupEditBox::_CurrSelection = NULL; CGroupEditBox *CGroupEditBox::_MenuFather = NULL; // ---------------------------------------------------------------------------- NLMISC_REGISTER_OBJECT(CViewBase, CGroupEditBox, std::string, "edit_box"); CGroupEditBox::CGroupEditBox(const TCtorParam ¶m) : CInterfaceGroup(param), _CursorPos(0), _BlinkTime(0.f), _BlinkState(false), _ViewText(NULL), _Setupped(false), _EntryType(Text), _MaxNumChar(~0), _MaxNumReturn(~0), _MaxFloatPrec(20), _MaxCharsSize(32768), _CursorAtPreviousLineEnd(false), _BackupFatherContainerPos(false), _WantReturn(false), _CanRedo(false), _CanUndo(false) { _FirstVisibleChar = 0; _LastVisibleChar = 0; _Prompt = ">"; _LooseFocusOnEnter= true; _RecoverFocusOnEnter= true; _ResetFocusOnHide = false; _BackSelectColor= CRGBA::White; _TextSelectColor= CRGBA::Black; _MaxHistoric= 0; _CurrentHistoricIndex= -1; _BypassNextKey = false; _PrevNumLine = 1; _Savable = true; _CursorTexID = -1; _CursorWidth = 0; _IntegerMinValue= INT_MIN; _IntegerMaxValue= INT_MAX; _PositiveIntegerMinValue= 0; _PositiveIntegerMaxValue= UINT_MAX; _ViewTextDeltaX= 0; _DefaultInputString= false; _Frozen= false; } // ---------------------------------------------------------------------------- CGroupEditBox::~CGroupEditBox() { if (this == _CurrSelection) _CurrSelection = NULL; CInterfaceManager *im = CInterfaceManager::getInstance(); if (im->getCaptureKeyboard() == this || im->getOldCaptureKeyboard() == this) { im->resetCaptureKeyboard(); } } // ---------------------------------------------------------------------------- bool CGroupEditBox::parse(xmlNodePtr cur, CInterfaceGroup * parentGroup) { if(!CInterfaceGroup::parse(cur, parentGroup)) return false; CXMLAutoPtr prop; // CInterfaceManager *pIM = CInterfaceManager::getInstance(); // CViewRenderer &rVR = pIM->getViewRenderer(); if (! CInterfaceGroup::parse(cur,parentGroup) ) { string tmp = "cannot parse view:"+getId()+", parent:"+parentGroup->getId(); nlinfo(tmp.c_str()); return false; } // NB: use InterfaceGroup "OnEnter" data. Different script params for an historic reason parseAH(cur, "onenter", "params", _AHOnEnter, _AHOnEnterParams); prop = (char*) xmlGetProp( cur, (xmlChar*)"onchange" ); if (prop) _AHOnChange = (const char *) prop; prop = (char*) xmlGetProp( cur, (xmlChar*)"onchange_params" ); if (prop) _ParamsOnChange = (const char *) prop; prop = (char*) xmlGetProp( cur, (xmlChar*)"on_focus_lost" ); if (prop) _AHOnFocusLost = (const char *) prop; prop = (char*) xmlGetProp( cur, (xmlChar*)"on_focus_lost_params" ); if (prop) _AHOnFocusLostParams = (const char *) prop; prop = (char*) xmlGetProp( cur, (xmlChar*)"on_focus" ); if (prop) _AHOnFocus = (const char *) prop; prop = (char*) xmlGetProp( cur, (xmlChar*)"on_focus_params" ); if (prop) _AHOnFocusParams = (const char *) prop; prop = (char*) xmlGetProp( cur, (xmlChar*)"max_num_chars" ); if (prop) fromString((const char*)prop, _MaxNumChar); prop = (char*) xmlGetProp( cur, (xmlChar*)"max_num_return" ); if (prop) fromString((const char*)prop, _MaxNumReturn); prop = (char*) xmlGetProp( cur, (xmlChar*)"max_chars_size" ); if (prop) fromString((const char*)prop, _MaxCharsSize); prop = (char*) xmlGetProp( cur, (xmlChar*)"enter_loose_focus" ); if (prop) _LooseFocusOnEnter = convertBool(prop); prop = (char*) xmlGetProp( cur, (xmlChar*)"enter_recover_focus" ); if (prop) _RecoverFocusOnEnter = convertBool(prop); prop = (char*) xmlGetProp( cur, (xmlChar*)"reset_focus_on_hide" ); if (prop) _ResetFocusOnHide = convertBool(prop); prop = (char*) xmlGetProp( cur, (xmlChar*)"prompt" ); if (prop) _Prompt = (const char*)prop; prop = (char*) xmlGetProp( cur, (xmlChar*)"entry_type" ); _EntryType = Text; if (prop) { if (stricmp(prop, "text") == 0) _EntryType = Text; else if (stricmp(prop, "integer") == 0) _EntryType = Integer; else if (stricmp(prop, "positive_integer") == 0) _EntryType = PositiveInteger; else if (stricmp(prop, "float") == 0) _EntryType = Float; else if (stricmp(prop, "positive_float") == 0) _EntryType = PositiveFloat; else if (stricmp(prop, "alpha") == 0) _EntryType = Alpha; else if (stricmp(prop, "alpha_num") == 0) _EntryType = AlphaNum; else if (stricmp(prop, "alpha_num_space") == 0) _EntryType = AlphaNumSpace; else if (stricmp(prop, "password") == 0) _EntryType = Password; else if (stricmp(prop, "filename") == 0) _EntryType = Filename; else if (stricmp(prop, "playername") == 0) _EntryType = PlayerName; else nlwarning(" Unknown entry type %s", (const char *) prop); } prop = (char*) xmlGetProp( cur, (xmlChar*)"menu_r" ); if (prop) { string tmp = (const char *) prop; _ListMenuRight = strlwr(tmp); } prop = (char*) xmlGetProp( cur, (xmlChar*)"max_historic" ); if (prop) fromString((const char*)prop, _MaxHistoric); prop = (char*) xmlGetProp( cur, (xmlChar*)"backup_father_container_pos" ); if (prop) _BackupFatherContainerPos = convertBool(prop); prop = (char*) xmlGetProp( cur, (xmlChar*)"want_return" ); if (prop) _WantReturn = convertBool(prop); prop = (char*) xmlGetProp( cur, (xmlChar*)"savable" ); if (prop) _Savable = convertBool(prop); // For float conversion only prop = (char*) xmlGetProp( cur, (xmlChar*)"max_float_prec" ); if (prop) { fromString((const char*)prop, _MaxFloatPrec); _MaxFloatPrec = min((uint32)20, _MaxFloatPrec); } // negative characters filter prop = (char*) xmlGetProp( cur, (xmlChar*)"negative_filter" ); if (prop) { uint length = strlen(prop); _NegativeFilter.resize(length); std::copy((const char *) prop, (const char *) prop + length, _NegativeFilter.begin()); } return true; } // ---------------------------------------------------------------------------- void CGroupEditBox::draw () { // CInterfaceManager *pIM = CInterfaceManager::getInstance(); CViewRenderer &rVR = pIM->getViewRenderer(); // /*CRGBA col; col.modulateFromColor(CRGBA(64,64,64,255), pIM->getGlobalColorForContent()); rVR.drawRotFlipBitmap (_RenderLayer, _XReal, _YReal, _WReal, _HReal, 0, false, rVR.getBlankTextureId(), col ); */ // Frozen? display text in "grayed" CRGBA bkupTextColor; if(_Frozen && _ViewText) { bkupTextColor= _ViewText->getColor(); CRGBA grayed= bkupTextColor; grayed.A>>=2; _ViewText->setColor(grayed); } // draw the group and thus the text CInterfaceGroup::draw(); // restore the text color if changed if(_Frozen && _ViewText) _ViewText->setColor(bkupTextColor); // no text setuped? abort if (!_ViewText) return; // Display the selection if needed if (_CurrSelection == this && _SelectCursorPos!=_CursorPos) { sint32 blankTextId= rVR.getBlankTextureId(); CRGBA col= _BackSelectColor; col.A= pIM->getGlobalColorForContent().A; sint32 minPos= min(_CursorPos, _SelectCursorPos) + _Prompt.length(); sint32 maxPos= max(_CursorPos, _SelectCursorPos) + _Prompt.length(); // get its position on screen sint cxMinPos, cyMinPos; sint cxMaxPos, cyMaxPos; sint height; _ViewText->getCharacterPositionFromIndex(minPos, false, cxMinPos, cyMinPos, height); _ViewText->getCharacterPositionFromIndex(maxPos, false, cxMaxPos, cyMaxPos, height); // Multiline selection if cy different! if(cyMinPos!=cyMaxPos) { nlassert(cyMaxPosgetXReal() + cxMinPos, _ViewText->getYReal() + cyMinPos, _ViewText->getW()-cxMinPos, height, 0, 0, blankTextId, col); // Draw One quad for all lines into the big selection (if any) sint32 cyBigQuad= cyMaxPos+height; if(cyBigQuadgetXReal(), _ViewText->getYReal() + cyBigQuad, _ViewText->getW(), cyMinPos-cyBigQuad, 0, 0, blankTextId, col); } // draw the 3rd quad from the last line from start of line to max pos rVR.drawRotFlipBitmap(_RenderLayer, _ViewText->getXReal(), _ViewText->getYReal() + cyMaxPos, cxMaxPos, height, 0, 0, blankTextId, col); } else { rVR.drawRotFlipBitmap(_RenderLayer, _ViewText->getXReal() + cxMinPos, _ViewText->getYReal() + cyMinPos, cxMaxPos-cxMinPos, height, 0, 0, blankTextId, col); } // Draw The Selection String in black CRGBA bkup= _ViewText->getColor(); _ViewText->setColor(_TextSelectColor); _ViewText->enableStringSelection(minPos, maxPos); // redraw just this string,clipped by the group CInterfaceGroup::drawElement(_ViewText); // bkup _ViewText->setColor(bkup); _ViewText->disableStringSelection(); } // Display the cursor if needed if (pIM->getCaptureKeyboard () == this) { _BlinkTime += DT; if (_BlinkTime > 0.25f) { _BlinkTime = fmodf(_BlinkTime, 0.25f); _BlinkState = !_BlinkState; } if (_BlinkState) // is the cursor shown ? { // get its position on screen sint cx, cy; sint height; _ViewText->getCharacterPositionFromIndex(_CursorPos + _Prompt.length(), _CursorAtPreviousLineEnd, cx, cy, height); // display the cursor // get the texture for the cursor if (_CursorTexID == -1) { _CursorTexID = rVR.getTextureIdFromName("text_cursor.tga"); sint32 dummyCursorHeight; rVR.getTextureSizeFromId(_CursorTexID, _CursorWidth, dummyCursorHeight); } // draw in a layer after to be on TOP of the text sint32 cursorx = std::max((sint) 0, (sint)(cx - (_CursorWidth >> 1))); rVR.drawRotFlipBitmap(_RenderLayer+1, _ViewText->getXReal() + cursorx, _ViewText->getYReal() + cy, 3, height, 0, 0, _CursorTexID); } } } // ---------------------------------------------------------------------------- void CGroupEditBox::copy() { if (_CurrSelection != this) { nlwarning("Selection can only be on focus"); } stopParentBlink(); // get the selection ucstring selection= getSelection(); // copy #ifdef NL_OS_WINDOWS if (selection.size() > 0 && OpenClipboard (NULL)) { bool isUnicode = IsClipboardFormatAvailable(CF_UNICODETEXT) == TRUE; HGLOBAL mem = GlobalAlloc (GHND|GMEM_DDESHARE, (selection.size()+1) * (isUnicode ? 2:1)); if (mem) { void *hLock = GlobalLock (mem); if (isUnicode) wcscpy ((wchar_t*)hLock, (const wchar_t*)selection.c_str()); else strcpy ((char*)hLock, selection.toString().c_str()); GlobalUnlock (mem); EmptyClipboard (); SetClipboardData (isUnicode ? CF_UNICODETEXT:CF_TEXT, mem); } CloseClipboard (); nlinfo ("Chat input was copied in the clipboard"); } #endif // NL_OS_WINDOWS } // ---------------------------------------------------------------------------- void CGroupEditBox::paste() { if(_CurrSelection != NULL) { if (_CurrSelection != this) { nlwarning("Selection can only be on focus"); } cutSelection(); } stopParentBlink(); makeTopWindow(); #ifdef NL_OS_WINDOWS if (OpenClipboard (NULL)) { bool isUnicode = IsClipboardFormatAvailable(CF_UNICODETEXT) == TRUE; HANDLE hObj = GetClipboardData(isUnicode ? CF_UNICODETEXT:CF_TEXT); if (hObj) { void *hLock = GlobalLock(hObj); if (hLock != NULL) { ucstring sString; if (isUnicode) sString = (const ucchar*)hLock; else sString = (const char*)hLock; sint length = sString.length(); ucstring toAppend; // filter character depending on the netry type switch (_EntryType) { case Text: case Password: { if (_NegativeFilter.empty()) { toAppend = sString; } else { for (sint k = 0; k < length; ++k) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } // remove '\r' characters toAppend.erase(std::remove(toAppend.begin(), toAppend.end(), (ucchar) '\r'), toAppend.end()); } break; case PositiveInteger: case PositiveFloat: { for (sint k = 0; k < length; ++k) { if (isdigit(sString[k]) || sString[k]== ' ' || (_EntryType==PositiveFloat && sString[k]=='.') ) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } } break; case Integer: case Float: { for (sint k = 0; k < length; ++k) { if (isdigit(sString[k]) || sString[k]== ' ' || sString[k]== '-' || (_EntryType==Float && sString[k]=='.') ) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } } break; case AlphaNumSpace: { for (sint k = 0; k < length; ++k) { if (isValidAlphaNumSpace(sString[k])) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } } break; case AlphaNum: { for (sint k = 0; k < length; ++k) { if (isValidAlphaNum(sString[k])) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } } break; case Alpha: { for (sint k = 0; k < length; ++k) { if (isValidAlpha(sString[k])) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } } break; case Filename: { for (sint k = 0; k < length; ++k) { if (isValidFilenameChar(sString[k])) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } } break; case PlayerName: { for (sint k = 0; k < length; ++k) { if (isValidPlayerNameChar(sString[k])) { if (!isFiltered(sString[k])) { toAppend += sString[k]; } } } } } length = toAppend.size(); if ((uint) (_InputString.length() + length) > _MaxNumChar) { length = _MaxNumChar - _InputString.length(); } ucstring toAdd = toAppend.substr(0, length); _InputString = _InputString.substr(0, _CursorPos) + toAdd + _InputString.substr(_CursorPos); _CursorPos += toAdd.length(); GlobalUnlock (hObj); nlinfo ("Chat input was pasted from the clipboard"); } } CloseClipboard (); triggerOnChangeAH(); } #endif // NL_OS_WINDOWS _CursorAtPreviousLineEnd = false; } // ---------------------------------------------------------------------------- void CGroupEditBox::handleEventChar(const CEventDescriptorKey &rEDK) { stopParentBlink(); switch(rEDK.getChar()) { case KeyESCAPE: _CurrentHistoricIndex= -1; CInterfaceManager::getInstance()->setCaptureKeyboard(NULL); // stop selection _CurrSelection = NULL; _CursorAtPreviousLineEnd = false; break; case KeyTAB: makeTopWindow(); break; // OTHER default: CInterfaceManager *pIM = CInterfaceManager::getInstance(); if ((rEDK.getChar() == KeyRETURN) && !_WantReturn) { // update historic. if(_MaxHistoric) { if( !_InputString.empty() ) { _Historic.push_front(_InputString); if(_Historic.size()>_MaxHistoric) _Historic.pop_back(); } _CurrentHistoricIndex= -1; } _CursorPos = 0; // loose the keyboard focus CInterfaceManager *im = CInterfaceManager::getInstance(); if (im->getDbProp("UI:SAVE:CHAT:ENTER_DONT_QUIT_CB")->getValue32() == 0) { if(_LooseFocusOnEnter) pIM->setCaptureKeyboard(NULL); } // stop selection _CurrSelection = NULL; _CursorAtPreviousLineEnd = false; pIM->runActionHandler(_AHOnEnter, this, _AHOnEnterParams); } else { // If the char is not alphanumeric -> return. // if(!isalnum(ec.Char)) // return if( (rEDK.getChar()>=32) || (rEDK.getChar() == KeyRETURN) ) { if (rEDK.getChar() == KeyRETURN) { ucstring copyStr= _InputString; if ((uint) std::count(copyStr.begin(), copyStr.end(), '\n') >= _MaxNumReturn) break; } // if selection is activated, then cut the selection if(_CurrSelection != NULL) { if (_CurrSelection != this) { nlwarning("Selection can only be on focus"); } cutSelection(); } ucchar c = (rEDK.getChar() == KeyRETURN)?'\n':rEDK.getChar(); if (isFiltered(c)) return; switch(_EntryType) { case Integer: if (c > 255 || !(c =='-' || c==' ' || isdigit((int) c))) return; break; case PositiveInteger: if (c > 255 || !(c == ' ' || isdigit((int) c))) return; break; case Float: if (c > 255 || !(c =='-' || c==' ' || c=='.' || isdigit((int) c))) return; break; case PositiveFloat: if (c > 255 || !(c == ' ' || c=='.' || isdigit((int) c))) return; break; case AlphaNumSpace: if (!isValidAlphaNumSpace(c)) return; break; case AlphaNum: if (!isValidAlphaNum(c)) return; break; case Alpha: if (!isValidAlpha(c)) return; break; case Filename: if (!isValidFilenameChar(c)) return; break; case PlayerName: if (!isValidPlayerNameChar(c)) return; break; } // verify integer bounds if(_EntryType==Integer && (_IntegerMinValue!=INT_MIN || _IntegerMaxValue!=INT_MAX)) { // estimate new string ucstring copyStr= _InputString; ucstring::iterator it = copyStr.begin() + _CursorPos; copyStr.insert(it, c); sint32 value; fromString(copyStr.toString(), value); // if out of bounds, abort char if(value<_IntegerMinValue || value>_IntegerMaxValue) return; } // verify integer bounds if(_EntryType==PositiveInteger && (_PositiveIntegerMinValue!=0 || _PositiveIntegerMaxValue!=UINT_MAX)) { // estimate new string ucstring copyStr= _InputString; ucstring::iterator it = copyStr.begin() + _CursorPos; copyStr.insert(it, c); // \todo yoyo: this doesn't really work i think.... uint32 value; fromString(copyStr.toString(), value); // if out of bounds, abort char if(value<_PositiveIntegerMinValue || value>_PositiveIntegerMaxValue) return; } // verify max num char if ((uint) _InputString.length() < _MaxNumChar) { makeTopWindow(); ucstring::iterator it = _InputString.begin() + _CursorPos; _InputString.insert(it, c); ++ _CursorPos; triggerOnChangeAH(); } if (rEDK.getChar() == KeyRETURN) { pIM->runActionHandler(_AHOnEnter, this, _AHOnEnterParams); } } _CursorAtPreviousLineEnd = false; } break; } } // ---------------------------------------------------------------------------- bool CGroupEditBox::undo() { CInterfaceManager *im = CInterfaceManager::getInstance(); if (im->getCaptureKeyboard() != this) return false; if (!_CanUndo) return false; _ModifiedInputString = _InputString; setInputString(_StartInputString); _CanUndo = false; _CanRedo = true; setCursorPos(_InputString.length()); setSelectionAll(); return true; } // ---------------------------------------------------------------------------- bool CGroupEditBox::redo() { CInterfaceManager *im = CInterfaceManager::getInstance(); if (im->getCaptureKeyboard() != this) return false; if (!_CanRedo) return false; setInputString(_ModifiedInputString); _CanUndo = true; _CanRedo = false; setCursorPos(_InputString.length()); setSelectionAll(); return true; } // ---------------------------------------------------------------------------- void CGroupEditBox::triggerOnChangeAH() { _CanUndo = true; _CanRedo = false; if (!_AHOnChange.empty()) { CInterfaceManager *pIM = CInterfaceManager::getInstance(); pIM->runActionHandler(_AHOnChange, this, _ParamsOnChange); } } // ---------------------------------------------------------------------------- bool CGroupEditBox::expand() { if ((_EntryType == Integer) || (_EntryType == PositiveInteger) || (_EntryType == Float) || (_EntryType == PositiveFloat) || (_EntryType == AlphaNumSpace) || (_EntryType == AlphaNum) || (_EntryType == Alpha) || (_EntryType == Filename) || (_EntryType == PlayerName) ) return false; if(!_InputString.empty()) { if (_InputString[0] == '/') { makeTopWindow(); // for french or deutch, be aware of unicode std::string command = ucstring(_InputString.substr(1)).toUtf8(); ICommand::expand(command); // then back to ucstring _InputString.fromUtf8(command); _InputString = '/' + _InputString; _CursorPos = _InputString.length(); _CursorAtPreviousLineEnd = false; triggerOnChangeAH(); return true; } } return false; } // ---------------------------------------------------------------------------- void CGroupEditBox::back() { makeTopWindow(); // if selection is activated and not same cursors pos, then cut the selection if(_CurrSelection != NULL && _CursorPos != _SelectCursorPos) { if (_CurrSelection != this) { nlwarning("Selection can only be on focus"); } cutSelection(); _CursorAtPreviousLineEnd = false; } // else delete last character else if(_InputString.size () > 0 && _CursorPos != 0) { ucstring::iterator it = _InputString.begin() + (_CursorPos - 1); _InputString.erase(it); -- _CursorPos; _CursorAtPreviousLineEnd = false; triggerOnChangeAH(); } // must stop selection in all case if (_CurrSelection) { if (_CurrSelection != this) { nlwarning("Selection can only be on focus"); } _CurrSelection = NULL; } } // ---------------------------------------------------------------------------- bool CGroupEditBox::handleEvent (const CEventDescriptor& event) { if (!_Active || !_ViewText) return false; if (event.getType() == CEventDescriptor::key) { if (_BypassNextKey) { _BypassNextKey = false; return true; } /////////////// // KEY EVENT // /////////////// const CEventDescriptorKey &rEDK = (const CEventDescriptorKey&)event; switch(rEDK.getKeyEventType()) { case CEventDescriptorKey::keychar: handleEventChar(rEDK); break; } // update the text setInputString(_InputString); // if event of type char, consider handle all of them if( rEDK.getKeyEventType()==CEventDescriptorKey::keychar ) return true; // Else filter the EventKeyDown AND EventKeyUp. else { // Look into the input handler Manager if the key combo has to be considered as handled CInputHandlerManager *pIH= CInputHandlerManager::getInstance(); if( pIH->isComboKeyChat(rEDK) ) return true; else return false; } } else if (event.getType() == CEventDescriptor::mouse) { ///////////////// // MOUSE EVENT // ///////////////// const CEventDescriptorMouse &eventDesc = (const CEventDescriptorMouse &)event; CInterfaceManager *pIM = CInterfaceManager::getInstance(); if (eventDesc.getEventTypeExtended() == CEventDescriptorMouse::mouserightup) { if (pIM->getCapturePointerRight() == this) { pIM->setCapturePointerRight(NULL); if (!_ListMenuRight.empty()) { if (CDBCtrlSheet::getDraggedSheet() == NULL) { _MenuFather = this; pIM->enableModalWindow (this, _ListMenuRight); stopParentBlink(); } } return true; } else { return false; } } if (!isIn(eventDesc.getX(), eventDesc.getY())) return false; // if click, and not frozen, then get the focus if (eventDesc.getEventTypeExtended() == CEventDescriptorMouse::mouseleftdown && !_Frozen) { stopParentBlink(); pIM->setCaptureKeyboard (this); // set the right cursor position uint newCurPos; bool cursorAtPreviousLineEnd; _ViewText->getCharacterIndexFromPosition(eventDesc.getX() - _ViewText->getXReal(), eventDesc.getY() - _ViewText->getYReal(), newCurPos, cursorAtPreviousLineEnd); _CursorAtPreviousLineEnd = cursorAtPreviousLineEnd; _CursorPos = newCurPos; _CursorPos -= _Prompt.length(); _CursorPos = std::max(_CursorPos, sint32(0)); return true; } if (eventDesc.getEventTypeExtended() == CEventDescriptorMouse::mouserightdown) { pIM->setCapturePointerRight(this); return true; } } else { ////////////////// // SYSTEM EVENT // ////////////////// const CEventDescriptorSystem &eventDesc = (const CEventDescriptorSystem &)event; if (eventDesc.getEventTypeExtended() == CEventDescriptorSystem::activecalledonparent) { CEventDescriptorActiveCalledOnParent &activeEvent = (CEventDescriptorActiveCalledOnParent &) eventDesc; if (activeEvent.getActive() == false && _ResetFocusOnHide) { CInterfaceManager::getInstance()->resetCaptureKeyboard(); // If a selection was shown, reset it if (_CurrSelection == this) _CurrSelection = NULL; } CInterfaceGroup::handleEvent(activeEvent); } } return false; } // ---------------------------------------------------------------------------- void CGroupEditBox::setupDisplayText() { if (_ViewText) { ucstring usTmp; if (_EntryType == Password) { usTmp = _Prompt; for (uint32 i = 0; i < _InputString.size(); ++i) usTmp += "*"; } else { usTmp = _Prompt + _InputString; } _ViewText->setText (usTmp); } } // ---------------------------------------------------------------------------- void CGroupEditBox::checkCoords() { setupDisplayText(); CInterfaceGroup::checkCoords(); } // ---------------------------------------------------------------------------- void CGroupEditBox::updateCoords() { if (!_Setupped) { setup(); _Setupped = true; } CInterfaceGroup::updateCoords(); if (_ViewText) { bool bRecalc = false; // if the length of the view text exceed the max w we allow, cut it while (_ViewText->getWReal() > _MaxCharsSize) { // Suppr last char _InputString = _InputString.substr(0, _InputString.size()-1); setupDisplayText(); _ViewText->updateCoords(); bRecalc = true; } // if the length of the view text exceed our real size, ensure the Cursor position is at least visible sint32 viewTextNewX = _ViewText->getX(); if (_ViewText->getWReal() > _WReal) { // Check if cursor visible sint xCursVT, xCurs, yTmp, hTmp; // Get the cursor pos from the BL of the viewtext _ViewText->getCharacterPositionFromIndex(_CursorPos+_Prompt.size(), false, xCursVT, yTmp, hTmp); // Get the cursor pos from the BL of the edit box xCurs = xCursVT - (_XReal - _ViewText->getXReal()); // If the cursor is outside the edit box move the view text to show the cursor if (xCurs > _WReal) { viewTextNewX = _ViewTextDeltaX - (xCursVT - _WReal); } if (xCurs < 0) { if ((xCursVT + xCurs) < 0) viewTextNewX = _ViewTextDeltaX; else viewTextNewX = _ViewTextDeltaX - (xCursVT + xCurs); } if (_CursorPos == 0) { viewTextNewX = _ViewTextDeltaX; } } else { viewTextNewX = _ViewTextDeltaX; } // if X position changed, must recompute if (viewTextNewX != _ViewText->getX()) { _ViewText->setX(viewTextNewX); bRecalc = true; } // recompute position if (bRecalc) CInterfaceGroup::updateCoords(); } if (_BackupFatherContainerPos) { CGroupContainer *gc = getEnclosingContainer(); if (gc && !gc->getTouchFlag(true)) { if (_ViewText && _ViewText->getNumLine() <= 1) { if (_PrevNumLine > 1) { gc->restorePosition(); CInterfaceGroup::updateCoords(); } } else { if (_PrevNumLine <= 1) { gc->backupPosition(); } } _PrevNumLine = _ViewText->getNumLine(); } else { gc->backupPosition(); } } } // ---------------------------------------------------------------------------- void CGroupEditBox::clearViews() { CInterfaceGroup::clearViews(); } // ---------------------------------------------------------------------------- void CGroupEditBox::setup() { // bind to the controls _ViewText = dynamic_cast(CInterfaceGroup::getView("edit_text")); if(_ViewText == NULL) nlwarning("Interface: CGroupEditBox: text 'edit_text' missing or bad type"); // For MultiLine editbox, clip the end space, else weird when edit space at end of line (nothing happens) if(_ViewText) _ViewText->setMultiLineClipEndSpace(true); // Bakcup the delta X of this view text if(_ViewText) _ViewTextDeltaX= _ViewText->getX(); else _ViewTextDeltaX= 0; // read options CInterfaceOptions *pIO = CInterfaceManager::getInstance()->getOptions("text_selection"); if (pIO != NULL) { _BackSelectColor= pIO->getValColor("back_select_color"); _TextSelectColor= pIO->getValColor("text_select_color"); } } // ---------------------------------------------------------------------------- void CGroupEditBox::setInputString(const ucstring &str) { _InputString = str; if (_CursorPos > (sint32) str.length()) { _CursorPos = str.length(); _CursorAtPreviousLineEnd = false; } if (!_ViewText) return; setupDisplayText(); invalidateCoords(); } // *************************************************************************** void CGroupEditBox::setDefaultInputString(const ucstring &str) { _DefaultInputString= true; setInputString(str); } // *************************************************************************** sint32 CGroupEditBox::getInputStringAsInt() const { sint32 value; fromString(_InputString.toString(), value); return value; } // *************************************************************************** void CGroupEditBox::setInputStringAsInt(sint32 val) { setInputString(NLMISC::toString(val)); } // *************************************************************************** sint64 CGroupEditBox::getInputStringAsInt64() const { sint32 value; fromString(_InputString.toString(), value); return value; } // *************************************************************************** void CGroupEditBox::setInputStringAsInt64(sint64 val) { setInputString(NLMISC::toString(val)); } // *************************************************************************** float CGroupEditBox::getInputStringAsFloat() const { float value; fromString(_InputString.toString(), value); return value; } // *************************************************************************** void CGroupEditBox::setInputStringAsFloat(float val) { string fmt= "%." + NLMISC::toString(_MaxFloatPrec) + "f"; setInputString(NLMISC::toString(fmt.c_str(), val)); } // *************************************************************************** void CGroupEditBox::cutSelection() { sint32 minPos= min(_CursorPos, _SelectCursorPos); sint32 maxPos= max(_CursorPos, _SelectCursorPos); // cut the selection if(!_InputString.empty()) { _InputString= _InputString.substr(0, minPos) + _InputString.substr(maxPos); } _CurrSelection = NULL; _CursorPos= minPos; triggerOnChangeAH(); } // *************************************************************************** ucstring CGroupEditBox::getSelection() { sint32 minPos= min(_CursorPos, _SelectCursorPos); sint32 maxPos= max(_CursorPos, _SelectCursorPos); // get the selection return _InputString.substr(minPos, maxPos-minPos); } // *************************************************************************** void CGroupEditBox::setSelectionAll() { if(!_InputString.empty()) { _CurrSelection = this; _SelectCursorPos= 0; _CursorPos= _InputString.size(); _CursorAtPreviousLineEnd = false; } } // *************************************************************************** void CGroupEditBox::setActive(bool active) { if (!active && _ResetFocusOnHide) { CInterfaceManager::getInstance()->resetCaptureKeyboard(); } CInterfaceGroup::setActive(active); } // *************************************************************************** void CGroupEditBox::setInputStringAsStdString(const std::string &str) { setInputString(ucstring(str)); } // *************************************************************************** std::string CGroupEditBox::getInputStringAsStdString() const { std::string result; _InputString.toString(result); return result; } // *************************************************************************** void CGroupEditBox::setInputStringAsUtf8(const std::string &str) { ucstring tmp; tmp.fromUtf8(str); setInputString(tmp); } // *************************************************************************** std::string CGroupEditBox::getInputStringAsUtf8() const { return _InputString.toUtf8(); } // *************************************************************************** void CGroupEditBox::setCommand(const ucstring &command, bool execute) { // do nothing if frozen if(_Frozen) return; // set the string and maybe execute setInputString((ucchar) '/' + command); if (execute) { // stop selection _CurrSelection = NULL; _CursorAtPreviousLineEnd = false; CInterfaceManager::getInstance()->runActionHandler(_AHOnEnter, this, _AHOnEnterParams); } else { CInterfaceManager::getInstance()->setCaptureKeyboard (this); _CursorPos = _InputString.length(); } } // *************************************************************************** void CGroupEditBox::makeTopWindow() { CInterfaceManager *im = CInterfaceManager::getInstance(); CInterfaceGroup *root = getRootWindow(); if (root) { im->setTopWindow(root); } } // *************************************************************************** void CGroupEditBox::clearAllEditBox() { _InputString = ""; _CursorPos = 0; _CursorAtPreviousLineEnd = false; if (!_ViewText) return; setupDisplayText(); invalidateCoords(); } // *************************************************************************** sint32 CGroupEditBox::getMaxUsedW() const { return _W; } // *************************************************************************** sint32 CGroupEditBox::getMinUsedW() const { return _W; } // *************************************************************************** bool CGroupEditBox::wantSerialConfig() const { return _Savable && !_InputString.empty(); } // *************************************************************************** void CGroupEditBox::serialConfig(NLMISC::IStream &f) { f.serialVersion(0); if(_DefaultInputString) // Don't want to save the default input { _DefaultInputString= false; _InputString.clear(); } f.serial(_InputString); f.serial(_CursorPos); f.serial(_PrevNumLine); if (f.isReading()) { setInputString(_InputString); } // serial selection bool isSelected = (_CurrSelection == this); f.serial(isSelected); if (isSelected) { _CurrSelection = this; f.serial(_SelectCursorPos); } } // *************************************************************************** void CGroupEditBox::onQuit() { // clear the text and restore backup pos before final save setInputString(ucstring("")); _CurrSelection = NULL; } // *************************************************************************** void CGroupEditBox::onLoadConfig() { // config is not saved when there's an empty string, so restore that default state. setInputString(ucstring("")); _CurrSelection = NULL; _PrevNumLine = 1; } // *************************************************************************** void CGroupEditBox::elementCaptured(CCtrlBase *capturedElement) { // If the input string is the default one, then reset it if(capturedElement == this) { if (_DefaultInputString) { _DefaultInputString= false; setInputString(ucstring()); } _CanRedo = false; _CanUndo = false; _StartInputString = _ModifiedInputString = _InputString; } CInterfaceGroup::elementCaptured(capturedElement); } // *************************************************************************** void CGroupEditBox::onKeyboardCaptureLost() { if (!_AHOnFocusLost.empty()) { CInterfaceManager *im = CInterfaceManager::getInstance(); im->runActionHandler(_AHOnFocusLost, this, _AHOnFocusLostParams); } } // *************************************************************************** int CGroupEditBox::luaSetSelectionAll(CLuaState &ls) { const char *funcName = "setSelectionAll"; CLuaIHM::checkArgCount(ls, funcName, 0); setSelectionAll(); return 0; } // *************************************************************************** void CGroupEditBox::setFocusOnText() { // do nothing if frozen if(_Frozen) return; // else set the focus CInterfaceManager *pIM = CInterfaceManager::getInstance(); pIM->setCaptureKeyboard (this); _CurrSelection = this; _SelectCursorPos= _InputString.size(); _CursorPos= _InputString.size(); _CursorAtPreviousLineEnd = false; } // *************************************************************************** int CGroupEditBox::luaSetFocusOnText(CLuaState &ls) { const char *funcName = "setFocusOnText"; CLuaIHM::checkArgCount(ls, funcName, 0); setFocusOnText(); return 0; } // *************************************************************************** int CGroupEditBox::luaCancelFocusOnText(CLuaState &ls) { const char *funcName = "cancelFocusOnText"; CLuaIHM::checkArgCount(ls, funcName, 0); CInterfaceManager *pIM = CInterfaceManager::getInstance(); if (pIM->getCaptureKeyboard()==this || pIM->getOldCaptureKeyboard()==this) pIM->resetCaptureKeyboard(); _CurrSelection = NULL; _SelectCursorPos= 0; _CursorPos= 0; _CursorAtPreviousLineEnd = false; return 0; } // *************************************************************************** int CGroupEditBox::luaSetupDisplayText(CLuaState &ls) { const char *funcName = "setupDisplayText"; CLuaIHM::checkArgCount(ls, funcName, 0); setupDisplayText(); return 0; } // *************************************************************************** void CGroupEditBox::setColor(NLMISC::CRGBA col) { if (_ViewText) _ViewText->setColor(col); } // *************************************************************************** void CGroupEditBox::setFrozen (bool state) { _Frozen= state; // if frozen, loose the focus if(_Frozen) { CInterfaceManager *pIM= CInterfaceManager::getInstance(); // stop capture and selection pIM->setCaptureKeyboard (NULL); if(_CurrSelection==this) _CurrSelection = NULL; // do not allow to recover focus if (pIM->getOldCaptureKeyboard() == this) { pIM->resetCaptureKeyboard(); } } }