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