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