19af9997f99f376d6a71037ee77fbbed7de422ed
[platform/core/uifw/dali-toolkit.git] / dali-toolkit / internal / text / text-controller.cpp
1 /*
2  * Copyright (c) 2021 Samsung Electronics Co., Ltd.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  *
16  */
17
18 // CLASS HEADER
19 #include <dali-toolkit/internal/text/text-controller.h>
20
21 // EXTERNAL INCLUDES
22 #include <dali-toolkit/devel-api/controls/control-depth-index-ranges.h>
23 #include <dali/devel-api/adaptor-framework/window-devel.h>
24 #include <dali/integration-api/debug.h>
25 #include <memory.h>
26 #include <cmath>
27 #include <limits>
28
29 // INTERNAL INCLUDES
30 #include <dali-toolkit/devel-api/text/text-enumerations-devel.h>
31 #include <dali-toolkit/internal/text/text-controller-event-handler.h>
32 #include <dali-toolkit/internal/text/text-controller-impl.h>
33 #include <dali-toolkit/internal/text/text-controller-input-font-handler.h>
34 #include <dali-toolkit/internal/text/text-controller-input-properties.h>
35 #include <dali-toolkit/internal/text/text-controller-placeholder-handler.h>
36 #include <dali-toolkit/internal/text/text-controller-relayouter.h>
37 #include <dali-toolkit/internal/text/text-controller-text-updater.h>
38 #include <dali-toolkit/internal/text/text-editable-control-interface.h>
39
40 namespace
41 {
42 #if defined(DEBUG_ENABLED)
43 Debug::Filter* gLogFilter = Debug::Filter::New(Debug::NoLogging, true, "LOG_TEXT_CONTROLS");
44 #endif
45
46 constexpr float MAX_FLOAT = std::numeric_limits<float>::max();
47
48 const std::string EMPTY_STRING("");
49
50 template<typename Type>
51 void EnsureCreated(Type*& object)
52 {
53   if(!object)
54   {
55     object = new Type();
56   }
57 }
58
59 template<typename Type>
60 void EnsureCreated(std::unique_ptr<Type>& object)
61 {
62   if(!object)
63   {
64     object = std::unique_ptr<Type>(new Type());
65   }
66 }
67
68 template<typename Type, typename Arg1>
69 void EnsureCreated(Type*& object, Arg1 arg1)
70 {
71   if(!object)
72   {
73     object = new Type(arg1);
74   }
75 }
76
77 template<typename Type, typename Arg1, typename Arg2>
78 void EnsureCreated(Type*& object, Arg1 arg1, Arg2 arg2)
79 {
80   if(!object)
81   {
82     object = new Type(arg1, arg2);
83   }
84 }
85
86 float GetDpi()
87 {
88   unsigned int                      horizontalDpi = 0u;
89   unsigned int                      verticalDpi   = 0u;
90   Dali::TextAbstraction::FontClient fontClient    = Dali::TextAbstraction::FontClient::Get();
91   fontClient.GetDpi(horizontalDpi, verticalDpi);
92   return static_cast<float>(horizontalDpi);
93 }
94
95 float ConvertPixelToPoint(float pixel)
96 {
97   return pixel * 72.0f / GetDpi();
98 }
99
100 float ConvertPointToPixel(float point)
101 {
102   // Pixel size = Point size * DPI / 72.f
103   return point * GetDpi() / 72.0f;
104 }
105
106 void UpdateCursorPosition(Dali::Toolkit::Text::EventData* eventData)
107 {
108   if(eventData && Dali::Toolkit::Text::EventData::IsEditingState(eventData->mState))
109   {
110     // Update the cursor position if it's in editing mode
111     eventData->mDecoratorUpdated     = true;
112     eventData->mUpdateCursorPosition = true; // Cursor position should be updated when the font size is updated.
113   }
114 }
115
116 } // namespace
117
118 namespace Dali::Toolkit::Text
119 {
120
121 void Controller::EnableTextInput(DecoratorPtr decorator, InputMethodContext& inputMethodContext)
122 {
123   if(!decorator)
124   {
125     delete mImpl->mEventData;
126     mImpl->mEventData = NULL;
127
128     // Nothing else to do.
129     return;
130   }
131
132   EnsureCreated(mImpl->mEventData, decorator, inputMethodContext);
133 }
134
135 void Controller::SetGlyphType(TextAbstraction::GlyphType glyphType)
136 {
137   // Metrics for bitmap & vector based glyphs are different
138   mImpl->mMetrics->SetGlyphType(glyphType);
139
140   // Clear the font-specific data
141   ClearFontData();
142
143   mImpl->RequestRelayout();
144 }
145
146 void Controller::SetMarkupProcessorEnabled(bool enable)
147 {
148   if(enable != mImpl->mMarkupProcessorEnabled)
149   {
150     //If Text was already set, call the SetText again for enabling or disabling markup
151     mImpl->mMarkupProcessorEnabled = enable;
152     std::string text;
153     GetText(text);
154     SetText(text);
155   }
156
157   mImpl->mModel->mVisualModel->SetMarkupProcessorEnabled(enable);
158 }
159
160 bool Controller::IsMarkupProcessorEnabled() const
161 {
162   return mImpl->mMarkupProcessorEnabled;
163 }
164
165 bool Controller::HasAnchors() const
166 {
167   return (mImpl->mMarkupProcessorEnabled && mImpl->mModel->mLogicalModel->mAnchors.Count() && mImpl->IsShowingRealText());
168 }
169
170 void Controller::SetAutoScrollEnabled(bool enable)
171 {
172   DALI_LOG_INFO(gLogFilter, Debug::General, "Controller::SetAutoScrollEnabled[%s] SingleBox[%s]-> [%p]\n", (enable) ? "true" : "false", (mImpl->mLayoutEngine.GetLayout() == Layout::Engine::SINGLE_LINE_BOX) ? "true" : "false", this);
173
174   if(mImpl->mLayoutEngine.GetLayout() == Layout::Engine::SINGLE_LINE_BOX)
175   {
176     mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending |
177                                                             LAYOUT |
178                                                             ALIGN |
179                                                             UPDATE_LAYOUT_SIZE |
180                                                             REORDER);
181
182     if(enable)
183     {
184       DALI_LOG_INFO(gLogFilter, Debug::General, "Controller::SetAutoScrollEnabled for SINGLE_LINE_BOX\n");
185       mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending | UPDATE_DIRECTION);
186     }
187     else
188     {
189       DALI_LOG_INFO(gLogFilter, Debug::General, "Controller::SetAutoScrollEnabled Disabling autoscroll\n");
190     }
191
192     mImpl->mIsAutoScrollEnabled = enable;
193     mImpl->RequestRelayout();
194   }
195   else
196   {
197     DALI_LOG_WARNING("Attempted AutoScrolling on a non SINGLE_LINE_BOX, request ignored\n");
198     mImpl->mIsAutoScrollEnabled = false;
199   }
200 }
201
202 bool Controller::IsAutoScrollEnabled() const
203 {
204   DALI_LOG_INFO(gLogFilter, Debug::Verbose, "Controller::IsAutoScrollEnabled[%s]\n", mImpl->mIsAutoScrollEnabled ? "true" : "false");
205
206   return mImpl->mIsAutoScrollEnabled;
207 }
208
209 CharacterDirection Controller::GetAutoScrollDirection() const
210 {
211   return mImpl->mIsTextDirectionRTL;
212 }
213
214 float Controller::GetAutoScrollLineAlignment() const
215 {
216   float offset = 0.f;
217   if(mImpl->mModel->mVisualModel && (0u != mImpl->mModel->mVisualModel->mLines.Count()))
218   {
219     offset = (*mImpl->mModel->mVisualModel->mLines.Begin()).alignmentOffset;
220   }
221   return offset;
222 }
223
224 void Controller::SetHorizontalScrollEnabled(bool enable)
225 {
226   if(mImpl->mEventData && mImpl->mEventData->mDecorator)
227   {
228     mImpl->mEventData->mDecorator->SetHorizontalScrollEnabled(enable);
229   }
230 }
231
232 bool Controller::IsHorizontalScrollEnabled() const
233 {
234   return mImpl->mEventData && mImpl->mEventData->mDecorator && mImpl->mEventData->mDecorator->IsHorizontalScrollEnabled();
235 }
236
237 void Controller::SetVerticalScrollEnabled(bool enable)
238 {
239   if(mImpl->mEventData && mImpl->mEventData->mDecorator)
240   {
241     mImpl->mEventData->mDecorator->SetVerticalScrollEnabled(enable);
242   }
243 }
244
245 bool Controller::IsVerticalScrollEnabled() const
246 {
247   return mImpl->mEventData && mImpl->mEventData->mDecorator && mImpl->mEventData->mDecorator->IsVerticalScrollEnabled();
248 }
249
250 void Controller::SetSmoothHandlePanEnabled(bool enable)
251 {
252   if(mImpl->mEventData && mImpl->mEventData->mDecorator)
253   {
254     mImpl->mEventData->mDecorator->SetSmoothHandlePanEnabled(enable);
255   }
256 }
257
258 bool Controller::IsSmoothHandlePanEnabled() const
259 {
260   return mImpl->mEventData && mImpl->mEventData->mDecorator && mImpl->mEventData->mDecorator->IsSmoothHandlePanEnabled();
261 }
262
263 void Controller::SetMaximumNumberOfCharacters(Length maxCharacters)
264 {
265   mImpl->mMaximumNumberOfCharacters = maxCharacters;
266 }
267
268 int Controller::GetMaximumNumberOfCharacters()
269 {
270   return mImpl->mMaximumNumberOfCharacters;
271 }
272
273 void Controller::SetEnableCursorBlink(bool enable)
274 {
275   DALI_ASSERT_DEBUG(NULL != mImpl->mEventData && "TextInput disabled");
276
277   if(mImpl->mEventData)
278   {
279     mImpl->mEventData->mCursorBlinkEnabled = enable;
280
281     if(!enable && mImpl->mEventData->mDecorator)
282     {
283       mImpl->mEventData->mDecorator->StopCursorBlink();
284     }
285   }
286 }
287
288 bool Controller::GetEnableCursorBlink() const
289 {
290   return mImpl->mEventData && mImpl->mEventData->mCursorBlinkEnabled;
291 }
292
293 void Controller::SetMultiLineEnabled(bool enable)
294 {
295   const Layout::Engine::Type layout = enable ? Layout::Engine::MULTI_LINE_BOX : Layout::Engine::SINGLE_LINE_BOX;
296
297   if(layout != mImpl->mLayoutEngine.GetLayout())
298   {
299     // Set the layout type.
300     mImpl->mLayoutEngine.SetLayout(layout);
301
302     // Set the flags to redo the layout operations
303     const OperationsMask layoutOperations = static_cast<OperationsMask>(LAYOUT |
304                                                                         UPDATE_LAYOUT_SIZE |
305                                                                         ALIGN |
306                                                                         REORDER);
307
308     mImpl->mTextUpdateInfo.mFullRelayoutNeeded = true;
309     mImpl->mOperationsPending                  = static_cast<OperationsMask>(mImpl->mOperationsPending | layoutOperations);
310
311     // Need to recalculate natural size
312     mImpl->mRecalculateNaturalSize = true;
313
314     mImpl->RequestRelayout();
315   }
316 }
317
318 bool Controller::IsMultiLineEnabled() const
319 {
320   return Layout::Engine::MULTI_LINE_BOX == mImpl->mLayoutEngine.GetLayout();
321 }
322
323 void Controller::SetHorizontalAlignment(Text::HorizontalAlignment::Type alignment)
324 {
325   if(alignment != mImpl->mModel->mHorizontalAlignment)
326   {
327     // Set the alignment.
328     mImpl->mModel->mHorizontalAlignment = alignment;
329
330     // Set the flag to redo the alignment operation.
331     mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending | ALIGN);
332
333     if(mImpl->mEventData)
334     {
335       mImpl->mEventData->mUpdateAlignment = true;
336
337       // Update the cursor if it's in editing mode
338       if(EventData::IsEditingState(mImpl->mEventData->mState))
339       {
340         mImpl->ChangeState(EventData::EDITING);
341         mImpl->mEventData->mUpdateCursorPosition = true;
342       }
343     }
344
345     mImpl->RequestRelayout();
346   }
347 }
348
349 Text::HorizontalAlignment::Type Controller::GetHorizontalAlignment() const
350 {
351   return mImpl->mModel->mHorizontalAlignment;
352 }
353
354 void Controller::SetVerticalAlignment(VerticalAlignment::Type alignment)
355 {
356   if(alignment != mImpl->mModel->mVerticalAlignment)
357   {
358     // Set the alignment.
359     mImpl->mModel->mVerticalAlignment = alignment;
360     mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending | ALIGN);
361     mImpl->RequestRelayout();
362   }
363 }
364
365 VerticalAlignment::Type Controller::GetVerticalAlignment() const
366 {
367   return mImpl->mModel->mVerticalAlignment;
368 }
369
370 bool Controller::IsIgnoreSpacesAfterText() const
371 {
372   return mImpl->mModel->mIgnoreSpacesAfterText;
373 }
374
375 void Controller::SetIgnoreSpacesAfterText(bool ignore)
376 {
377   mImpl->mModel->mIgnoreSpacesAfterText = ignore;
378 }
379
380 void Controller::ChangedLayoutDirection()
381 {
382   mImpl->mIsLayoutDirectionChanged = true;
383 }
384
385 void Controller::SetMatchLayoutDirection(DevelText::MatchLayoutDirection type)
386 {
387   mImpl->mModel->mMatchLayoutDirection = type;
388 }
389
390 DevelText::MatchLayoutDirection Controller::GetMatchLayoutDirection() const
391 {
392   return mImpl->mModel->mMatchLayoutDirection;
393 }
394
395 void Controller::SetLayoutDirection(Dali::LayoutDirection::Type layoutDirection)
396 {
397   mImpl->mLayoutDirection = layoutDirection;
398 }
399
400 Dali::LayoutDirection::Type Controller::GetLayoutDirection(Dali::Actor& actor) const
401 {
402   if(mImpl->mModel->mMatchLayoutDirection == DevelText::MatchLayoutDirection::LOCALE ||
403      (mImpl->mModel->mMatchLayoutDirection == DevelText::MatchLayoutDirection::INHERIT && !mImpl->mIsLayoutDirectionChanged))
404   {
405     return static_cast<Dali::LayoutDirection::Type>(DevelWindow::Get(actor).GetRootLayer().GetProperty(Dali::Actor::Property::LAYOUT_DIRECTION).Get<int>());
406   }
407   else
408   {
409     return static_cast<Dali::LayoutDirection::Type>(actor.GetProperty(Dali::Actor::Property::LAYOUT_DIRECTION).Get<int>());
410   }
411 }
412
413 bool Controller::IsShowingRealText() const
414 {
415   return mImpl->IsShowingRealText();
416 }
417
418 void Controller::SetLineWrapMode(Text::LineWrap::Mode lineWrapMode)
419 {
420   if(lineWrapMode != mImpl->mModel->mLineWrapMode)
421   {
422     // Update Text layout for applying wrap mode
423     mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending |
424                                                             ALIGN |
425                                                             LAYOUT |
426                                                             UPDATE_LAYOUT_SIZE |
427                                                             REORDER);
428
429     if((mImpl->mModel->mLineWrapMode == (Text::LineWrap::Mode)DevelText::LineWrap::HYPHENATION) || (lineWrapMode == (Text::LineWrap::Mode)DevelText::LineWrap::HYPHENATION) ||
430        (mImpl->mModel->mLineWrapMode == (Text::LineWrap::Mode)DevelText::LineWrap::MIXED) || (lineWrapMode == (Text::LineWrap::Mode)DevelText::LineWrap::MIXED)) // hyphen is treated as line break
431     {
432       mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending | GET_LINE_BREAKS);
433     }
434
435     // Set the text wrap mode.
436     mImpl->mModel->mLineWrapMode = lineWrapMode;
437
438     mImpl->mTextUpdateInfo.mCharacterIndex             = 0u;
439     mImpl->mTextUpdateInfo.mNumberOfCharactersToRemove = mImpl->mTextUpdateInfo.mPreviousNumberOfCharacters;
440     mImpl->mTextUpdateInfo.mNumberOfCharactersToAdd    = mImpl->mModel->mLogicalModel->mText.Count();
441
442     // Request relayout
443     mImpl->RequestRelayout();
444   }
445 }
446
447 Text::LineWrap::Mode Controller::GetLineWrapMode() const
448 {
449   return mImpl->mModel->mLineWrapMode;
450 }
451
452 void Controller::SetTextElideEnabled(bool enabled)
453 {
454   mImpl->mModel->mElideEnabled = enabled;
455   mImpl->mModel->mVisualModel->SetTextElideEnabled(enabled);
456 }
457
458 bool Controller::IsTextElideEnabled() const
459 {
460   return mImpl->mModel->mElideEnabled;
461 }
462
463 void Controller::SetTextFitEnabled(bool enabled)
464 {
465   mImpl->mTextFitEnabled = enabled;
466 }
467
468 bool Controller::IsTextFitEnabled() const
469 {
470   return mImpl->mTextFitEnabled;
471 }
472
473 void Controller::SetTextFitMinSize(float minSize, FontSizeType type)
474 {
475   mImpl->mTextFitMinSize = (type == POINT_SIZE) ? minSize : ConvertPixelToPoint(minSize);
476 }
477
478 float Controller::GetTextFitMinSize() const
479 {
480   return mImpl->mTextFitMinSize;
481 }
482
483 void Controller::SetTextFitMaxSize(float maxSize, FontSizeType type)
484 {
485   mImpl->mTextFitMaxSize = (type == POINT_SIZE) ? maxSize : ConvertPixelToPoint(maxSize);
486 }
487
488 float Controller::GetTextFitMaxSize() const
489 {
490   return mImpl->mTextFitMaxSize;
491 }
492
493 void Controller::SetTextFitStepSize(float step, FontSizeType type)
494 {
495   mImpl->mTextFitStepSize = (type == POINT_SIZE) ? step : ConvertPixelToPoint(step);
496 }
497
498 float Controller::GetTextFitStepSize() const
499 {
500   return mImpl->mTextFitStepSize;
501 }
502
503 void Controller::SetTextFitContentSize(Vector2 size)
504 {
505   mImpl->mTextFitContentSize = size;
506 }
507
508 Vector2 Controller::GetTextFitContentSize() const
509 {
510   return mImpl->mTextFitContentSize;
511 }
512
513 void Controller::SetPlaceholderTextElideEnabled(bool enabled)
514 {
515   PlaceholderHandler::SetPlaceholderTextElideEnabled(*this, enabled);
516 }
517
518 bool Controller::IsPlaceholderTextElideEnabled() const
519 {
520   return PlaceholderHandler::IsPlaceholderTextElideEnabled(*this);
521 }
522
523 void Controller::SetSelectionEnabled(bool enabled)
524 {
525   mImpl->mEventData->mSelectionEnabled = enabled;
526 }
527
528 bool Controller::IsSelectionEnabled() const
529 {
530   return mImpl->mEventData->mSelectionEnabled;
531 }
532
533 void Controller::SetShiftSelectionEnabled(bool enabled)
534 {
535   mImpl->mEventData->mShiftSelectionFlag = enabled;
536 }
537
538 bool Controller::IsShiftSelectionEnabled() const
539 {
540   return mImpl->mEventData->mShiftSelectionFlag;
541 }
542
543 void Controller::SetGrabHandleEnabled(bool enabled)
544 {
545   mImpl->mEventData->mGrabHandleEnabled = enabled;
546 }
547
548 bool Controller::IsGrabHandleEnabled() const
549 {
550   return mImpl->mEventData->mGrabHandleEnabled;
551 }
552
553 void Controller::SetGrabHandlePopupEnabled(bool enabled)
554 {
555   mImpl->mEventData->mGrabHandlePopupEnabled = enabled;
556 }
557
558 bool Controller::IsGrabHandlePopupEnabled() const
559 {
560   return mImpl->mEventData->mGrabHandlePopupEnabled;
561 }
562
563 void Controller::SetText(const std::string& text)
564 {
565   TextUpdater::SetText(*this, text);
566 }
567
568 void Controller::GetText(std::string& text) const
569 {
570   if(!mImpl->IsShowingPlaceholderText())
571   {
572     // Retrieves the text string.
573     mImpl->GetText(0u, text);
574   }
575   else
576   {
577     DALI_LOG_INFO(gLogFilter, Debug::Verbose, "Controller::GetText %p empty (but showing placeholder)\n", this);
578   }
579 }
580
581 void Controller::SetPlaceholderText(PlaceholderType type, const std::string& text)
582 {
583   PlaceholderHandler::SetPlaceholderText(*this, type, text);
584 }
585
586 void Controller::GetPlaceholderText(PlaceholderType type, std::string& text) const
587 {
588   PlaceholderHandler::GetPlaceholderText(*this, type, text);
589 }
590
591 void Controller::UpdateAfterFontChange(const std::string& newDefaultFont)
592 {
593   DALI_LOG_INFO(gLogFilter, Debug::Verbose, "Controller::UpdateAfterFontChange\n");
594
595   if(!mImpl->mFontDefaults->familyDefined) // If user defined font then should not update when system font changes
596   {
597     DALI_LOG_INFO(gLogFilter, Debug::Concise, "Controller::UpdateAfterFontChange newDefaultFont(%s)\n", newDefaultFont.c_str());
598     mImpl->mFontDefaults->mFontDescription.family = newDefaultFont;
599
600     ClearFontData();
601
602     mImpl->RequestRelayout();
603   }
604 }
605
606 void Controller::RetrieveSelection(std::string& selectedText) const
607 {
608   mImpl->RetrieveSelection(selectedText, false);
609 }
610
611 void Controller::SetSelection(int start, int end)
612 {
613   mImpl->SetSelection(start, end);
614 }
615
616 std::pair<int, int> Controller::GetSelectionIndexes() const
617 {
618   return mImpl->GetSelectionIndexes();
619 }
620
621 void Controller::CopyStringToClipboard(const std::string& source)
622 {
623   mImpl->CopyStringToClipboard(source);
624 }
625
626 void Controller::SendSelectionToClipboard(bool deleteAfterSending)
627 {
628   mImpl->SendSelectionToClipboard(deleteAfterSending);
629 }
630
631 void Controller::SetDefaultFontFamily(const std::string& defaultFontFamily)
632 {
633   EnsureCreated(mImpl->mFontDefaults);
634
635   mImpl->mFontDefaults->mFontDescription.family = defaultFontFamily;
636   DALI_LOG_INFO(gLogFilter, Debug::General, "Controller::SetDefaultFontFamily %s\n", defaultFontFamily.c_str());
637   mImpl->mFontDefaults->familyDefined = !defaultFontFamily.empty();
638
639   // Update the cursor position if it's in editing mode
640   UpdateCursorPosition(mImpl->mEventData);
641
642   // Clear the font-specific data
643   ClearFontData();
644
645   mImpl->RequestRelayout();
646 }
647
648 const std::string& Controller::GetDefaultFontFamily() const
649 {
650   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.family : EMPTY_STRING;
651 }
652
653 void Controller::SetPlaceholderFontFamily(const std::string& placeholderTextFontFamily)
654 {
655   PlaceholderHandler::SetPlaceholderFontFamily(*this, placeholderTextFontFamily);
656 }
657
658 const std::string& Controller::GetPlaceholderFontFamily() const
659 {
660   return PlaceholderHandler::GetPlaceholderFontFamily(*this);
661 }
662
663 void Controller::SetDefaultFontWeight(FontWeight weight)
664 {
665   EnsureCreated(mImpl->mFontDefaults);
666
667   mImpl->mFontDefaults->mFontDescription.weight = weight;
668   mImpl->mFontDefaults->weightDefined           = true;
669
670   // Update the cursor position if it's in editing mode
671   UpdateCursorPosition(mImpl->mEventData);
672
673   // Clear the font-specific data
674   ClearFontData();
675
676   mImpl->RequestRelayout();
677 }
678
679 bool Controller::IsDefaultFontWeightDefined() const
680 {
681   return mImpl->mFontDefaults && mImpl->mFontDefaults->weightDefined;
682 }
683
684 FontWeight Controller::GetDefaultFontWeight() const
685 {
686   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.weight : TextAbstraction::FontWeight::NORMAL;
687 }
688
689 void Controller::SetPlaceholderTextFontWeight(FontWeight weight)
690 {
691   PlaceholderHandler::SetPlaceholderTextFontWeight(*this, weight);
692 }
693
694 bool Controller::IsPlaceholderTextFontWeightDefined() const
695 {
696   return PlaceholderHandler::IsPlaceholderTextFontWeightDefined(*this);
697 }
698
699 FontWeight Controller::GetPlaceholderTextFontWeight() const
700 {
701   return PlaceholderHandler::GetPlaceholderTextFontWeight(*this);
702 }
703
704 void Controller::SetDefaultFontWidth(FontWidth width)
705 {
706   EnsureCreated(mImpl->mFontDefaults);
707
708   mImpl->mFontDefaults->mFontDescription.width = width;
709   mImpl->mFontDefaults->widthDefined           = true;
710
711   // Update the cursor position if it's in editing mode
712   UpdateCursorPosition(mImpl->mEventData);
713
714   // Clear the font-specific data
715   ClearFontData();
716
717   mImpl->RequestRelayout();
718 }
719
720 bool Controller::IsDefaultFontWidthDefined() const
721 {
722   return mImpl->mFontDefaults && mImpl->mFontDefaults->widthDefined;
723 }
724
725 FontWidth Controller::GetDefaultFontWidth() const
726 {
727   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.width : TextAbstraction::FontWidth::NORMAL;
728 }
729
730 void Controller::SetPlaceholderTextFontWidth(FontWidth width)
731 {
732   PlaceholderHandler::SetPlaceholderTextFontWidth(*this, width);
733 }
734
735 bool Controller::IsPlaceholderTextFontWidthDefined() const
736 {
737   return PlaceholderHandler::IsPlaceholderTextFontWidthDefined(*this);
738 }
739
740 FontWidth Controller::GetPlaceholderTextFontWidth() const
741 {
742   return PlaceholderHandler::GetPlaceholderTextFontWidth(*this);
743 }
744
745 void Controller::SetDefaultFontSlant(FontSlant slant)
746 {
747   EnsureCreated(mImpl->mFontDefaults);
748
749   mImpl->mFontDefaults->mFontDescription.slant = slant;
750   mImpl->mFontDefaults->slantDefined           = true;
751
752   // Update the cursor position if it's in editing mode
753   UpdateCursorPosition(mImpl->mEventData);
754
755   // Clear the font-specific data
756   ClearFontData();
757
758   mImpl->RequestRelayout();
759 }
760
761 bool Controller::IsDefaultFontSlantDefined() const
762 {
763   return mImpl->mFontDefaults && mImpl->mFontDefaults->slantDefined;
764 }
765
766 FontSlant Controller::GetDefaultFontSlant() const
767 {
768   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.slant : TextAbstraction::FontSlant::NORMAL;
769 }
770
771 void Controller::SetPlaceholderTextFontSlant(FontSlant slant)
772 {
773   PlaceholderHandler::SetPlaceholderTextFontSlant(*this, slant);
774 }
775
776 bool Controller::IsPlaceholderTextFontSlantDefined() const
777 {
778   return PlaceholderHandler::IsPlaceholderTextFontSlantDefined(*this);
779 }
780
781 FontSlant Controller::GetPlaceholderTextFontSlant() const
782 {
783   return PlaceholderHandler::GetPlaceholderTextFontSlant(*this);
784 }
785
786 void Controller::SetFontSizeScale(float scale)
787 {
788   mImpl->mFontSizeScale = scale;
789
790   // Update the cursor position if it's in editing mode
791   UpdateCursorPosition(mImpl->mEventData);
792
793   // Clear the font-specific data
794   ClearFontData();
795
796   mImpl->RequestRelayout();
797 }
798
799 float Controller::GetFontSizeScale() const
800 {
801   return mImpl->mFontDefaults ? mImpl->mFontSizeScale : 1.0f;
802 }
803
804 void Controller::SetDefaultFontSize(float fontSize, FontSizeType type)
805 {
806   EnsureCreated(mImpl->mFontDefaults);
807
808   mImpl->mFontDefaults->mDefaultPointSize = (type == POINT_SIZE) ? fontSize : ConvertPixelToPoint(fontSize);
809   mImpl->mFontDefaults->sizeDefined       = true;
810
811   // Update the cursor position if it's in editing mode
812   UpdateCursorPosition(mImpl->mEventData);
813
814   // Clear the font-specific data
815   ClearFontData();
816
817   mImpl->RequestRelayout();
818 }
819
820 float Controller::GetDefaultFontSize(FontSizeType type) const
821 {
822   if(mImpl->mFontDefaults)
823   {
824     return (type == POINT_SIZE) ? mImpl->mFontDefaults->mDefaultPointSize : ConvertPointToPixel(mImpl->mFontDefaults->mDefaultPointSize);
825   }
826   return 0.0f;
827 }
828
829 void Controller::SetPlaceholderTextFontSize(float fontSize, FontSizeType type)
830 {
831   PlaceholderHandler::SetPlaceholderTextFontSize(*this, fontSize, type);
832 }
833
834 float Controller::GetPlaceholderTextFontSize(FontSizeType type) const
835 {
836   return PlaceholderHandler::GetPlaceholderTextFontSize(*this, type);
837 }
838
839 void Controller::SetDefaultColor(const Vector4& color)
840 {
841   mImpl->mTextColor = color;
842
843   if(!mImpl->IsShowingPlaceholderText())
844   {
845     mImpl->mModel->mVisualModel->SetTextColor(color);
846     mImpl->mModel->mLogicalModel->mColorRuns.Clear();
847     mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending | COLOR);
848     mImpl->RequestRelayout();
849   }
850 }
851
852 const Vector4& Controller::GetDefaultColor() const
853 {
854   return mImpl->mTextColor;
855 }
856
857 void Controller::SetPlaceholderTextColor(const Vector4& textColor)
858 {
859   PlaceholderHandler::SetPlaceholderTextColor(*this, textColor);
860 }
861
862 const Vector4& Controller::GetPlaceholderTextColor() const
863 {
864   return PlaceholderHandler::GetPlaceholderTextColor(*this);
865 }
866
867 void Controller::SetShadowOffset(const Vector2& shadowOffset)
868 {
869   mImpl->mModel->mVisualModel->SetShadowOffset(shadowOffset);
870   mImpl->RequestRelayout();
871 }
872
873 const Vector2& Controller::GetShadowOffset() const
874 {
875   return mImpl->mModel->mVisualModel->GetShadowOffset();
876 }
877
878 void Controller::SetShadowColor(const Vector4& shadowColor)
879 {
880   mImpl->mModel->mVisualModel->SetShadowColor(shadowColor);
881   mImpl->RequestRelayout();
882 }
883
884 const Vector4& Controller::GetShadowColor() const
885 {
886   return mImpl->mModel->mVisualModel->GetShadowColor();
887 }
888
889 void Controller::SetShadowBlurRadius(const float& shadowBlurRadius)
890 {
891   if(fabsf(GetShadowBlurRadius() - shadowBlurRadius) > Math::MACHINE_EPSILON_1)
892   {
893     mImpl->mModel->mVisualModel->SetShadowBlurRadius(shadowBlurRadius);
894     mImpl->RequestRelayout();
895   }
896 }
897
898 const float& Controller::GetShadowBlurRadius() const
899 {
900   return mImpl->mModel->mVisualModel->GetShadowBlurRadius();
901 }
902
903 void Controller::SetUnderlineColor(const Vector4& color)
904 {
905   mImpl->mModel->mVisualModel->SetUnderlineColor(color);
906   mImpl->RequestRelayout();
907 }
908
909 const Vector4& Controller::GetUnderlineColor() const
910 {
911   return mImpl->mModel->mVisualModel->GetUnderlineColor();
912 }
913
914 void Controller::SetUnderlineEnabled(bool enabled)
915 {
916   mImpl->mModel->mVisualModel->SetUnderlineEnabled(enabled);
917   mImpl->RequestRelayout();
918 }
919
920 bool Controller::IsUnderlineEnabled() const
921 {
922   return mImpl->mModel->mVisualModel->IsUnderlineEnabled();
923 }
924
925 void Controller::SetUnderlineHeight(float height)
926 {
927   mImpl->mModel->mVisualModel->SetUnderlineHeight(height);
928   mImpl->RequestRelayout();
929 }
930
931 float Controller::GetUnderlineHeight() const
932 {
933   return mImpl->mModel->mVisualModel->GetUnderlineHeight();
934 }
935
936 void Controller::SetOutlineColor(const Vector4& color)
937 {
938   mImpl->mModel->mVisualModel->SetOutlineColor(color);
939   mImpl->RequestRelayout();
940 }
941
942 const Vector4& Controller::GetOutlineColor() const
943 {
944   return mImpl->mModel->mVisualModel->GetOutlineColor();
945 }
946
947 void Controller::SetOutlineWidth(uint16_t width)
948 {
949   mImpl->mModel->mVisualModel->SetOutlineWidth(width);
950   mImpl->RequestRelayout();
951 }
952
953 uint16_t Controller::GetOutlineWidth() const
954 {
955   return mImpl->mModel->mVisualModel->GetOutlineWidth();
956 }
957
958 void Controller::SetBackgroundColor(const Vector4& color)
959 {
960   mImpl->mModel->mVisualModel->SetBackgroundColor(color);
961   mImpl->RequestRelayout();
962 }
963
964 const Vector4& Controller::GetBackgroundColor() const
965 {
966   return mImpl->mModel->mVisualModel->GetBackgroundColor();
967 }
968
969 void Controller::SetBackgroundEnabled(bool enabled)
970 {
971   mImpl->mModel->mVisualModel->SetBackgroundEnabled(enabled);
972   mImpl->RequestRelayout();
973 }
974
975 bool Controller::IsBackgroundEnabled() const
976 {
977   return mImpl->mModel->mVisualModel->IsBackgroundEnabled();
978 }
979
980 void Controller::SetDefaultEmbossProperties(const std::string& embossProperties)
981 {
982   EnsureCreated(mImpl->mEmbossDefaults);
983   mImpl->mEmbossDefaults->properties = embossProperties;
984 }
985
986 const std::string& Controller::GetDefaultEmbossProperties() const
987 {
988   return mImpl->mEmbossDefaults ? mImpl->mEmbossDefaults->properties : EMPTY_STRING;
989 }
990
991 void Controller::SetDefaultOutlineProperties(const std::string& outlineProperties)
992 {
993   EnsureCreated(mImpl->mOutlineDefaults);
994   mImpl->mOutlineDefaults->properties = outlineProperties;
995 }
996
997 const std::string& Controller::GetDefaultOutlineProperties() const
998 {
999   return mImpl->mOutlineDefaults ? mImpl->mOutlineDefaults->properties : EMPTY_STRING;
1000 }
1001
1002 bool Controller::SetDefaultLineSpacing(float lineSpacing)
1003 {
1004   if(std::fabs(lineSpacing - mImpl->mLayoutEngine.GetDefaultLineSpacing()) > Math::MACHINE_EPSILON_1000)
1005   {
1006     mImpl->mLayoutEngine.SetDefaultLineSpacing(lineSpacing);
1007     mImpl->mRecalculateNaturalSize = true;
1008
1009     mImpl->RelayoutForNewLineSize();
1010     return true;
1011   }
1012   return false;
1013 }
1014
1015 float Controller::GetDefaultLineSpacing() const
1016 {
1017   return mImpl->mLayoutEngine.GetDefaultLineSpacing();
1018 }
1019
1020 bool Controller::SetDefaultLineSize(float lineSize)
1021 {
1022   if(std::fabs(lineSize - mImpl->mLayoutEngine.GetDefaultLineSize()) > Math::MACHINE_EPSILON_1000)
1023   {
1024     mImpl->mLayoutEngine.SetDefaultLineSize(lineSize);
1025     mImpl->mRecalculateNaturalSize = true;
1026
1027     mImpl->RelayoutForNewLineSize();
1028     return true;
1029   }
1030   return false;
1031 }
1032
1033 float Controller::GetDefaultLineSize() const
1034 {
1035   return mImpl->mLayoutEngine.GetDefaultLineSize();
1036 }
1037
1038 void Controller::SetInputColor(const Vector4& color)
1039 {
1040   InputProperties::SetInputColor(*this, color);
1041 }
1042
1043 const Vector4& Controller::GetInputColor() const
1044 {
1045   return InputProperties::GetInputColor(*this);
1046 }
1047
1048 void Controller::SetInputFontFamily(const std::string& fontFamily)
1049 {
1050   InputFontHandler::SetInputFontFamily(*this, fontFamily);
1051 }
1052
1053 const std::string& Controller::GetInputFontFamily() const
1054 {
1055   return InputFontHandler::GetInputFontFamily(*this);
1056 }
1057
1058 void Controller::SetInputFontWeight(FontWeight weight)
1059 {
1060   InputFontHandler::SetInputFontWeight(*this, weight);
1061 }
1062
1063 bool Controller::IsInputFontWeightDefined() const
1064 {
1065   return InputFontHandler::IsInputFontWeightDefined(*this);
1066 }
1067
1068 FontWeight Controller::GetInputFontWeight() const
1069 {
1070   return InputFontHandler::GetInputFontWeight(*this);
1071 }
1072
1073 void Controller::SetInputFontWidth(FontWidth width)
1074 {
1075   InputFontHandler::SetInputFontWidth(*this, width);
1076 }
1077
1078 bool Controller::IsInputFontWidthDefined() const
1079 {
1080   return InputFontHandler::IsInputFontWidthDefined(*this);
1081 }
1082
1083 FontWidth Controller::GetInputFontWidth() const
1084 {
1085   return InputFontHandler::GetInputFontWidth(*this);
1086 }
1087
1088 void Controller::SetInputFontSlant(FontSlant slant)
1089 {
1090   InputFontHandler::SetInputFontSlant(*this, slant);
1091 }
1092
1093 bool Controller::IsInputFontSlantDefined() const
1094 {
1095   return InputFontHandler::IsInputFontSlantDefined(*this);
1096 }
1097
1098 FontSlant Controller::GetInputFontSlant() const
1099 {
1100   return InputFontHandler::GetInputFontSlant(*this);
1101 }
1102
1103 void Controller::SetInputFontPointSize(float size)
1104 {
1105   InputFontHandler::SetInputFontPointSize(*this, size);
1106 }
1107
1108 float Controller::GetInputFontPointSize() const
1109 {
1110   return InputFontHandler::GetInputFontPointSize(*this);
1111 }
1112
1113 void Controller::SetInputLineSpacing(float lineSpacing)
1114 {
1115   InputProperties::SetInputLineSpacing(*this, lineSpacing);
1116 }
1117
1118 float Controller::GetInputLineSpacing() const
1119 {
1120   return InputProperties::GetInputLineSpacing(*this);
1121 }
1122
1123 void Controller::SetInputShadowProperties(const std::string& shadowProperties)
1124 {
1125   InputProperties::SetInputShadowProperties(*this, shadowProperties);
1126 }
1127
1128 const std::string& Controller::GetInputShadowProperties() const
1129 {
1130   return InputProperties::GetInputShadowProperties(*this);
1131 }
1132
1133 void Controller::SetInputUnderlineProperties(const std::string& underlineProperties)
1134 {
1135   InputProperties::SetInputUnderlineProperties(*this, underlineProperties);
1136 }
1137
1138 const std::string& Controller::GetInputUnderlineProperties() const
1139 {
1140   return InputProperties::GetInputUnderlineProperties(*this);
1141 }
1142
1143 void Controller::SetInputEmbossProperties(const std::string& embossProperties)
1144 {
1145   InputProperties::SetInputEmbossProperties(*this, embossProperties);
1146 }
1147
1148 const std::string& Controller::GetInputEmbossProperties() const
1149 {
1150   return InputProperties::GetInputEmbossProperties(*this);
1151 }
1152
1153 void Controller::SetInputOutlineProperties(const std::string& outlineProperties)
1154 {
1155   InputProperties::SetInputOutlineProperties(*this, outlineProperties);
1156 }
1157
1158 const std::string& Controller::GetInputOutlineProperties() const
1159 {
1160   return InputProperties::GetInputOutlineProperties(*this);
1161 }
1162
1163 void Controller::SetInputModePassword(bool passwordInput)
1164 {
1165   InputProperties::SetInputModePassword(*this, passwordInput);
1166 }
1167
1168 bool Controller::IsInputModePassword()
1169 {
1170   return InputProperties::IsInputModePassword(*this);
1171 }
1172
1173 void Controller::SetNoTextDoubleTapAction(NoTextTap::Action action)
1174 {
1175   if(mImpl->mEventData)
1176   {
1177     mImpl->mEventData->mDoubleTapAction = action;
1178   }
1179 }
1180
1181 Controller::NoTextTap::Action Controller::GetNoTextDoubleTapAction() const
1182 {
1183   return mImpl->mEventData ? mImpl->mEventData->mDoubleTapAction : NoTextTap::NO_ACTION;
1184 }
1185
1186 void Controller::SetNoTextLongPressAction(NoTextTap::Action action)
1187 {
1188   if(mImpl->mEventData)
1189   {
1190     mImpl->mEventData->mLongPressAction = action;
1191   }
1192 }
1193
1194 Controller::NoTextTap::Action Controller::GetNoTextLongPressAction() const
1195 {
1196   return mImpl->mEventData ? mImpl->mEventData->mLongPressAction : NoTextTap::NO_ACTION;
1197 }
1198
1199 bool Controller::IsUnderlineSetByString()
1200 {
1201   return mImpl->mUnderlineSetByString;
1202 }
1203
1204 void Controller::UnderlineSetByString(bool setByString)
1205 {
1206   mImpl->mUnderlineSetByString = setByString;
1207 }
1208
1209 bool Controller::IsShadowSetByString()
1210 {
1211   return mImpl->mShadowSetByString;
1212 }
1213
1214 void Controller::ShadowSetByString(bool setByString)
1215 {
1216   mImpl->mShadowSetByString = setByString;
1217 }
1218
1219 bool Controller::IsOutlineSetByString()
1220 {
1221   return mImpl->mOutlineSetByString;
1222 }
1223
1224 void Controller::OutlineSetByString(bool setByString)
1225 {
1226   mImpl->mOutlineSetByString = setByString;
1227 }
1228
1229 bool Controller::IsFontStyleSetByString()
1230 {
1231   return mImpl->mFontStyleSetByString;
1232 }
1233
1234 void Controller::FontStyleSetByString(bool setByString)
1235 {
1236   mImpl->mFontStyleSetByString = setByString;
1237 }
1238
1239 Layout::Engine& Controller::GetLayoutEngine()
1240 {
1241   return mImpl->mLayoutEngine;
1242 }
1243
1244 View& Controller::GetView()
1245 {
1246   return mImpl->mView;
1247 }
1248
1249 Vector3 Controller::GetNaturalSize()
1250 {
1251   return Relayouter::GetNaturalSize(*this);
1252 }
1253
1254 bool Controller::CheckForTextFit(float pointSize, Size& layoutSize)
1255 {
1256   return Relayouter::CheckForTextFit(*this, pointSize, layoutSize);
1257 }
1258
1259 void Controller::FitPointSizeforLayout(Size layoutSize)
1260 {
1261   Relayouter::FitPointSizeforLayout(*this, layoutSize);
1262 }
1263
1264 float Controller::GetHeightForWidth(float width)
1265 {
1266   return Relayouter::GetHeightForWidth(*this, width);
1267 }
1268
1269 int Controller::GetLineCount(float width)
1270 {
1271   GetHeightForWidth(width);
1272   return mImpl->mModel->GetNumberOfLines();
1273 }
1274
1275 const ModelInterface* const Controller::GetTextModel() const
1276 {
1277   return mImpl->mModel.Get();
1278 }
1279
1280 float Controller::GetScrollAmountByUserInput()
1281 {
1282   float scrollAmount = 0.0f;
1283
1284   if(NULL != mImpl->mEventData && mImpl->mEventData->mCheckScrollAmount)
1285   {
1286     scrollAmount                          = mImpl->mModel->mScrollPosition.y - mImpl->mModel->mScrollPositionLast.y;
1287     mImpl->mEventData->mCheckScrollAmount = false;
1288   }
1289   return scrollAmount;
1290 }
1291
1292 bool Controller::GetTextScrollInfo(float& scrollPosition, float& controlHeight, float& layoutHeight)
1293 {
1294   const Vector2& layout = mImpl->mModel->mVisualModel->GetLayoutSize();
1295   bool           isScrolled;
1296
1297   controlHeight  = mImpl->mModel->mVisualModel->mControlSize.height;
1298   layoutHeight   = layout.height;
1299   scrollPosition = mImpl->mModel->mScrollPosition.y;
1300   isScrolled     = !Equals(mImpl->mModel->mScrollPosition.y, mImpl->mModel->mScrollPositionLast.y, Math::MACHINE_EPSILON_1);
1301   return isScrolled;
1302 }
1303
1304 void Controller::SetHiddenInputOption(const Property::Map& options)
1305 {
1306   EnsureCreated<HiddenText, Controller*>(mImpl->mHiddenInput, this);
1307   mImpl->mHiddenInput->SetProperties(options);
1308 }
1309
1310 void Controller::GetHiddenInputOption(Property::Map& options)
1311 {
1312   if(mImpl->mHiddenInput)
1313   {
1314     mImpl->mHiddenInput->GetProperties(options);
1315   }
1316 }
1317
1318 void Controller::SetInputFilterOption(const Property::Map& options)
1319 {
1320   EnsureCreated(mImpl->mInputFilter);
1321   mImpl->mInputFilter->SetProperties(options);
1322 }
1323
1324 void Controller::GetInputFilterOption(Property::Map& options)
1325 {
1326   if(mImpl->mInputFilter)
1327   {
1328     mImpl->mInputFilter->GetProperties(options);
1329   }
1330 }
1331
1332 void Controller::SetPlaceholderProperty(const Property::Map& map)
1333 {
1334   PlaceholderHandler::SetPlaceholderProperty(*this, map);
1335 }
1336
1337 void Controller::GetPlaceholderProperty(Property::Map& map)
1338 {
1339   PlaceholderHandler::GetPlaceholderProperty(*this, map);
1340 }
1341
1342 Toolkit::DevelText::TextDirection::Type Controller::GetTextDirection()
1343 {
1344   // Make sure the model is up-to-date before layouting
1345   ProcessModifyEvents();
1346
1347   if(mImpl->mUpdateTextDirection)
1348   {
1349     // Operations that can be done only once until the text changes.
1350     const OperationsMask onlyOnceOperations = static_cast<OperationsMask>(CONVERT_TO_UTF32 |
1351                                                                           GET_SCRIPTS |
1352                                                                           VALIDATE_FONTS |
1353                                                                           GET_LINE_BREAKS |
1354                                                                           BIDI_INFO |
1355                                                                           SHAPE_TEXT |
1356                                                                           GET_GLYPH_METRICS);
1357
1358     // Set the update info to relayout the whole text.
1359     mImpl->mTextUpdateInfo.mParagraphCharacterIndex     = 0u;
1360     mImpl->mTextUpdateInfo.mRequestedNumberOfCharacters = mImpl->mModel->mLogicalModel->mText.Count();
1361
1362     // Make sure the model is up-to-date before layouting
1363     mImpl->UpdateModel(onlyOnceOperations);
1364
1365     Vector3 naturalSize;
1366     DoRelayout(Size(MAX_FLOAT, MAX_FLOAT),
1367                static_cast<OperationsMask>(onlyOnceOperations |
1368                                            LAYOUT | REORDER | UPDATE_DIRECTION),
1369                naturalSize.GetVectorXY());
1370
1371     // Do not do again the only once operations.
1372     mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending & ~onlyOnceOperations);
1373
1374     // Clear the update info. This info will be set the next time the text is updated.
1375     mImpl->mTextUpdateInfo.Clear();
1376
1377     // FullRelayoutNeeded should be true because DoRelayout is MAX_FLOAT, MAX_FLOAT.
1378     mImpl->mTextUpdateInfo.mFullRelayoutNeeded = true;
1379
1380     mImpl->mUpdateTextDirection = false;
1381   }
1382
1383   return mImpl->mIsTextDirectionRTL ? Toolkit::DevelText::TextDirection::RIGHT_TO_LEFT : Toolkit::DevelText::TextDirection::LEFT_TO_RIGHT;
1384 }
1385
1386 Toolkit::DevelText::VerticalLineAlignment::Type Controller::GetVerticalLineAlignment() const
1387 {
1388   return mImpl->mModel->GetVerticalLineAlignment();
1389 }
1390
1391 void Controller::SetVerticalLineAlignment(Toolkit::DevelText::VerticalLineAlignment::Type alignment)
1392 {
1393   mImpl->mModel->mVerticalLineAlignment = alignment;
1394 }
1395
1396 Toolkit::DevelText::EllipsisPosition::Type Controller::GetEllipsisPosition() const
1397 {
1398   return mImpl->mModel->GetEllipsisPosition();
1399 }
1400
1401 void Controller::SetEllipsisPosition(Toolkit::DevelText::EllipsisPosition::Type ellipsisPosition)
1402 {
1403   mImpl->mModel->mEllipsisPosition = ellipsisPosition;
1404   mImpl->mModel->mVisualModel->SetEllipsisPosition(ellipsisPosition);
1405 }
1406
1407 Controller::UpdateTextType Controller::Relayout(const Size& size, Dali::LayoutDirection::Type layoutDirection)
1408 {
1409   return Relayouter::Relayout(*this, size, layoutDirection);
1410 }
1411
1412 void Controller::RequestRelayout()
1413 {
1414   mImpl->RequestRelayout();
1415 }
1416
1417 bool Controller::IsInputStyleChangedSignalsQueueEmpty()
1418 {
1419   return mImpl->IsInputStyleChangedSignalsQueueEmpty();
1420 }
1421
1422 void Controller::ProcessInputStyleChangedSignals()
1423 {
1424   mImpl->ProcessInputStyleChangedSignals();
1425 }
1426
1427 void Controller::KeyboardFocusGainEvent()
1428 {
1429   EventHandler::KeyboardFocusGainEvent(*this);
1430 }
1431
1432 void Controller::KeyboardFocusLostEvent()
1433 {
1434   EventHandler::KeyboardFocusLostEvent(*this);
1435 }
1436
1437 bool Controller::KeyEvent(const Dali::KeyEvent& keyEvent)
1438 {
1439   return EventHandler::KeyEvent(*this, keyEvent);
1440 }
1441
1442 void Controller::AnchorEvent(float x, float y)
1443 {
1444   EventHandler::AnchorEvent(*this, x, y);
1445 }
1446
1447 void Controller::TapEvent(unsigned int tapCount, float x, float y)
1448 {
1449   EventHandler::TapEvent(*this, tapCount, x, y);
1450 }
1451
1452 void Controller::PanEvent(GestureState state, const Vector2& displacement)
1453 {
1454   EventHandler::PanEvent(*this, state, displacement);
1455 }
1456
1457 void Controller::LongPressEvent(GestureState state, float x, float y)
1458 {
1459   EventHandler::LongPressEvent(*this, state, x, y);
1460 }
1461
1462 void Controller::SelectEvent(float x, float y, SelectionType selectType)
1463 {
1464   EventHandler::SelectEvent(*this, x, y, selectType);
1465 }
1466
1467 void Controller::SelectEvent(const uint32_t start, const uint32_t end, SelectionType selectType)
1468 {
1469   EventHandler::SelectEvent(*this, start, end, selectType);
1470 }
1471
1472 void Controller::SetTextSelectionRange(const uint32_t* start, const uint32_t* end)
1473 {
1474   if(mImpl->mEventData)
1475   {
1476     mImpl->mEventData->mCheckScrollAmount     = true;
1477     mImpl->mEventData->mIsLeftHandleSelected  = true;
1478     mImpl->mEventData->mIsRightHandleSelected = true;
1479     mImpl->SetTextSelectionRange(start, end);
1480     mImpl->RequestRelayout();
1481     KeyboardFocusGainEvent();
1482   }
1483 }
1484
1485 Uint32Pair Controller::GetTextSelectionRange() const
1486 {
1487   return mImpl->GetTextSelectionRange();
1488 }
1489
1490 CharacterIndex Controller::GetPrimaryCursorPosition() const
1491 {
1492   return mImpl->GetPrimaryCursorPosition();
1493 }
1494
1495 bool Controller::SetPrimaryCursorPosition(CharacterIndex index, bool focused)
1496 {
1497   if(mImpl->mEventData)
1498   {
1499     mImpl->mEventData->mCheckScrollAmount     = true;
1500     mImpl->mEventData->mIsLeftHandleSelected  = true;
1501     mImpl->mEventData->mIsRightHandleSelected = true;
1502     mImpl->mEventData->mCheckScrollAmount     = true;
1503     if(mImpl->SetPrimaryCursorPosition(index, focused) && focused)
1504     {
1505       KeyboardFocusGainEvent();
1506       return true;
1507     }
1508   }
1509   return false;
1510 }
1511
1512 void Controller::SelectWholeText()
1513 {
1514   SelectEvent(0.f, 0.f, SelectionType::ALL);
1515 }
1516
1517 void Controller::SelectNone()
1518 {
1519   SelectEvent(0.f, 0.f, SelectionType::NONE);
1520 }
1521
1522 void Controller::SelectText(const uint32_t start, const uint32_t end)
1523 {
1524   SelectEvent(start, end, SelectionType::RANGE);
1525 }
1526
1527 string Controller::GetSelectedText() const
1528 {
1529   string text;
1530   if(EventData::SELECTING == mImpl->mEventData->mState)
1531   {
1532     mImpl->RetrieveSelection(text, false);
1533   }
1534   return text;
1535 }
1536
1537 string Controller::CopyText()
1538 {
1539   string text;
1540   mImpl->RetrieveSelection(text, false);
1541   mImpl->SendSelectionToClipboard(false); // Text not modified
1542
1543   mImpl->mEventData->mUpdateCursorPosition = true;
1544
1545   mImpl->RequestRelayout(); // Cursor, Handles, Selection Highlight, Popup
1546
1547   return text;
1548 }
1549
1550 string Controller::CutText()
1551 {
1552   string text;
1553   mImpl->RetrieveSelection(text, false);
1554
1555   if(!IsEditable())
1556   {
1557     return EMPTY_STRING;
1558   }
1559
1560   mImpl->SendSelectionToClipboard(true); // Synchronous call to modify text
1561   mImpl->mOperationsPending = ALL_OPERATIONS;
1562
1563   if((0u != mImpl->mModel->mLogicalModel->mText.Count()) ||
1564      !mImpl->IsPlaceholderAvailable())
1565   {
1566     mImpl->QueueModifyEvent(ModifyEvent::TEXT_DELETED);
1567   }
1568   else
1569   {
1570     ShowPlaceholderText();
1571   }
1572
1573   mImpl->mEventData->mUpdateCursorPosition = true;
1574   mImpl->mEventData->mScrollAfterDelete    = true;
1575
1576   mImpl->RequestRelayout();
1577
1578   if(nullptr != mImpl->mEditableControlInterface)
1579   {
1580     mImpl->mEditableControlInterface->TextChanged(true);
1581   }
1582   return text;
1583 }
1584
1585 void Controller::PasteText()
1586 {
1587   mImpl->RequestGetTextFromClipboard(); // Request clipboard service to retrieve an item
1588 }
1589
1590 InputMethodContext::CallbackData Controller::OnInputMethodContextEvent(InputMethodContext& inputMethodContext, const InputMethodContext::EventData& inputMethodContextEvent)
1591 {
1592   return EventHandler::OnInputMethodContextEvent(*this, inputMethodContext, inputMethodContextEvent);
1593 }
1594
1595 void Controller::PasteClipboardItemEvent()
1596 {
1597   EventHandler::PasteClipboardItemEvent(*this);
1598 }
1599
1600 void Controller::GetTargetSize(Vector2& targetSize)
1601 {
1602   targetSize = mImpl->mModel->mVisualModel->mControlSize;
1603 }
1604
1605 void Controller::AddDecoration(Actor& actor, bool needsClipping)
1606 {
1607   if(mImpl->mEditableControlInterface)
1608   {
1609     mImpl->mEditableControlInterface->AddDecoration(actor, needsClipping);
1610   }
1611 }
1612
1613 bool Controller::IsEditable() const
1614 {
1615   return mImpl->IsEditable();
1616 }
1617
1618 void Controller::SetEditable(bool editable)
1619 {
1620   mImpl->SetEditable(editable);
1621 }
1622
1623 void Controller::ScrollBy(Vector2 scroll)
1624 {
1625   mImpl->ScrollBy(scroll);
1626 }
1627
1628 float Controller::GetHorizontalScrollPosition()
1629 {
1630   return mImpl->GetHorizontalScrollPosition();
1631 }
1632
1633 float Controller::GetVerticalScrollPosition()
1634 {
1635   return mImpl->GetVerticalScrollPosition();
1636 }
1637
1638 void Controller::DecorationEvent(HandleType handleType, HandleState state, float x, float y)
1639 {
1640   EventHandler::DecorationEvent(*this, handleType, state, x, y);
1641 }
1642
1643 void Controller::TextPopupButtonTouched(Dali::Toolkit::TextSelectionPopup::Buttons button)
1644 {
1645   EventHandler::TextPopupButtonTouched(*this, button);
1646 }
1647
1648 void Controller::DisplayTimeExpired()
1649 {
1650   mImpl->mEventData->mUpdateCursorPosition = true;
1651   // Apply modifications to the model
1652   mImpl->mOperationsPending = ALL_OPERATIONS;
1653
1654   mImpl->RequestRelayout();
1655 }
1656
1657 void Controller::InsertText(const std::string& text, Controller::InsertType type)
1658 {
1659   TextUpdater::InsertText(*this, text, type);
1660 }
1661
1662 void Controller::PasteText(const std::string& stringToPaste)
1663 {
1664   TextUpdater::PasteText(*this, stringToPaste);
1665 }
1666
1667 bool Controller::RemoveText(int cursorOffset, int numberOfCharacters, UpdateInputStyleType type)
1668 {
1669   return TextUpdater::RemoveText(*this, cursorOffset, numberOfCharacters, type);
1670 }
1671
1672 bool Controller::RemoveSelectedText()
1673 {
1674   return TextUpdater::RemoveSelectedText(*this);
1675 }
1676
1677 void Controller::InsertTextAnchor(int numberOfCharacters, CharacterIndex previousCursorIndex)
1678 {
1679   TextUpdater::InsertTextAnchor(*this, numberOfCharacters, previousCursorIndex);
1680 }
1681
1682 void Controller::RemoveTextAnchor(int cursorOffset, int numberOfCharacters, CharacterIndex previousCursorIndex)
1683 {
1684   TextUpdater::RemoveTextAnchor(*this, cursorOffset, numberOfCharacters, previousCursorIndex);
1685 }
1686
1687 bool Controller::DoRelayout(const Size& size, OperationsMask operationsRequired, Size& layoutSize)
1688 {
1689   return Relayouter::DoRelayout(*this, size, operationsRequired, layoutSize);
1690 }
1691
1692 void Controller::CalculateVerticalOffset(const Size& controlSize)
1693 {
1694   Relayouter::CalculateVerticalOffset(*this, controlSize);
1695 }
1696
1697 void Controller::ProcessModifyEvents()
1698 {
1699   EventHandler::ProcessModifyEvents(*this);
1700 }
1701
1702 void Controller::TextReplacedEvent()
1703 {
1704   EventHandler::TextReplacedEvent(*this);
1705 }
1706
1707 void Controller::TextInsertedEvent()
1708 {
1709   EventHandler::TextInsertedEvent(*this);
1710 }
1711
1712 void Controller::TextDeletedEvent()
1713 {
1714   EventHandler::TextDeletedEvent(*this);
1715 }
1716
1717 bool Controller::DeleteEvent(int keyCode)
1718 {
1719   return EventHandler::DeleteEvent(*this, keyCode);
1720 }
1721
1722 // private : Helpers.
1723
1724 void Controller::ResetText()
1725 {
1726   TextUpdater::ResetText(*this);
1727 }
1728
1729 void Controller::ShowPlaceholderText()
1730 {
1731   PlaceholderHandler::ShowPlaceholderText(*this);
1732 }
1733
1734 void Controller::ClearFontData()
1735 {
1736   if(mImpl->mFontDefaults)
1737   {
1738     mImpl->mFontDefaults->mFontId = 0u; // Remove old font ID
1739   }
1740
1741   // Set flags to update the model.
1742   mImpl->mTextUpdateInfo.mCharacterIndex             = 0u;
1743   mImpl->mTextUpdateInfo.mNumberOfCharactersToRemove = mImpl->mTextUpdateInfo.mPreviousNumberOfCharacters;
1744   mImpl->mTextUpdateInfo.mNumberOfCharactersToAdd    = mImpl->mModel->mLogicalModel->mText.Count();
1745
1746   mImpl->mTextUpdateInfo.mClearAll           = true;
1747   mImpl->mTextUpdateInfo.mFullRelayoutNeeded = true;
1748   mImpl->mRecalculateNaturalSize             = true;
1749
1750   mImpl->mOperationsPending = static_cast<OperationsMask>(mImpl->mOperationsPending |
1751                                                           VALIDATE_FONTS |
1752                                                           SHAPE_TEXT |
1753                                                           BIDI_INFO |
1754                                                           GET_GLYPH_METRICS |
1755                                                           LAYOUT |
1756                                                           UPDATE_LAYOUT_SIZE |
1757                                                           REORDER |
1758                                                           ALIGN);
1759 }
1760
1761 void Controller::ClearStyleData()
1762 {
1763   mImpl->mModel->mLogicalModel->mColorRuns.Clear();
1764   mImpl->mModel->mLogicalModel->ClearFontDescriptionRuns();
1765 }
1766
1767 void Controller::ResetCursorPosition(CharacterIndex cursorIndex)
1768 {
1769   // Reset the cursor position
1770   if(NULL != mImpl->mEventData)
1771   {
1772     mImpl->mEventData->mPrimaryCursorPosition = cursorIndex;
1773
1774     // Update the cursor if it's in editing mode.
1775     if(EventData::IsEditingState(mImpl->mEventData->mState))
1776     {
1777       mImpl->mEventData->mUpdateCursorPosition = true;
1778     }
1779   }
1780 }
1781
1782 CharacterIndex Controller::GetCursorPosition()
1783 {
1784   if(!mImpl->mEventData)
1785     return 0;
1786
1787   return mImpl->mEventData->mPrimaryCursorPosition;
1788 }
1789
1790 void Controller::ResetScrollPosition()
1791 {
1792   if(mImpl->mEventData)
1793   {
1794     // Reset the scroll position.
1795     mImpl->mModel->mScrollPosition                = Vector2::ZERO;
1796     mImpl->mEventData->mScrollAfterUpdatePosition = true;
1797   }
1798 }
1799
1800 void Controller::SetControlInterface(ControlInterface* controlInterface)
1801 {
1802   mImpl->mControlInterface = controlInterface;
1803 }
1804
1805 void Controller::SetAnchorControlInterface(AnchorControlInterface* anchorControlInterface)
1806 {
1807   mImpl->mAnchorControlInterface = anchorControlInterface;
1808 }
1809
1810 bool Controller::ShouldClearFocusOnEscape() const
1811 {
1812   return mImpl->mShouldClearFocusOnEscape;
1813 }
1814
1815 Actor Controller::CreateBackgroundActor()
1816 {
1817   return mImpl->CreateBackgroundActor();
1818 }
1819
1820 Controller::Controller(ControlInterface*           controlInterface,
1821                        EditableControlInterface*   editableControlInterface,
1822                        SelectableControlInterface* selectableControlInterface,
1823                        AnchorControlInterface*     anchorControlInterface)
1824 : mImpl(new Controller::Impl(controlInterface, editableControlInterface, selectableControlInterface, anchorControlInterface))
1825 {
1826 }
1827
1828 Controller::~Controller()
1829 {
1830   delete mImpl;
1831 }
1832
1833 } // namespace Dali::Toolkit::Text