diff options
Diffstat (limited to 'linden/indra/llui/lltexteditor.cpp')
-rw-r--r-- | linden/indra/llui/lltexteditor.cpp | 144 |
1 files changed, 69 insertions, 75 deletions
diff --git a/linden/indra/llui/lltexteditor.cpp b/linden/indra/llui/lltexteditor.cpp index e56002f..281dbe4 100644 --- a/linden/indra/llui/lltexteditor.cpp +++ b/linden/indra/llui/lltexteditor.cpp | |||
@@ -260,7 +260,7 @@ LLTextEditor::LLTextEditor( | |||
260 | mIsSelecting( FALSE ), | 260 | mIsSelecting( FALSE ), |
261 | mSelectionStart( 0 ), | 261 | mSelectionStart( 0 ), |
262 | mSelectionEnd( 0 ), | 262 | mSelectionEnd( 0 ), |
263 | mScrolledToBottom( FALSE ), | 263 | mScrolledToBottom( TRUE ), |
264 | mOnScrollEndCallback( NULL ), | 264 | mOnScrollEndCallback( NULL ), |
265 | mOnScrollEndData( NULL ), | 265 | mOnScrollEndData( NULL ), |
266 | mCursorColor( LLUI::sColorsGroup->getColor( "TextCursorColor" ) ), | 266 | mCursorColor( LLUI::sColorsGroup->getColor( "TextCursorColor" ) ), |
@@ -277,14 +277,16 @@ LLTextEditor::LLTextEditor( | |||
277 | mCommitOnFocusLost( FALSE ), | 277 | mCommitOnFocusLost( FALSE ), |
278 | mHideScrollbarForShortDocs( FALSE ), | 278 | mHideScrollbarForShortDocs( FALSE ), |
279 | mTakesNonScrollClicks( TRUE ), | 279 | mTakesNonScrollClicks( TRUE ), |
280 | mTrackBottom( TRUE ), | 280 | mTrackBottom( FALSE ), |
281 | mAllowEmbeddedItems( allow_embedded_items ), | 281 | mAllowEmbeddedItems( allow_embedded_items ), |
282 | mAcceptCallingCardNames(FALSE), | 282 | mAcceptCallingCardNames(FALSE), |
283 | mHandleEditKeysDirectly( FALSE ), | 283 | mHandleEditKeysDirectly( FALSE ), |
284 | mMouseDownX(0), | 284 | mMouseDownX(0), |
285 | mMouseDownY(0), | 285 | mMouseDownY(0), |
286 | mLastSelectionX(-1), | 286 | mLastSelectionX(-1), |
287 | mLastSelectionY(-1) | 287 | mLastSelectionY(-1), |
288 | mReflowNeeded(FALSE), | ||
289 | mScrollNeeded(FALSE) | ||
288 | { | 290 | { |
289 | mSourceID.generate(); | 291 | mSourceID.generate(); |
290 | 292 | ||
@@ -468,6 +470,13 @@ void LLTextEditor::updateLineStartList(S32 startpos) | |||
468 | mScrollbar->setVisible(!short_doc); | 470 | mScrollbar->setVisible(!short_doc); |
469 | } | 471 | } |
470 | 472 | ||
473 | // if scrolled to bottom, stay at bottom | ||
474 | // unless user is editing text | ||
475 | // do this after updating page size | ||
476 | if (mScrolledToBottom && mTrackBottom && !hasFocus()) | ||
477 | { | ||
478 | endOfDoc(); | ||
479 | } | ||
471 | } | 480 | } |
472 | 481 | ||
473 | //////////////////////////////////////////////////////////// | 482 | //////////////////////////////////////////////////////////// |
@@ -511,8 +520,7 @@ void LLTextEditor::setText(const LLStringExplicit &utf8str) | |||
511 | setCursorPos(0); | 520 | setCursorPos(0); |
512 | deselect(); | 521 | deselect(); |
513 | 522 | ||
514 | updateLineStartList(); | 523 | needsReflow(); |
515 | updateScrollFromCursor(); | ||
516 | 524 | ||
517 | resetDirty(); | 525 | resetDirty(); |
518 | } | 526 | } |
@@ -529,8 +537,7 @@ void LLTextEditor::setWText(const LLWString &wtext) | |||
529 | setCursorPos(0); | 537 | setCursorPos(0); |
530 | deselect(); | 538 | deselect(); |
531 | 539 | ||
532 | updateLineStartList(); | 540 | needsReflow(); |
533 | updateScrollFromCursor(); | ||
534 | 541 | ||
535 | resetDirty(); | 542 | resetDirty(); |
536 | } | 543 | } |
@@ -568,8 +575,7 @@ void LLTextEditor::setWordWrap(BOOL b) | |||
568 | setCursorPos(0); | 575 | setCursorPos(0); |
569 | deselect(); | 576 | deselect(); |
570 | 577 | ||
571 | updateLineStartList(); | 578 | needsReflow(); |
572 | updateScrollFromCursor(); | ||
573 | } | 579 | } |
574 | 580 | ||
575 | 581 | ||
@@ -734,6 +740,7 @@ S32 LLTextEditor::getLineStart( S32 line ) const | |||
734 | { | 740 | { |
735 | return 0; | 741 | return 0; |
736 | } | 742 | } |
743 | |||
737 | line = llclamp(line, 0, num_lines-1); | 744 | line = llclamp(line, 0, num_lines-1); |
738 | S32 segidx = mLineStartList[line].mSegment; | 745 | S32 segidx = mLineStartList[line].mSegment; |
739 | S32 segoffset = mLineStartList[line].mOffset; | 746 | S32 segoffset = mLineStartList[line].mOffset; |
@@ -781,14 +788,14 @@ void LLTextEditor::getSegmentAndOffset( S32 startpos, S32* segidxp, S32* offsetp | |||
781 | *offsetp = startpos - (*seg_iter)->getStart(); | 788 | *offsetp = startpos - (*seg_iter)->getStart(); |
782 | } | 789 | } |
783 | 790 | ||
784 | const LLTextSegment* LLTextEditor::getPreviousSegment() | 791 | const LLTextSegment* LLTextEditor::getPreviousSegment() const |
785 | { | 792 | { |
786 | // find segment index at character to left of cursor (or rightmost edge of selection) | 793 | // find segment index at character to left of cursor (or rightmost edge of selection) |
787 | S32 idx = llmax(0, getSegmentIdxAtOffset(mCursorPos) - 1); | 794 | S32 idx = llmax(0, getSegmentIdxAtOffset(mCursorPos) - 1); |
788 | return idx >= 0 ? mSegments[idx] : NULL; | 795 | return idx >= 0 ? mSegments[idx] : NULL; |
789 | } | 796 | } |
790 | 797 | ||
791 | void LLTextEditor::getSelectedSegments(std::vector<const LLTextSegment*>& segments) | 798 | void LLTextEditor::getSelectedSegments(std::vector<const LLTextSegment*>& segments) const |
792 | { | 799 | { |
793 | S32 left = hasSelection() ? llmin(mSelectionStart, mSelectionEnd) : mCursorPos; | 800 | S32 left = hasSelection() ? llmin(mSelectionStart, mSelectionEnd) : mCursorPos; |
794 | S32 right = hasSelection() ? llmax(mSelectionStart, mSelectionEnd) : mCursorPos; | 801 | S32 right = hasSelection() ? llmax(mSelectionStart, mSelectionEnd) : mCursorPos; |
@@ -875,13 +882,12 @@ void LLTextEditor::setCursor(S32 row, S32 column) | |||
875 | } | 882 | } |
876 | doc += column; | 883 | doc += column; |
877 | setCursorPos(doc - mWText.c_str()); | 884 | setCursorPos(doc - mWText.c_str()); |
878 | updateScrollFromCursor(); | ||
879 | } | 885 | } |
880 | 886 | ||
881 | void LLTextEditor::setCursorPos(S32 offset) | 887 | void LLTextEditor::setCursorPos(S32 offset) |
882 | { | 888 | { |
883 | mCursorPos = llclamp(offset, 0, (S32)getLength()); | 889 | mCursorPos = llclamp(offset, 0, (S32)getLength()); |
884 | updateScrollFromCursor(); | 890 | needsScroll(); |
885 | // reset desired x cursor position | 891 | // reset desired x cursor position |
886 | mDesiredXPixel = -1; | 892 | mDesiredXPixel = -1; |
887 | } | 893 | } |
@@ -1222,8 +1228,6 @@ BOOL LLTextEditor::handleHover(S32 x, S32 y, MASK mask) | |||
1222 | 1228 | ||
1223 | setCursorAtLocalPos( x, y, TRUE ); | 1229 | setCursorAtLocalPos( x, y, TRUE ); |
1224 | mSelectionEnd = mCursorPos; | 1230 | mSelectionEnd = mCursorPos; |
1225 | |||
1226 | updateScrollFromCursor(); | ||
1227 | } | 1231 | } |
1228 | 1232 | ||
1229 | lldebugst(LLERR_USER_INPUT) << "hover handled by " << getName() << " (active)" << llendl; | 1233 | lldebugst(LLERR_USER_INPUT) << "hover handled by " << getName() << " (active)" << llendl; |
@@ -1317,8 +1321,6 @@ BOOL LLTextEditor::handleMouseUp(S32 x, S32 y, MASK mask) | |||
1317 | 1321 | ||
1318 | setCursorAtLocalPos( x, y, TRUE ); | 1322 | setCursorAtLocalPos( x, y, TRUE ); |
1319 | endSelection(); | 1323 | endSelection(); |
1320 | |||
1321 | updateScrollFromCursor(); | ||
1322 | } | 1324 | } |
1323 | 1325 | ||
1324 | if( !hasSelection() ) | 1326 | if( !hasSelection() ) |
@@ -1839,8 +1841,7 @@ void LLTextEditor::cut() | |||
1839 | gClipboard.copyFromSubstring( mWText, left_pos, length, mSourceID ); | 1841 | gClipboard.copyFromSubstring( mWText, left_pos, length, mSourceID ); |
1840 | deleteSelection( FALSE ); | 1842 | deleteSelection( FALSE ); |
1841 | 1843 | ||
1842 | updateLineStartList(); | 1844 | needsReflow(); |
1843 | updateScrollFromCursor(); | ||
1844 | } | 1845 | } |
1845 | 1846 | ||
1846 | BOOL LLTextEditor::canCopy() const | 1847 | BOOL LLTextEditor::canCopy() const |
@@ -1910,8 +1911,7 @@ void LLTextEditor::paste() | |||
1910 | setCursorPos(mCursorPos + insert(mCursorPos, clean_string, FALSE)); | 1911 | setCursorPos(mCursorPos + insert(mCursorPos, clean_string, FALSE)); |
1911 | deselect(); | 1912 | deselect(); |
1912 | 1913 | ||
1913 | updateLineStartList(); | 1914 | needsReflow(); |
1914 | updateScrollFromCursor(); | ||
1915 | } | 1915 | } |
1916 | 1916 | ||
1917 | 1917 | ||
@@ -2235,9 +2235,9 @@ BOOL LLTextEditor::handleKeyHere(KEY key, MASK mask ) | |||
2235 | 2235 | ||
2236 | if(text_may_have_changed) | 2236 | if(text_may_have_changed) |
2237 | { | 2237 | { |
2238 | updateLineStartList(); | 2238 | needsReflow(); |
2239 | } | 2239 | } |
2240 | updateScrollFromCursor(); | 2240 | needsScroll(); |
2241 | } | 2241 | } |
2242 | } | 2242 | } |
2243 | 2243 | ||
@@ -2280,8 +2280,7 @@ BOOL LLTextEditor::handleUnicodeCharHere(llwchar uni_char) | |||
2280 | // Most keystrokes will make the selection box go away, but not all will. | 2280 | // Most keystrokes will make the selection box go away, but not all will. |
2281 | deselect(); | 2281 | deselect(); |
2282 | 2282 | ||
2283 | updateLineStartList(); | 2283 | needsReflow(); |
2284 | updateScrollFromCursor(); | ||
2285 | } | 2284 | } |
2286 | } | 2285 | } |
2287 | 2286 | ||
@@ -2339,8 +2338,7 @@ void LLTextEditor::doDelete() | |||
2339 | } | 2338 | } |
2340 | } | 2339 | } |
2341 | 2340 | ||
2342 | updateLineStartList(); | 2341 | needsReflow(); |
2343 | updateScrollFromCursor(); | ||
2344 | } | 2342 | } |
2345 | 2343 | ||
2346 | //---------------------------------------------------------------------------- | 2344 | //---------------------------------------------------------------------------- |
@@ -2383,8 +2381,7 @@ void LLTextEditor::undo() | |||
2383 | 2381 | ||
2384 | setCursorPos(pos); | 2382 | setCursorPos(pos); |
2385 | 2383 | ||
2386 | updateLineStartList(); | 2384 | needsReflow(); |
2387 | updateScrollFromCursor(); | ||
2388 | } | 2385 | } |
2389 | 2386 | ||
2390 | BOOL LLTextEditor::canRedo() const | 2387 | BOOL LLTextEditor::canRedo() const |
@@ -2426,8 +2423,7 @@ void LLTextEditor::redo() | |||
2426 | 2423 | ||
2427 | setCursorPos(pos); | 2424 | setCursorPos(pos); |
2428 | 2425 | ||
2429 | updateLineStartList(); | 2426 | needsReflow(); |
2430 | updateScrollFromCursor(); | ||
2431 | } | 2427 | } |
2432 | 2428 | ||
2433 | void LLTextEditor::onFocusReceived() | 2429 | void LLTextEditor::onFocusReceived() |
@@ -2594,7 +2590,7 @@ void LLTextEditor::drawSelectionBackground() | |||
2594 | BOOL selection_visible = (left_visible_pos <= selection_right) && (selection_left <= right_visible_pos); | 2590 | BOOL selection_visible = (left_visible_pos <= selection_right) && (selection_left <= right_visible_pos); |
2595 | if( selection_visible ) | 2591 | if( selection_visible ) |
2596 | { | 2592 | { |
2597 | LLGLSNoTexture no_texture; | 2593 | gGL.getTexUnit(0)->unbind(LLTexUnit::TT_TEXTURE); |
2598 | const LLColor4& color = mReadOnly ? mReadOnlyBgColor : mWriteableBgColor; | 2594 | const LLColor4& color = mReadOnly ? mReadOnlyBgColor : mWriteableBgColor; |
2599 | F32 alpha = hasFocus() ? 1.f : 0.5f; | 2595 | F32 alpha = hasFocus() ? 1.f : 0.5f; |
2600 | gGL.color4f( 1.f - color.mV[0], 1.f - color.mV[1], 1.f - color.mV[2], alpha ); | 2596 | gGL.color4f( 1.f - color.mV[0], 1.f - color.mV[1], 1.f - color.mV[2], alpha ); |
@@ -2729,7 +2725,7 @@ void LLTextEditor::drawCursor() | |||
2729 | } | 2725 | } |
2730 | } | 2726 | } |
2731 | 2727 | ||
2732 | LLGLSNoTexture no_texture; | 2728 | gGL.getTexUnit(0)->unbind(LLTexUnit::TT_TEXTURE); |
2733 | 2729 | ||
2734 | gGL.color4fv( mCursorColor.mV ); | 2730 | gGL.color4fv( mCursorColor.mV ); |
2735 | 2731 | ||
@@ -3100,6 +3096,20 @@ void LLTextEditor::drawClippedSegment(const LLWString &text, S32 seg_start, S32 | |||
3100 | 3096 | ||
3101 | void LLTextEditor::draw() | 3097 | void LLTextEditor::draw() |
3102 | { | 3098 | { |
3099 | // do on-demand reflow | ||
3100 | if (mReflowNeeded) | ||
3101 | { | ||
3102 | updateLineStartList(); | ||
3103 | mReflowNeeded = FALSE; | ||
3104 | } | ||
3105 | |||
3106 | // then update scroll position, as cursor may have moved | ||
3107 | if (mScrollNeeded) | ||
3108 | { | ||
3109 | updateScrollFromCursor(); | ||
3110 | mScrollNeeded = FALSE; | ||
3111 | } | ||
3112 | |||
3103 | { | 3113 | { |
3104 | LLLocalClipRect clip(LLRect(0, getRect().getHeight(), getRect().getWidth() - (mScrollbar->getVisible() ? SCROLLBAR_SIZE : 0), 0)); | 3114 | LLLocalClipRect clip(LLRect(0, getRect().getHeight(), getRect().getWidth() - (mScrollbar->getVisible() ? SCROLLBAR_SIZE : 0), 0)); |
3105 | 3115 | ||
@@ -3118,10 +3128,10 @@ void LLTextEditor::draw() | |||
3118 | mBorder->setKeyboardFocusHighlight( gFocusMgr.getKeyboardFocus() == this);// && !mReadOnly); | 3128 | mBorder->setKeyboardFocusHighlight( gFocusMgr.getKeyboardFocus() == this);// && !mReadOnly); |
3119 | } | 3129 | } |
3120 | 3130 | ||
3131 | LLView::draw(); // Draw children (scrollbar and border) | ||
3132 | |||
3121 | // remember if we are supposed to be at the bottom of the buffer | 3133 | // remember if we are supposed to be at the bottom of the buffer |
3122 | mScrolledToBottom = isScrolledToBottom(); | 3134 | mScrolledToBottom = isScrolledToBottom(); |
3123 | |||
3124 | LLView::draw(); // Draw children (scrollbar and border) | ||
3125 | } | 3135 | } |
3126 | 3136 | ||
3127 | 3137 | ||
@@ -3311,7 +3321,7 @@ void LLTextEditor::setCursorAndScrollToEnd() | |||
3311 | { | 3321 | { |
3312 | deselect(); | 3322 | deselect(); |
3313 | endOfDoc(); | 3323 | endOfDoc(); |
3314 | updateScrollFromCursor(); | 3324 | needsScroll(); |
3315 | } | 3325 | } |
3316 | 3326 | ||
3317 | void LLTextEditor::getLineAndColumnForPosition( S32 position, S32* line, S32* col, BOOL include_wordwrap ) | 3327 | void LLTextEditor::getLineAndColumnForPosition( S32 position, S32* line, S32* col, BOOL include_wordwrap ) |
@@ -3438,7 +3448,7 @@ void LLTextEditor::reshape(S32 width, S32 height, BOOL called_from_parent) | |||
3438 | // up-to-date mTextRect | 3448 | // up-to-date mTextRect |
3439 | updateTextRect(); | 3449 | updateTextRect(); |
3440 | 3450 | ||
3441 | updateLineStartList(); | 3451 | needsReflow(); |
3442 | 3452 | ||
3443 | // propagate shape information to scrollbar | 3453 | // propagate shape information to scrollbar |
3444 | mScrollbar->setDocSize( getLineCount() ); | 3454 | mScrollbar->setDocSize( getLineCount() ); |
@@ -3446,14 +3456,6 @@ void LLTextEditor::reshape(S32 width, S32 height, BOOL called_from_parent) | |||
3446 | S32 line_height = llround( mGLFont->getLineHeight() ); | 3456 | S32 line_height = llround( mGLFont->getLineHeight() ); |
3447 | S32 page_lines = mTextRect.getHeight() / line_height; | 3457 | S32 page_lines = mTextRect.getHeight() / line_height; |
3448 | mScrollbar->setPageSize( page_lines ); | 3458 | mScrollbar->setPageSize( page_lines ); |
3449 | |||
3450 | // if scrolled to bottom, stay at bottom | ||
3451 | // unless user is editing text | ||
3452 | // do this after updating page size | ||
3453 | if (mScrolledToBottom && mTrackBottom && !hasFocus()) | ||
3454 | { | ||
3455 | endOfDoc(); | ||
3456 | } | ||
3457 | } | 3459 | } |
3458 | 3460 | ||
3459 | void LLTextEditor::autoIndent() | 3461 | void LLTextEditor::autoIndent() |
@@ -3500,8 +3502,7 @@ void LLTextEditor::insertText(const std::string &new_text) | |||
3500 | 3502 | ||
3501 | setCursorPos(mCursorPos + insert( mCursorPos, utf8str_to_wstring(new_text), FALSE )); | 3503 | setCursorPos(mCursorPos + insert( mCursorPos, utf8str_to_wstring(new_text), FALSE )); |
3502 | 3504 | ||
3503 | updateLineStartList(); | 3505 | needsReflow(); |
3504 | updateScrollFromCursor(); | ||
3505 | 3506 | ||
3506 | setEnabled( enabled ); | 3507 | setEnabled( enabled ); |
3507 | } | 3508 | } |
@@ -3600,7 +3601,7 @@ void LLTextEditor::appendText(const std::string &new_text, bool allow_undo, bool | |||
3600 | mSegments.push_back(segment); | 3601 | mSegments.push_back(segment); |
3601 | } | 3602 | } |
3602 | 3603 | ||
3603 | updateLineStartList(old_length); | 3604 | needsReflow(); |
3604 | 3605 | ||
3605 | // Set the cursor and scroll position | 3606 | // Set the cursor and scroll position |
3606 | // Maintain the scroll position unless the scroll was at the end of the doc (in which | 3607 | // Maintain the scroll position unless the scroll was at the end of the doc (in which |
@@ -3639,14 +3640,6 @@ void LLTextEditor::appendText(const std::string &new_text, bool allow_undo, bool | |||
3639 | { | 3640 | { |
3640 | blockUndo(); | 3641 | blockUndo(); |
3641 | } | 3642 | } |
3642 | |||
3643 | // if scrolled to bottom, stay at bottom | ||
3644 | // unless user is editing text | ||
3645 | // do this after updating page size | ||
3646 | if (mScrolledToBottom && mTrackBottom && !hasFocus()) | ||
3647 | { | ||
3648 | endOfDoc(); | ||
3649 | } | ||
3650 | } | 3643 | } |
3651 | 3644 | ||
3652 | void LLTextEditor::removeTextFromEnd(S32 num_chars) | 3645 | void LLTextEditor::removeTextFromEnd(S32 num_chars) |
@@ -3661,7 +3654,10 @@ void LLTextEditor::removeTextFromEnd(S32 num_chars) | |||
3661 | mSelectionEnd = llclamp(mSelectionEnd, 0, len); | 3654 | mSelectionEnd = llclamp(mSelectionEnd, 0, len); |
3662 | 3655 | ||
3663 | pruneSegments(); | 3656 | pruneSegments(); |
3657 | |||
3658 | // pruneSegments will invalidate mLineStartList. | ||
3664 | updateLineStartList(); | 3659 | updateLineStartList(); |
3660 | needsScroll(); | ||
3665 | } | 3661 | } |
3666 | 3662 | ||
3667 | /////////////////////////////////////////////////////////////////// | 3663 | /////////////////////////////////////////////////////////////////// |
@@ -3759,8 +3755,7 @@ BOOL LLTextEditor::tryToRevertToPristineState() | |||
3759 | } | 3755 | } |
3760 | } | 3756 | } |
3761 | 3757 | ||
3762 | updateLineStartList(); | 3758 | needsReflow(); |
3763 | updateScrollFromCursor(); | ||
3764 | } | 3759 | } |
3765 | 3760 | ||
3766 | return isPristine(); // TRUE => success | 3761 | return isPristine(); // TRUE => success |
@@ -3808,6 +3803,7 @@ void LLTextEditor::updateSegments() | |||
3808 | { | 3803 | { |
3809 | findEmbeddedItemSegments(); | 3804 | findEmbeddedItemSegments(); |
3810 | } | 3805 | } |
3806 | |||
3811 | // Make sure we have at least one segment | 3807 | // Make sure we have at least one segment |
3812 | if (mSegments.size() == 1 && mSegments[0]->getIsDefault()) | 3808 | if (mSegments.size() == 1 && mSegments[0]->getIsDefault()) |
3813 | { | 3809 | { |
@@ -3824,6 +3820,7 @@ void LLTextEditor::updateSegments() | |||
3824 | } | 3820 | } |
3825 | 3821 | ||
3826 | // Only effective if text was removed from the end of the editor | 3822 | // Only effective if text was removed from the end of the editor |
3823 | // *NOTE: Using this will invalidate references to mSegments from mLineStartList. | ||
3827 | void LLTextEditor::pruneSegments() | 3824 | void LLTextEditor::pruneSegments() |
3828 | { | 3825 | { |
3829 | S32 len = mWText.length(); | 3826 | S32 len = mWText.length(); |
@@ -4066,9 +4063,7 @@ BOOL LLTextEditor::importBuffer(const char* buffer, S32 length ) | |||
4066 | setCursorPos(0); | 4063 | setCursorPos(0); |
4067 | deselect(); | 4064 | deselect(); |
4068 | 4065 | ||
4069 | updateLineStartList(); | 4066 | needsReflow(); |
4070 | updateScrollFromCursor(); | ||
4071 | |||
4072 | return success; | 4067 | return success; |
4073 | } | 4068 | } |
4074 | 4069 | ||
@@ -4260,35 +4255,35 @@ S32 LLTextEditor::findHTMLToken(const std::string &line, S32 pos, BOOL reverse) | |||
4260 | std::string openers=" \t\n('\"[{<>"; | 4255 | std::string openers=" \t\n('\"[{<>"; |
4261 | std::string closers=" \t\n)'\"]}><;"; | 4256 | std::string closers=" \t\n)'\"]}><;"; |
4262 | 4257 | ||
4263 | S32 m2 = 0; | 4258 | S32 index = 0; |
4264 | S32 retval = 0; | ||
4265 | 4259 | ||
4266 | if (reverse) | 4260 | if (reverse) |
4267 | { | 4261 | { |
4268 | 4262 | for (index=pos; index >= 0; index--) | |
4269 | for (retval=pos; retval >= 0; retval--) | ||
4270 | { | 4263 | { |
4271 | m2 = openers.find(line.substr(retval,1)); | 4264 | char c = line[index]; |
4265 | S32 m2 = openers.find(c); | ||
4272 | if (m2 >= 0) | 4266 | if (m2 >= 0) |
4273 | { | 4267 | { |
4274 | break; | 4268 | return index+1; |
4275 | } | 4269 | } |
4276 | } | 4270 | } |
4277 | return retval+1; | ||
4278 | } | 4271 | } |
4279 | else | 4272 | else |
4280 | { | 4273 | { |
4281 | 4274 | ||
4282 | for (retval=pos; retval<(S32)line.length(); retval++) | 4275 | for (index=pos; index<(S32)line.length(); index++) |
4283 | { | 4276 | { |
4284 | m2 = closers.find(line.substr(retval,1)); | 4277 | char c = line[index]; |
4278 | S32 m2 = closers.find(c); | ||
4285 | if (m2 >= 0) | 4279 | if (m2 >= 0) |
4286 | { | 4280 | { |
4287 | break; | 4281 | return index; |
4288 | } | 4282 | } |
4289 | } | 4283 | } |
4290 | return retval; | 4284 | } |
4291 | } | 4285 | |
4286 | return index; | ||
4292 | } | 4287 | } |
4293 | 4288 | ||
4294 | BOOL LLTextEditor::findHTML(const std::string &line, S32 *begin, S32 *end) const | 4289 | BOOL LLTextEditor::findHTML(const std::string &line, S32 *begin, S32 *end) const |
@@ -4476,9 +4471,8 @@ void LLTextEditor::updatePreedit(const LLWString &preedit_string, | |||
4476 | 4471 | ||
4477 | mPreeditStandouts = preedit_standouts; | 4472 | mPreeditStandouts = preedit_standouts; |
4478 | 4473 | ||
4479 | updateLineStartList(); | 4474 | needsReflow(); |
4480 | setCursorPos(insert_preedit_at + caret_position); | 4475 | setCursorPos(insert_preedit_at + caret_position); |
4481 | // updateScrollFromCursor(); | ||
4482 | 4476 | ||
4483 | // Update of the preedit should be caused by some key strokes. | 4477 | // Update of the preedit should be caused by some key strokes. |
4484 | mKeystrokeTimer.reset(); | 4478 | mKeystrokeTimer.reset(); |