Merge "Keep reference when member callback excute + Make NPatchData as RefObject...
[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::SetCurrentLineSize(float lineSize)
378 {
379   mImpl->mCurrentLineSize = lineSize;
380 }
381
382 float Controller::GetCurrentLineSize() const
383 {
384   return mImpl->mCurrentLineSize;
385 }
386
387 void Controller::SetTextFitMinSize(float minSize, FontSizeType type)
388 {
389   mImpl->mTextFitMinSize = (type == POINT_SIZE) ? minSize : ConvertPixelToPoint(minSize);
390 }
391
392 float Controller::GetTextFitMinSize() const
393 {
394   return mImpl->mTextFitMinSize;
395 }
396
397 void Controller::SetTextFitMaxSize(float maxSize, FontSizeType type)
398 {
399   mImpl->mTextFitMaxSize = (type == POINT_SIZE) ? maxSize : ConvertPixelToPoint(maxSize);
400 }
401
402 float Controller::GetTextFitMaxSize() const
403 {
404   return mImpl->mTextFitMaxSize;
405 }
406
407 void Controller::SetTextFitStepSize(float step, FontSizeType type)
408 {
409   mImpl->mTextFitStepSize = (type == POINT_SIZE) ? step : ConvertPixelToPoint(step);
410 }
411
412 float Controller::GetTextFitStepSize() const
413 {
414   return mImpl->mTextFitStepSize;
415 }
416
417 void Controller::SetTextFitContentSize(Vector2 size)
418 {
419   mImpl->mTextFitContentSize = size;
420 }
421
422 Vector2 Controller::GetTextFitContentSize() const
423 {
424   return mImpl->mTextFitContentSize;
425 }
426
427 float Controller::GetTextFitPointSize() const
428 {
429   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFitPointSize : 0.0f;
430 }
431
432 void Controller::SetTextFitLineSize(float lineSize)
433 {
434   mImpl->mTextFitLineSize = lineSize;
435 }
436
437 void Controller::SetTextFitArrayEnabled(bool enabled)
438 {
439   mImpl->mTextFitArrayEnabled = enabled;
440   mImpl->ClearFontData();
441   mImpl->RequestRelayout();
442 }
443
444 bool Controller::IsTextFitArrayEnabled() const
445 {
446   return mImpl->mTextFitArrayEnabled;
447 }
448
449 void Controller::SetTextFitArray(std::vector<Toolkit::DevelTextLabel::FitOption>& fitOptions)
450 {
451   mImpl->mTextFitArray = fitOptions;
452 }
453
454 std::vector<Toolkit::DevelTextLabel::FitOption>& Controller::GetTextFitArray()
455 {
456   return mImpl->mTextFitArray;
457 }
458
459 void Controller::SetPlaceholderTextElideEnabled(bool enabled)
460 {
461   PlaceholderHandler::SetPlaceholderTextElideEnabled(*this, enabled);
462 }
463
464 bool Controller::IsPlaceholderTextElideEnabled() const
465 {
466   return PlaceholderHandler::IsPlaceholderTextElideEnabled(*this);
467 }
468
469 void Controller::SetSelectionEnabled(bool enabled)
470 {
471   mImpl->mEventData->mSelectionEnabled = enabled;
472 }
473
474 bool Controller::IsSelectionEnabled() const
475 {
476   return mImpl->mEventData->mSelectionEnabled;
477 }
478
479 void Controller::SetShiftSelectionEnabled(bool enabled)
480 {
481   mImpl->mEventData->mShiftSelectionFlag = enabled;
482 }
483
484 bool Controller::IsShiftSelectionEnabled() const
485 {
486   return mImpl->mEventData->mShiftSelectionFlag;
487 }
488
489 void Controller::SetGrabHandleEnabled(bool enabled)
490 {
491   mImpl->mEventData->mGrabHandleEnabled = enabled;
492 }
493
494 bool Controller::IsGrabHandleEnabled() const
495 {
496   return mImpl->mEventData->mGrabHandleEnabled;
497 }
498
499 void Controller::SetGrabHandlePopupEnabled(bool enabled)
500 {
501   mImpl->mEventData->mGrabHandlePopupEnabled = enabled;
502 }
503
504 bool Controller::IsGrabHandlePopupEnabled() const
505 {
506   return mImpl->mEventData->mGrabHandlePopupEnabled;
507 }
508
509 void Controller::SetText(const std::string& text)
510 {
511   TextUpdater::SetText(*this, text);
512 }
513
514 void Controller::GetText(std::string& text) const
515 {
516   mImpl->GetText(text);
517 }
518
519 Length Controller::GetNumberOfCharacters() const
520 {
521   return mImpl->GetNumberOfCharacters();
522 }
523
524 void Controller::SetSpannedText(const Text::Spanned& spannedText)
525 {
526   SpannableHandler::SetSpannedText(*this, spannedText);
527 }
528
529 void Controller::SetPlaceholderText(PlaceholderType type, const std::string& text)
530 {
531   PlaceholderHandler::SetPlaceholderText(*this, type, text);
532 }
533
534 void Controller::GetPlaceholderText(PlaceholderType type, std::string& text) const
535 {
536   PlaceholderHandler::GetPlaceholderText(*this, type, text);
537 }
538
539 void Controller::UpdateAfterFontChange(const std::string& newDefaultFont)
540 {
541   mImpl->UpdateAfterFontChange(newDefaultFont);
542 }
543
544 void Controller::RetrieveSelection(std::string& selectedText) const
545 {
546   mImpl->RetrieveSelection(selectedText, false);
547 }
548
549 void Controller::SetSelection(int start, int end)
550 {
551   mImpl->SetSelection(start, end);
552 }
553
554 std::pair<int, int> Controller::GetSelectionIndexes() const
555 {
556   return mImpl->GetSelectionIndexes();
557 }
558
559 void Controller::CopyStringToClipboard(const std::string& source)
560 {
561   mImpl->CopyStringToClipboard(source);
562 }
563
564 void Controller::SendSelectionToClipboard(bool deleteAfterSending)
565 {
566   mImpl->SendSelectionToClipboard(deleteAfterSending);
567 }
568
569 void Controller::SetDefaultFontFamily(const std::string& defaultFontFamily)
570 {
571   EnsureCreated(mImpl->mFontDefaults);
572
573   if(mImpl->mFontDefaults->mFontDescription.family != defaultFontFamily)
574   {
575     mImpl->mFontDefaults->mFontDescription.family = defaultFontFamily;
576     DALI_LOG_INFO(gLogFilter, Debug::General, "Controller::SetDefaultFontFamily %s\n", defaultFontFamily.c_str());
577     mImpl->mFontDefaults->familyDefined = !defaultFontFamily.empty();
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
589 std::string Controller::GetDefaultFontFamily() const
590 {
591   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.family : EMPTY_STRING;
592 }
593
594 void Controller::SetPlaceholderFontFamily(const std::string& placeholderTextFontFamily)
595 {
596   PlaceholderHandler::SetPlaceholderFontFamily(*this, placeholderTextFontFamily);
597 }
598
599 std::string Controller::GetPlaceholderFontFamily() const
600 {
601   return PlaceholderHandler::GetPlaceholderFontFamily(*this);
602 }
603
604 void Controller::SetDefaultFontWeight(FontWeight weight)
605 {
606   EnsureCreated(mImpl->mFontDefaults);
607
608   mImpl->mFontDefaults->mFontDescription.weight = weight;
609   mImpl->mFontDefaults->weightDefined           = true;
610
611   // Update the cursor position if it's in editing mode
612   UpdateCursorPosition(mImpl->mEventData);
613
614   // Clear the font-specific data
615   mImpl->ClearFontData();
616
617   mImpl->RequestRelayout();
618 }
619
620 bool Controller::IsDefaultFontWeightDefined() const
621 {
622   return mImpl->mFontDefaults && mImpl->mFontDefaults->weightDefined;
623 }
624
625 FontWeight Controller::GetDefaultFontWeight() const
626 {
627   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.weight : TextAbstraction::FontWeight::NORMAL;
628 }
629
630 void Controller::SetPlaceholderTextFontWeight(FontWeight weight)
631 {
632   PlaceholderHandler::SetPlaceholderTextFontWeight(*this, weight);
633 }
634
635 bool Controller::IsPlaceholderTextFontWeightDefined() const
636 {
637   return PlaceholderHandler::IsPlaceholderTextFontWeightDefined(*this);
638 }
639
640 FontWeight Controller::GetPlaceholderTextFontWeight() const
641 {
642   return PlaceholderHandler::GetPlaceholderTextFontWeight(*this);
643 }
644
645 void Controller::SetDefaultFontWidth(FontWidth width)
646 {
647   EnsureCreated(mImpl->mFontDefaults);
648
649   mImpl->mFontDefaults->mFontDescription.width = width;
650   mImpl->mFontDefaults->widthDefined           = true;
651
652   // Update the cursor position if it's in editing mode
653   UpdateCursorPosition(mImpl->mEventData);
654
655   // Clear the font-specific data
656   mImpl->ClearFontData();
657
658   mImpl->RequestRelayout();
659 }
660
661 bool Controller::IsDefaultFontWidthDefined() const
662 {
663   return mImpl->mFontDefaults && mImpl->mFontDefaults->widthDefined;
664 }
665
666 FontWidth Controller::GetDefaultFontWidth() const
667 {
668   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.width : TextAbstraction::FontWidth::NORMAL;
669 }
670
671 void Controller::SetPlaceholderTextFontWidth(FontWidth width)
672 {
673   PlaceholderHandler::SetPlaceholderTextFontWidth(*this, width);
674 }
675
676 bool Controller::IsPlaceholderTextFontWidthDefined() const
677 {
678   return PlaceholderHandler::IsPlaceholderTextFontWidthDefined(*this);
679 }
680
681 FontWidth Controller::GetPlaceholderTextFontWidth() const
682 {
683   return PlaceholderHandler::GetPlaceholderTextFontWidth(*this);
684 }
685
686 void Controller::SetDefaultFontSlant(FontSlant slant)
687 {
688   EnsureCreated(mImpl->mFontDefaults);
689
690   mImpl->mFontDefaults->mFontDescription.slant = slant;
691   mImpl->mFontDefaults->slantDefined           = true;
692
693   // Update the cursor position if it's in editing mode
694   UpdateCursorPosition(mImpl->mEventData);
695
696   // Clear the font-specific data
697   mImpl->ClearFontData();
698
699   mImpl->RequestRelayout();
700 }
701
702 bool Controller::IsDefaultFontSlantDefined() const
703 {
704   return mImpl->mFontDefaults && mImpl->mFontDefaults->slantDefined;
705 }
706
707 FontSlant Controller::GetDefaultFontSlant() const
708 {
709   return mImpl->mFontDefaults ? mImpl->mFontDefaults->mFontDescription.slant : TextAbstraction::FontSlant::NORMAL;
710 }
711
712 void Controller::SetPlaceholderTextFontSlant(FontSlant slant)
713 {
714   PlaceholderHandler::SetPlaceholderTextFontSlant(*this, slant);
715 }
716
717 bool Controller::IsPlaceholderTextFontSlantDefined() const
718 {
719   return PlaceholderHandler::IsPlaceholderTextFontSlantDefined(*this);
720 }
721
722 FontSlant Controller::GetPlaceholderTextFontSlant() const
723 {
724   return PlaceholderHandler::GetPlaceholderTextFontSlant(*this);
725 }
726
727 void Controller::SetFontSizeScale(float scale)
728 {
729   mImpl->mFontSizeScale = scale;
730
731   // No relayout is required
732   if(!mImpl->mFontSizeScaleEnabled) return;
733
734   // Update the cursor position if it's in editing mode
735   UpdateCursorPosition(mImpl->mEventData);
736
737   // Clear the font-specific data
738   mImpl->ClearFontData();
739
740   mImpl->RequestRelayout();
741 }
742
743 float Controller::GetFontSizeScale() const
744 {
745   return mImpl->mFontDefaults ? mImpl->mFontSizeScale : 1.0f;
746 }
747
748 void Controller::SetFontSizeScaleEnabled(bool enabled)
749 {
750   mImpl->mFontSizeScaleEnabled = enabled;
751
752   // Update the cursor position if it's in editing mode
753   UpdateCursorPosition(mImpl->mEventData);
754
755   // Clear the font-specific data
756   mImpl->ClearFontData();
757
758   mImpl->RequestRelayout();
759 }
760
761 bool Controller::IsFontSizeScaleEnabled() const
762 {
763   return mImpl->mFontSizeScaleEnabled;
764 }
765
766 void Controller::SetDefaultFontSize(float fontSize, FontSizeType type)
767 {
768   EnsureCreated(mImpl->mFontDefaults);
769
770   mImpl->mFontDefaults->mDefaultPointSize = (type == POINT_SIZE) ? fontSize : ConvertPixelToPoint(fontSize);
771   mImpl->mFontDefaults->sizeDefined       = true;
772
773   // Update the cursor position if it's in editing mode
774   UpdateCursorPosition(mImpl->mEventData);
775
776   // Clear the font-specific data
777   mImpl->ClearFontData();
778
779   mImpl->RequestRelayout();
780 }
781
782 float Controller::GetDefaultFontSize(FontSizeType type) const
783 {
784   if(mImpl->mFontDefaults)
785   {
786     return (type == POINT_SIZE) ? mImpl->mFontDefaults->mDefaultPointSize : ConvertPointToPixel(mImpl->mFontDefaults->mDefaultPointSize);
787   }
788   return 0.0f;
789 }
790
791 void Controller::SetPlaceholderTextFontSize(float fontSize, FontSizeType type)
792 {
793   PlaceholderHandler::SetPlaceholderTextFontSize(*this, fontSize, type);
794 }
795
796 float Controller::GetPlaceholderTextFontSize(FontSizeType type) const
797 {
798   return PlaceholderHandler::GetPlaceholderTextFontSize(*this, type);
799 }
800
801 void Controller::SetDefaultColor(const Vector4& color)
802 {
803   mImpl->SetDefaultColor(color);
804 }
805
806 const Vector4& Controller::GetDefaultColor() const
807 {
808   return mImpl->mTextColor;
809 }
810
811 void Controller::SetDisabledColorOpacity(float opacity)
812 {
813   mImpl->mDisabledColorOpacity = opacity;
814 }
815
816 float Controller::GetDisabledColorOpacity() const
817 {
818   return mImpl->mDisabledColorOpacity;
819 }
820
821 void Controller::SetUserInteractionEnabled(bool enabled)
822 {
823   mImpl->SetUserInteractionEnabled(enabled);
824 }
825
826 bool Controller::IsUserInteractionEnabled() const
827 {
828   return mImpl->mIsUserInteractionEnabled;
829 }
830
831 void Controller::SetPlaceholderTextColor(const Vector4& textColor)
832 {
833   PlaceholderHandler::SetPlaceholderTextColor(*this, textColor);
834 }
835
836 const Vector4& Controller::GetPlaceholderTextColor() const
837 {
838   return PlaceholderHandler::GetPlaceholderTextColor(*this);
839 }
840
841 void Controller::SetShadowOffset(const Vector2& shadowOffset)
842 {
843   mImpl->mModel->mVisualModel->SetShadowOffset(shadowOffset);
844   mImpl->RequestRelayout();
845 }
846
847 const Vector2& Controller::GetShadowOffset() const
848 {
849   return mImpl->mModel->mVisualModel->GetShadowOffset();
850 }
851
852 void Controller::SetShadowColor(const Vector4& shadowColor)
853 {
854   mImpl->mModel->mVisualModel->SetShadowColor(shadowColor);
855   mImpl->RequestRelayout();
856 }
857
858 const Vector4& Controller::GetShadowColor() const
859 {
860   return mImpl->mModel->mVisualModel->GetShadowColor();
861 }
862
863 void Controller::SetShadowBlurRadius(const float& shadowBlurRadius)
864 {
865   if(fabsf(GetShadowBlurRadius() - shadowBlurRadius) > Math::MACHINE_EPSILON_1)
866   {
867     mImpl->mModel->mVisualModel->SetShadowBlurRadius(shadowBlurRadius);
868     mImpl->RequestRelayout();
869   }
870 }
871
872 const float& Controller::GetShadowBlurRadius() const
873 {
874   return mImpl->mModel->mVisualModel->GetShadowBlurRadius();
875 }
876
877 void Controller::SetUnderlineColor(const Vector4& color)
878 {
879   mImpl->mModel->mVisualModel->SetUnderlineColor(color);
880   mImpl->RequestRelayout();
881 }
882
883 const Vector4& Controller::GetUnderlineColor() const
884 {
885   return mImpl->mModel->mVisualModel->GetUnderlineColor();
886 }
887
888 void Controller::SetUnderlineEnabled(bool enabled)
889 {
890   mImpl->mModel->mVisualModel->SetUnderlineEnabled(enabled);
891   mImpl->RequestRelayout();
892 }
893
894 bool Controller::IsUnderlineEnabled() const
895 {
896   return mImpl->mModel->mVisualModel->IsUnderlineEnabled();
897 }
898
899 void Controller::SetUnderlineHeight(float height)
900 {
901   mImpl->mModel->mVisualModel->SetUnderlineHeight(height);
902   mImpl->RequestRelayout();
903 }
904
905 float Controller::GetUnderlineHeight() const
906 {
907   return mImpl->mModel->mVisualModel->GetUnderlineHeight();
908 }
909
910 void Controller::SetUnderlineType(Text::Underline::Type type)
911 {
912   mImpl->mModel->mVisualModel->SetUnderlineType(type);
913
914   mImpl->RequestRelayout();
915 }
916
917 Text::Underline::Type Controller::GetUnderlineType() const
918 {
919   return mImpl->mModel->mVisualModel->GetUnderlineType();
920 }
921
922 void Controller::SetDashedUnderlineWidth(float width)
923 {
924   mImpl->mModel->mVisualModel->SetDashedUnderlineWidth(width);
925
926   mImpl->RequestRelayout();
927 }
928
929 float Controller::GetDashedUnderlineWidth() const
930 {
931   return mImpl->mModel->mVisualModel->GetDashedUnderlineWidth();
932 }
933
934 void Controller::SetDashedUnderlineGap(float gap)
935 {
936   mImpl->mModel->mVisualModel->SetDashedUnderlineGap(gap);
937
938   mImpl->RequestRelayout();
939 }
940
941 float Controller::GetDashedUnderlineGap() const
942 {
943   return mImpl->mModel->mVisualModel->GetDashedUnderlineGap();
944 }
945
946 void Controller::SetOutlineColor(const Vector4& color)
947 {
948   mImpl->mModel->mVisualModel->SetOutlineColor(color);
949   mImpl->RequestRelayout();
950 }
951
952 const Vector4& Controller::GetOutlineColor() const
953 {
954   return mImpl->mModel->mVisualModel->GetOutlineColor();
955 }
956
957 void Controller::SetOutlineWidth(uint16_t width)
958 {
959   mImpl->mModel->mVisualModel->SetOutlineWidth(width);
960   mImpl->RequestRelayout();
961 }
962
963 uint16_t Controller::GetOutlineWidth() const
964 {
965   return mImpl->mModel->mVisualModel->GetOutlineWidth();
966 }
967
968 void Controller::SetBackgroundColor(const Vector4& color)
969 {
970   mImpl->mModel->mVisualModel->SetBackgroundColor(color);
971   mImpl->RequestRelayout();
972 }
973
974 const Vector4& Controller::GetBackgroundColor() const
975 {
976   return mImpl->mModel->mVisualModel->GetBackgroundColor();
977 }
978
979 void Controller::SetBackgroundEnabled(bool enabled)
980 {
981   mImpl->mModel->mVisualModel->SetBackgroundEnabled(enabled);
982   mImpl->RequestRelayout();
983 }
984
985 bool Controller::IsBackgroundEnabled() const
986 {
987   return mImpl->mModel->mVisualModel->IsBackgroundEnabled();
988 }
989
990 void Controller::SetDefaultEmbossProperties(const std::string& embossProperties)
991 {
992   EnsureCreated(mImpl->mEmbossDefaults);
993   mImpl->mEmbossDefaults->properties = embossProperties;
994 }
995
996 std::string Controller::GetDefaultEmbossProperties() const
997 {
998   return mImpl->mEmbossDefaults ? mImpl->mEmbossDefaults->properties : EMPTY_STRING;
999 }
1000
1001 void Controller::SetDefaultOutlineProperties(const std::string& outlineProperties)
1002 {
1003   EnsureCreated(mImpl->mOutlineDefaults);
1004   mImpl->mOutlineDefaults->properties = outlineProperties;
1005 }
1006
1007 std::string Controller::GetDefaultOutlineProperties() const
1008 {
1009   return mImpl->mOutlineDefaults ? mImpl->mOutlineDefaults->properties : EMPTY_STRING;
1010 }
1011
1012 bool Controller::SetDefaultLineSpacing(float lineSpacing)
1013 {
1014   return mImpl->SetDefaultLineSpacing(lineSpacing);
1015 }
1016
1017 float Controller::GetDefaultLineSpacing() const
1018 {
1019   return mImpl->mLayoutEngine.GetDefaultLineSpacing();
1020 }
1021
1022 bool Controller::SetDefaultLineSize(float lineSize)
1023 {
1024   return mImpl->SetDefaultLineSize(lineSize);
1025 }
1026
1027 float Controller::GetDefaultLineSize() const
1028 {
1029   return mImpl->mLayoutEngine.GetDefaultLineSize();
1030 }
1031
1032 bool Controller::SetRelativeLineSize(float relativeLineSize)
1033 {
1034   return mImpl->SetRelativeLineSize(relativeLineSize);
1035 }
1036
1037 float Controller::GetRelativeLineSize() const
1038 {
1039   return mImpl->GetRelativeLineSize();
1040 }
1041
1042 void Controller::SetInputColor(const Vector4& color)
1043 {
1044   InputProperties::SetInputColor(*this, color);
1045 }
1046
1047 const Vector4& Controller::GetInputColor() const
1048 {
1049   return InputProperties::GetInputColor(*this);
1050 }
1051
1052 void Controller::SetInputFontFamily(const std::string& fontFamily)
1053 {
1054   InputFontHandler::SetInputFontFamily(*this, fontFamily);
1055 }
1056
1057 std::string Controller::GetInputFontFamily() const
1058 {
1059   return InputFontHandler::GetInputFontFamily(*this);
1060 }
1061
1062 void Controller::SetInputFontWeight(FontWeight weight)
1063 {
1064   InputFontHandler::SetInputFontWeight(*this, weight);
1065 }
1066
1067 bool Controller::IsInputFontWeightDefined() const
1068 {
1069   return InputFontHandler::IsInputFontWeightDefined(*this);
1070 }
1071
1072 FontWeight Controller::GetInputFontWeight() const
1073 {
1074   return InputFontHandler::GetInputFontWeight(*this);
1075 }
1076
1077 void Controller::SetInputFontWidth(FontWidth width)
1078 {
1079   InputFontHandler::SetInputFontWidth(*this, width);
1080 }
1081
1082 bool Controller::IsInputFontWidthDefined() const
1083 {
1084   return InputFontHandler::IsInputFontWidthDefined(*this);
1085 }
1086
1087 FontWidth Controller::GetInputFontWidth() const
1088 {
1089   return InputFontHandler::GetInputFontWidth(*this);
1090 }
1091
1092 void Controller::SetInputFontSlant(FontSlant slant)
1093 {
1094   InputFontHandler::SetInputFontSlant(*this, slant);
1095 }
1096
1097 bool Controller::IsInputFontSlantDefined() const
1098 {
1099   return InputFontHandler::IsInputFontSlantDefined(*this);
1100 }
1101
1102 FontSlant Controller::GetInputFontSlant() const
1103 {
1104   return InputFontHandler::GetInputFontSlant(*this);
1105 }
1106
1107 void Controller::SetInputFontPointSize(float size)
1108 {
1109   InputFontHandler::SetInputFontPointSize(*this, size);
1110 }
1111
1112 float Controller::GetInputFontPointSize() const
1113 {
1114   return InputFontHandler::GetInputFontPointSize(*this);
1115 }
1116
1117 void Controller::SetInputLineSpacing(float lineSpacing)
1118 {
1119   InputProperties::SetInputLineSpacing(*this, lineSpacing);
1120 }
1121
1122 float Controller::GetInputLineSpacing() const
1123 {
1124   return InputProperties::GetInputLineSpacing(*this);
1125 }
1126
1127 void Controller::SetInputShadowProperties(const std::string& shadowProperties)
1128 {
1129   InputProperties::SetInputShadowProperties(*this, shadowProperties);
1130 }
1131
1132 std::string Controller::GetInputShadowProperties() const
1133 {
1134   return InputProperties::GetInputShadowProperties(*this);
1135 }
1136
1137 void Controller::SetInputUnderlineProperties(const std::string& underlineProperties)
1138 {
1139   InputProperties::SetInputUnderlineProperties(*this, underlineProperties);
1140 }
1141
1142 std::string Controller::GetInputUnderlineProperties() const
1143 {
1144   return InputProperties::GetInputUnderlineProperties(*this);
1145 }
1146
1147 void Controller::SetInputEmbossProperties(const std::string& embossProperties)
1148 {
1149   InputProperties::SetInputEmbossProperties(*this, embossProperties);
1150 }
1151
1152 std::string Controller::GetInputEmbossProperties() const
1153 {
1154   return InputProperties::GetInputEmbossProperties(*this);
1155 }
1156
1157 void Controller::SetInputOutlineProperties(const std::string& outlineProperties)
1158 {
1159   InputProperties::SetInputOutlineProperties(*this, outlineProperties);
1160 }
1161
1162 std::string Controller::GetInputOutlineProperties() const
1163 {
1164   return InputProperties::GetInputOutlineProperties(*this);
1165 }
1166
1167 void Controller::SetInputModePassword(bool passwordInput)
1168 {
1169   InputProperties::SetInputModePassword(*this, passwordInput);
1170 }
1171
1172 bool Controller::IsInputModePassword()
1173 {
1174   return InputProperties::IsInputModePassword(*this);
1175 }
1176
1177 void Controller::SetNoTextDoubleTapAction(NoTextTap::Action action)
1178 {
1179   if(mImpl->mEventData)
1180   {
1181     mImpl->mEventData->mDoubleTapAction = action;
1182   }
1183 }
1184
1185 Controller::NoTextTap::Action Controller::GetNoTextDoubleTapAction() const
1186 {
1187   return mImpl->mEventData ? mImpl->mEventData->mDoubleTapAction : NoTextTap::NO_ACTION;
1188 }
1189
1190 void Controller::SetNoTextLongPressAction(NoTextTap::Action action)
1191 {
1192   if(mImpl->mEventData)
1193   {
1194     mImpl->mEventData->mLongPressAction = action;
1195   }
1196 }
1197
1198 Controller::NoTextTap::Action Controller::GetNoTextLongPressAction() const
1199 {
1200   return mImpl->mEventData ? mImpl->mEventData->mLongPressAction : NoTextTap::NO_ACTION;
1201 }
1202
1203 bool Controller::IsUnderlineSetByString()
1204 {
1205   return mImpl->mUnderlineSetByString;
1206 }
1207
1208 void Controller::UnderlineSetByString(bool setByString)
1209 {
1210   mImpl->mUnderlineSetByString = setByString;
1211 }
1212
1213 bool Controller::IsShadowSetByString()
1214 {
1215   return mImpl->mShadowSetByString;
1216 }
1217
1218 void Controller::ShadowSetByString(bool setByString)
1219 {
1220   mImpl->mShadowSetByString = setByString;
1221 }
1222
1223 bool Controller::IsOutlineSetByString()
1224 {
1225   return mImpl->mOutlineSetByString;
1226 }
1227
1228 void Controller::OutlineSetByString(bool setByString)
1229 {
1230   mImpl->mOutlineSetByString = setByString;
1231 }
1232
1233 bool Controller::IsFontStyleSetByString()
1234 {
1235   return mImpl->mFontStyleSetByString;
1236 }
1237
1238 void Controller::FontStyleSetByString(bool setByString)
1239 {
1240   mImpl->mFontStyleSetByString = setByString;
1241 }
1242
1243 void Controller::SetStrikethroughHeight(float height)
1244 {
1245   mImpl->mModel->mVisualModel->SetStrikethroughHeight(height);
1246
1247   mImpl->RequestRelayout();
1248 }
1249
1250 float Controller::GetStrikethroughHeight() const
1251 {
1252   return mImpl->mModel->mVisualModel->GetStrikethroughHeight();
1253 }
1254
1255 void Controller::SetStrikethroughColor(const Vector4& color)
1256 {
1257   mImpl->mModel->mVisualModel->SetStrikethroughColor(color);
1258
1259   mImpl->RequestRelayout();
1260 }
1261
1262 const Vector4& Controller::GetStrikethroughColor() const
1263 {
1264   return mImpl->mModel->mVisualModel->GetStrikethroughColor();
1265 }
1266
1267 void Controller::SetStrikethroughEnabled(bool enabled)
1268 {
1269   mImpl->mModel->mVisualModel->SetStrikethroughEnabled(enabled);
1270
1271   mImpl->RequestRelayout();
1272 }
1273
1274 bool Controller::IsStrikethroughEnabled() const
1275 {
1276   return mImpl->mModel->mVisualModel->IsStrikethroughEnabled();
1277 }
1278
1279 void Controller::SetInputStrikethroughProperties(const std::string& strikethroughProperties)
1280 {
1281   if(NULL != mImpl->mEventData)
1282   {
1283     mImpl->mEventData->mInputStyle.strikethroughProperties = strikethroughProperties;
1284   }
1285 }
1286
1287 std::string Controller::GetInputStrikethroughProperties() const
1288 {
1289   return (NULL != mImpl->mEventData) ? mImpl->mEventData->mInputStyle.strikethroughProperties : EMPTY_STRING;
1290 }
1291
1292 bool Controller::IsStrikethroughSetByString()
1293 {
1294   return mImpl->mStrikethroughSetByString;
1295 }
1296
1297 void Controller::StrikethroughSetByString(bool setByString)
1298 {
1299   mImpl->mStrikethroughSetByString = setByString;
1300 }
1301
1302 Layout::Engine& Controller::GetLayoutEngine()
1303 {
1304   return mImpl->mLayoutEngine;
1305 }
1306
1307 View& Controller::GetView()
1308 {
1309   return mImpl->mView;
1310 }
1311
1312 Vector3 Controller::GetNaturalSize()
1313 {
1314   return Relayouter::GetNaturalSize(*this);
1315 }
1316
1317 bool Controller::CheckForTextFit(float pointSize, Size& layoutSize)
1318 {
1319   return Relayouter::CheckForTextFit(*this, pointSize, layoutSize);
1320 }
1321
1322 void Controller::FitPointSizeforLayout(Size layoutSize)
1323 {
1324   Relayouter::FitPointSizeforLayout(*this, layoutSize);
1325 }
1326
1327 void Controller::FitArrayPointSizeforLayout(Size layoutSize)
1328 {
1329   Relayouter::FitArrayPointSizeforLayout(*this, layoutSize);
1330 }
1331
1332 float Controller::GetHeightForWidth(float width)
1333 {
1334   return Relayouter::GetHeightForWidth(*this, width);
1335 }
1336
1337 int Controller::GetLineCount(float width)
1338 {
1339   GetHeightForWidth(width);
1340   return mImpl->mModel->GetNumberOfLines();
1341 }
1342
1343 const ModelInterface* Controller::GetTextModel() const
1344 {
1345   return mImpl->mModel.Get();
1346 }
1347
1348 float Controller::GetScrollAmountByUserInput()
1349 {
1350   float scrollAmount = 0.0f;
1351
1352   if(NULL != mImpl->mEventData && mImpl->mEventData->mCheckScrollAmount)
1353   {
1354     scrollAmount                          = mImpl->mModel->mScrollPosition.y - mImpl->mModel->mScrollPositionLast.y;
1355     mImpl->mEventData->mCheckScrollAmount = false;
1356   }
1357   return scrollAmount;
1358 }
1359
1360 bool Controller::GetTextScrollInfo(float& scrollPosition, float& controlHeight, float& layoutHeight)
1361 {
1362   const Vector2& layout = mImpl->mModel->mVisualModel->GetLayoutSize();
1363   bool           isScrolled;
1364
1365   controlHeight  = mImpl->mModel->mVisualModel->mControlSize.height;
1366   layoutHeight   = layout.height;
1367   scrollPosition = mImpl->mModel->mScrollPosition.y;
1368   isScrolled     = !Equals(mImpl->mModel->mScrollPosition.y, mImpl->mModel->mScrollPositionLast.y, Math::MACHINE_EPSILON_1);
1369   return isScrolled;
1370 }
1371
1372 void Controller::SetHiddenInputOption(const Property::Map& options)
1373 {
1374   EnsureCreated<HiddenText, Controller*>(mImpl->mHiddenInput, this);
1375   mImpl->mHiddenInput->SetProperties(options);
1376 }
1377
1378 void Controller::GetHiddenInputOption(Property::Map& options)
1379 {
1380   if(mImpl->mHiddenInput)
1381   {
1382     mImpl->mHiddenInput->GetProperties(options);
1383   }
1384 }
1385
1386 void Controller::SetInputFilterOption(const Property::Map& options)
1387 {
1388   EnsureCreated(mImpl->mInputFilter);
1389   mImpl->mInputFilter->SetProperties(options);
1390 }
1391
1392 void Controller::GetInputFilterOption(Property::Map& options)
1393 {
1394   if(mImpl->mInputFilter)
1395   {
1396     mImpl->mInputFilter->GetProperties(options);
1397   }
1398 }
1399
1400 void Controller::SetPlaceholderProperty(const Property::Map& map)
1401 {
1402   PlaceholderHandler::SetPlaceholderProperty(*this, map);
1403 }
1404
1405 void Controller::GetPlaceholderProperty(Property::Map& map)
1406 {
1407   PlaceholderHandler::GetPlaceholderProperty(*this, map);
1408 }
1409
1410 Toolkit::DevelText::TextDirection::Type Controller::GetTextDirection()
1411 {
1412   // Make sure the model is up-to-date before layouting
1413   EventHandler::ProcessModifyEvents(*this);
1414
1415   return mImpl->GetTextDirection();
1416 }
1417
1418 Toolkit::DevelText::VerticalLineAlignment::Type Controller::GetVerticalLineAlignment() const
1419 {
1420   return mImpl->mModel->GetVerticalLineAlignment();
1421 }
1422
1423 void Controller::SetVerticalLineAlignment(Toolkit::DevelText::VerticalLineAlignment::Type alignment)
1424 {
1425   mImpl->mModel->mVerticalLineAlignment = alignment;
1426 }
1427
1428 Toolkit::DevelText::EllipsisPosition::Type Controller::GetEllipsisPosition() const
1429 {
1430   return mImpl->mModel->GetEllipsisPosition();
1431 }
1432
1433 void Controller::SetEllipsisPosition(Toolkit::DevelText::EllipsisPosition::Type ellipsisPosition)
1434 {
1435   mImpl->mModel->mEllipsisPosition = ellipsisPosition;
1436   mImpl->mModel->mVisualModel->SetEllipsisPosition(ellipsisPosition);
1437 }
1438
1439 void Controller::SetCharacterSpacing(float characterSpacing)
1440 {
1441   mImpl->mModel->mVisualModel->SetCharacterSpacing(characterSpacing);
1442
1443   mImpl->RelayoutAllCharacters();
1444   mImpl->RequestRelayout();
1445 }
1446
1447 const float Controller::GetCharacterSpacing() const
1448 {
1449   return mImpl->mModel->mVisualModel->GetCharacterSpacing();
1450 }
1451
1452 Controller::UpdateTextType Controller::Relayout(const Size& size, Dali::LayoutDirection::Type layoutDirection)
1453 {
1454   return Relayouter::Relayout(*this, size, layoutDirection);
1455 }
1456
1457 void Controller::RequestRelayout()
1458 {
1459   mImpl->RequestRelayout();
1460 }
1461
1462 Vector<Vector2> Controller::GetTextSize(CharacterIndex startIndex, CharacterIndex endIndex)
1463 {
1464   Vector<Vector2> sizesList;
1465   Vector<Vector2> positionsList;
1466
1467   GetTextGeometry(mImpl->mModel, startIndex, endIndex, sizesList, positionsList);
1468   return sizesList;
1469 }
1470
1471 Vector<Vector2> Controller::GetTextPosition(CharacterIndex startIndex, CharacterIndex endIndex)
1472 {
1473   Vector<Vector2> sizesList;
1474   Vector<Vector2> positionsList;
1475
1476   GetTextGeometry(mImpl->mModel, startIndex, endIndex, sizesList, positionsList);
1477   return positionsList;
1478 }
1479
1480 Rect<float> Controller::GetLineBoundingRectangle(const uint32_t lineIndex)
1481 {
1482   return GetLineBoundingRect(mImpl->mModel, lineIndex);
1483 }
1484
1485 Rect<float> Controller::GetCharacterBoundingRectangle(const uint32_t charIndex)
1486 {
1487   return GetCharacterBoundingRect(mImpl->mModel, charIndex);
1488 }
1489
1490 int Controller::GetCharacterIndexAtPosition(float visualX, float visualY)
1491 {
1492   return GetCharIndexAtPosition(mImpl->mModel, visualX, visualY);
1493 }
1494
1495 Rect<> Controller::GetTextBoundingRectangle(CharacterIndex startIndex, CharacterIndex endIndex)
1496 {
1497   Vector<Vector2> sizeList;
1498   Vector<Vector2> positionList;
1499
1500   GetTextGeometry(mImpl->mModel, startIndex, endIndex, sizeList, positionList);
1501
1502   if(sizeList.Empty() || sizeList.Size() != positionList.Size())
1503   {
1504     return {0, 0, 0, 0};
1505   }
1506
1507   auto controlWidth = mImpl->mModel->mVisualModel->mControlSize.width;
1508   auto minX         = positionList[0].x;
1509   auto minY         = positionList[0].y;
1510   auto maxRight     = positionList[0].x + sizeList[0].x;
1511   auto maxBottom    = positionList[0].y + sizeList[0].y;
1512
1513   for(unsigned int i = 1; i < sizeList.Size(); i++)
1514   {
1515     minX      = std::min(minX, positionList[i].x);
1516     minY      = std::min(minY, positionList[i].y);
1517     maxRight  = std::max(maxRight, positionList[i].x + sizeList[i].x);
1518     maxBottom = std::max(maxBottom, positionList[i].y + sizeList[i].y);
1519   }
1520
1521   if(minX < 0.0f)
1522   {
1523     minX = 0.0f;
1524   }
1525
1526   if(maxRight > controlWidth)
1527   {
1528     maxRight = controlWidth;
1529   }
1530
1531   return {minX, minY, maxRight - minX, maxBottom - minY};
1532 }
1533
1534 bool Controller::IsInputStyleChangedSignalsQueueEmpty()
1535 {
1536   return mImpl->IsInputStyleChangedSignalsQueueEmpty();
1537 }
1538
1539 void Controller::ProcessInputStyleChangedSignals()
1540 {
1541   mImpl->ProcessInputStyleChangedSignals();
1542 }
1543
1544 void Controller::KeyboardFocusGainEvent()
1545 {
1546   EventHandler::KeyboardFocusGainEvent(*this);
1547 }
1548
1549 void Controller::KeyboardFocusLostEvent()
1550 {
1551   EventHandler::KeyboardFocusLostEvent(*this);
1552 }
1553
1554 bool Controller::KeyEvent(const Dali::KeyEvent& keyEvent)
1555 {
1556   return EventHandler::KeyEvent(*this, keyEvent);
1557 }
1558
1559 void Controller::AnchorEvent(float x, float y)
1560 {
1561   EventHandler::AnchorEvent(*this, x, y);
1562 }
1563
1564 void Controller::TapEvent(unsigned int tapCount, float x, float y)
1565 {
1566   EventHandler::TapEvent(*this, tapCount, x, y);
1567 }
1568
1569 void Controller::PanEvent(GestureState state, const Vector2& displacement)
1570 {
1571   EventHandler::PanEvent(*this, state, displacement);
1572 }
1573
1574 void Controller::LongPressEvent(GestureState state, float x, float y)
1575 {
1576   EventHandler::LongPressEvent(*this, state, x, y);
1577 }
1578
1579 void Controller::SelectEvent(float x, float y, SelectionType selectType)
1580 {
1581   EventHandler::SelectEvent(*this, x, y, selectType);
1582 }
1583
1584 void Controller::SetTextSelectionRange(const uint32_t* start, const uint32_t* end)
1585 {
1586   if(mImpl->mEventData)
1587   {
1588     mImpl->mEventData->mCheckScrollAmount     = true;
1589     mImpl->mEventData->mIsLeftHandleSelected  = true;
1590     mImpl->mEventData->mIsRightHandleSelected = true;
1591     mImpl->SetTextSelectionRange(start, end);
1592     mImpl->RequestRelayout();
1593     EventHandler::KeyboardFocusGainEvent(*this);
1594   }
1595 }
1596
1597 Uint32Pair Controller::GetTextSelectionRange() const
1598 {
1599   return mImpl->GetTextSelectionRange();
1600 }
1601
1602 CharacterIndex Controller::GetPrimaryCursorPosition() const
1603 {
1604   return mImpl->GetPrimaryCursorPosition();
1605 }
1606
1607 bool Controller::SetPrimaryCursorPosition(CharacterIndex index, bool focused)
1608 {
1609   if(mImpl->mEventData)
1610   {
1611     mImpl->mEventData->mCheckScrollAmount     = true;
1612     mImpl->mEventData->mIsLeftHandleSelected  = true;
1613     mImpl->mEventData->mIsRightHandleSelected = true;
1614     mImpl->mEventData->mCheckScrollAmount     = true;
1615     if(mImpl->SetPrimaryCursorPosition(index, focused) && focused)
1616     {
1617       EventHandler::KeyboardFocusGainEvent(*this);
1618       return true;
1619     }
1620   }
1621   return false;
1622 }
1623
1624 void Controller::SelectWholeText()
1625 {
1626   EventHandler::SelectEvent(*this, 0.f, 0.f, SelectionType::ALL);
1627 }
1628
1629 void Controller::SelectNone()
1630 {
1631   EventHandler::SelectEvent(*this, 0.f, 0.f, SelectionType::NONE);
1632 }
1633
1634 void Controller::SelectText(const uint32_t start, const uint32_t end)
1635 {
1636   EventHandler::SelectEvent(*this, start, end, SelectionType::RANGE);
1637 }
1638
1639 string Controller::GetSelectedText() const
1640 {
1641   return mImpl->GetSelectedText();
1642 }
1643
1644 string Controller::CopyText()
1645 {
1646   return mImpl->CopyText();
1647 }
1648
1649 string Controller::CutText()
1650 {
1651   return mImpl->CutText();
1652 }
1653
1654 void Controller::PasteClipboardItemEvent(uint32_t id, const char* mimeType, const char* data)
1655 {
1656   // Upon receiving the data, it is important to disconnect the signal
1657   // to avoid potential unintended pasting caused by subsequent requests.
1658   mImpl->mClipboard.DataReceivedSignal().Disconnect(this, &Controller::PasteClipboardItemEvent);
1659
1660   // If the id is 0u, it is an invalid response.
1661   // text-controller allows only plain text type.
1662   if(id != 0u && !strncmp(mimeType, MIME_TYPE_TEXT_PLAIN, strlen(MIME_TYPE_TEXT_PLAIN)))
1663   {
1664     EventHandler::PasteClipboardItemEvent(*this, data);
1665   }
1666 }
1667
1668 void Controller::PasteText()
1669 {
1670   if(mImpl->EnsureClipboardCreated())
1671   {
1672     // Connect the signal before calling GetData() of the clipboard.
1673     mImpl->mClipboard.DataReceivedSignal().Connect(this, &Controller::PasteClipboardItemEvent);
1674
1675     // Request clipboard service to retrieve an item.
1676     uint id = mImpl->mClipboard.GetData(MIME_TYPE_TEXT_PLAIN);
1677     if(id == 0u)
1678     {
1679       // If the return id is 0u, the signal is not emitted, we must disconnect signal here.
1680       mImpl->mClipboard.DataReceivedSignal().Disconnect(this, &Controller::PasteClipboardItemEvent);
1681     }
1682   }
1683 }
1684
1685 InputMethodContext::CallbackData Controller::OnInputMethodContextEvent(InputMethodContext& inputMethodContext, const InputMethodContext::EventData& inputMethodContextEvent)
1686 {
1687   return EventHandler::OnInputMethodContextEvent(*this, inputMethodContext, inputMethodContextEvent);
1688 }
1689
1690 void Controller::GetTargetSize(Vector2& targetSize)
1691 {
1692   targetSize = mImpl->mModel->mVisualModel->mControlSize;
1693 }
1694
1695 void Controller::AddDecoration(Actor& actor, DecorationType type, bool needsClipping)
1696 {
1697   if(mImpl->mEditableControlInterface)
1698   {
1699     mImpl->mEditableControlInterface->AddDecoration(actor, type, needsClipping);
1700   }
1701 }
1702
1703 bool Controller::IsEditable() const
1704 {
1705   return mImpl->IsEditable();
1706 }
1707
1708 void Controller::SetEditable(bool editable)
1709 {
1710   mImpl->SetEditable(editable);
1711 }
1712
1713 void Controller::ScrollBy(Vector2 scroll)
1714 {
1715   mImpl->ScrollBy(scroll);
1716 }
1717
1718 bool Controller::IsScrollable(const Vector2& displacement)
1719 {
1720   return mImpl->IsScrollable(displacement);
1721 }
1722
1723 float Controller::GetHorizontalScrollPosition()
1724 {
1725   return mImpl->GetHorizontalScrollPosition();
1726 }
1727
1728 float Controller::GetVerticalScrollPosition()
1729 {
1730   return mImpl->GetVerticalScrollPosition();
1731 }
1732
1733 void Controller::DecorationEvent(HandleType handleType, HandleState state, float x, float y)
1734 {
1735   EventHandler::DecorationEvent(*this, handleType, state, x, y);
1736 }
1737
1738 void Controller::TextPopupButtonTouched(Dali::Toolkit::TextSelectionPopup::Buttons button)
1739 {
1740   EventHandler::TextPopupButtonTouched(*this, button);
1741 }
1742
1743 void Controller::DisplayTimeExpired()
1744 {
1745   mImpl->mEventData->mUpdateCursorPosition = true;
1746   // Apply modifications to the model
1747   mImpl->mOperationsPending = ALL_OPERATIONS;
1748
1749   mImpl->RequestRelayout();
1750 }
1751
1752 void Controller::ResetCursorPosition(CharacterIndex cursorIndex)
1753 {
1754   // Reset the cursor position
1755   if(NULL != mImpl->mEventData)
1756   {
1757     mImpl->mEventData->mPrimaryCursorPosition = cursorIndex;
1758
1759     // Update the cursor if it's in editing mode.
1760     if(EventData::IsEditingState(mImpl->mEventData->mState))
1761     {
1762       mImpl->mEventData->mUpdateCursorPosition = true;
1763     }
1764   }
1765 }
1766
1767 CharacterIndex Controller::GetCursorPosition()
1768 {
1769   return mImpl->mEventData ? mImpl->mEventData->mPrimaryCursorPosition : 0;
1770 }
1771
1772 void Controller::SetControlInterface(ControlInterface* controlInterface)
1773 {
1774   mImpl->mControlInterface = controlInterface;
1775 }
1776
1777 void Controller::SetAnchorControlInterface(AnchorControlInterface* anchorControlInterface)
1778 {
1779   mImpl->mAnchorControlInterface = anchorControlInterface;
1780 }
1781
1782 bool Controller::ShouldClearFocusOnEscape() const
1783 {
1784   return mImpl->mShouldClearFocusOnEscape;
1785 }
1786
1787 Actor Controller::CreateBackgroundActor()
1788 {
1789   return CreateControllerBackgroundActor(mImpl->mView, mImpl->mModel->mVisualModel, mImpl->mModel->mLogicalModel, mImpl->mShaderBackground);
1790 }
1791
1792 void Controller::GetAnchorActors(std::vector<Toolkit::TextAnchor>& anchorActors)
1793 {
1794   mImpl->GetAnchorActors(anchorActors);
1795 }
1796
1797 int Controller::GetAnchorIndex(size_t characterOffset)
1798 {
1799   return mImpl->GetAnchorIndex(characterOffset);
1800 }
1801
1802 Controller::Controller(ControlInterface*           controlInterface,
1803                        EditableControlInterface*   editableControlInterface,
1804                        SelectableControlInterface* selectableControlInterface,
1805                        AnchorControlInterface*     anchorControlInterface)
1806 : mImpl(new Controller::Impl(controlInterface, editableControlInterface, selectableControlInterface, anchorControlInterface))
1807 {
1808 }
1809
1810 Controller::~Controller()
1811 {
1812 }
1813
1814 } // namespace Dali::Toolkit::Text