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