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