11 #include "flutter/fml/string_conversion.h"
24 "TextInput.setEditableSizeAndTransform";
29 "TextInputClient.updateEditingState";
31 "TextInputClient.updateEditingStateWithDeltas";
41 static constexpr
char kViewId[] =
"viewId";
52 static constexpr
char kXKey[] =
"x";
53 static constexpr
char kYKey[] =
"y";
62 "Internal Consistency Error";
69 if (active_model_ ==
nullptr) {
72 std::u16string text_before_change =
73 fml::Utf8ToUtf16(active_model_->GetText());
74 TextRange selection_before_change = active_model_->selection();
75 active_model_->AddText(
text);
77 if (enable_delta_model) {
80 SendStateUpdateWithDelta(*active_model_, &delta);
82 SendStateUpdate(*active_model_);
92 if (active_model_ ==
nullptr) {
100 EnterPressed(active_model_.get());
115 active_model_(nullptr) {
116 channel_->SetMethodCallHandler(
120 HandleMethodCall(call, std::move(result));
127 if (active_model_ ==
nullptr) {
130 active_model_->BeginComposing();
131 if (enable_delta_model) {
132 std::string
text = active_model_->GetText();
133 TextRange selection = active_model_->selection();
135 SendStateUpdateWithDelta(*active_model_, &delta);
137 SendStateUpdate(*active_model_);
142 if (active_model_ ==
nullptr) {
145 std::string text_before_change = active_model_->GetText();
146 TextRange selection_before_change = active_model_->selection();
147 TextRange composing_before_change = active_model_->composing_range();
148 std::string composing_text_before_change = text_before_change.substr(
149 composing_before_change.
start(), composing_before_change.
length());
150 active_model_->CommitComposing();
177 if (active_model_ ==
nullptr) {
180 std::string text_before_change = active_model_->GetText();
181 TextRange selection_before_change = active_model_->selection();
182 active_model_->CommitComposing();
183 active_model_->EndComposing();
184 if (enable_delta_model) {
185 std::string
text = active_model_->GetText();
187 SendStateUpdateWithDelta(*active_model_, &delta);
189 SendStateUpdate(*active_model_);
195 if (active_model_ ==
nullptr) {
198 std::string text_before_change = active_model_->GetText();
199 TextRange composing_before_change = active_model_->composing_range();
200 active_model_->AddText(
text);
201 active_model_->UpdateComposingText(
text,
TextRange(cursor_pos, cursor_pos));
202 std::string text_after_change = active_model_->GetText();
203 if (enable_delta_model) {
205 fml::Utf8ToUtf16(text_before_change), composing_before_change,
text);
206 SendStateUpdateWithDelta(*active_model_, &delta);
208 SendStateUpdate(*active_model_);
212 void TextInputPlugin::HandleMethodCall(
215 const std::string& method = method_call.
method_name();
220 FlutterWindowsView* view = engine_->
view(view_id_);
221 if (view ==
nullptr) {
222 std::stringstream ss;
223 ss <<
"Text input is not available because view with view_id=" << view_id_
224 <<
" cannot be found";
228 if (active_model_ !=
nullptr && active_model_->composing()) {
229 active_model_->CommitComposing();
230 active_model_->EndComposing();
231 SendStateUpdate(*active_model_);
233 view->OnResetImeComposing();
234 active_model_ =
nullptr;
240 const rapidjson::Document& args = *method_call.
arguments();
242 const rapidjson::Value& client_id_json = args[0];
243 const rapidjson::Value& client_config = args[1];
244 if (client_id_json.IsNull()) {
248 if (client_config.IsNull()) {
250 "Could not set client, missing arguments.");
253 client_id_ = client_id_json.GetInt();
255 if (enable_delta_model_json != client_config.MemberEnd() &&
256 enable_delta_model_json->value.IsBool()) {
257 enable_delta_model = enable_delta_model_json->value.GetBool();
259 auto view_id_json = client_config.FindMember(
kViewId);
260 if (view_id_json != client_config.MemberEnd() &&
261 view_id_json->value.IsInt()) {
262 view_id_ = view_id_json->value.GetInt();
265 "Could not set client, view ID is null.");
270 if (input_action_json != client_config.MemberEnd() &&
271 input_action_json->value.IsString()) {
272 input_action_ = input_action_json->value.GetString();
275 auto input_type_info_json = client_config.FindMember(
kTextInputType);
276 if (input_type_info_json != client_config.MemberEnd() &&
277 input_type_info_json->value.IsObject()) {
278 auto input_type_json =
280 if (input_type_json != input_type_info_json->value.MemberEnd() &&
281 input_type_json->value.IsString()) {
282 input_type_ = input_type_json->value.GetString();
285 active_model_ = std::make_unique<TextInputModel>();
291 const rapidjson::Document& args = *method_call.
arguments();
293 if (active_model_ ==
nullptr) {
296 "Set editing state has been invoked, but no client is set.");
300 if (
text == args.MemberEnd() ||
text->value.IsNull()) {
302 "Set editing state has been invoked, but without text.");
307 if (base == args.MemberEnd() || base->value.IsNull() ||
308 extent == args.MemberEnd() || extent->value.IsNull()) {
310 "Selection base/extent values invalid.");
314 int selection_base = base->value.GetInt();
315 int selection_extent = extent->value.GetInt();
316 if (selection_base == -1 && selection_extent == -1) {
317 selection_base = selection_extent = 0;
319 active_model_->SetText(
text->value.GetString());
320 active_model_->SetSelection(TextRange(selection_base, selection_extent));
324 if (base == args.MemberEnd() || base->value.IsNull() ||
325 extent == args.MemberEnd() || extent->value.IsNull()) {
327 "Composing base/extent values invalid.");
330 int composing_base = base->value.GetInt();
331 int composing_extent = base->value.GetInt();
332 if (composing_base == -1 && composing_extent == -1) {
333 active_model_->EndComposing();
335 int composing_start = std::min(composing_base, composing_extent);
336 int cursor_offset = selection_base - composing_start;
337 active_model_->SetComposingRange(
338 TextRange(composing_base, composing_extent), cursor_offset);
341 FlutterWindowsView* view = engine_->
view(view_id_);
342 if (view ==
nullptr) {
343 std::stringstream ss;
344 ss <<
"Text input is not available because view with view_id=" << view_id_
345 <<
" cannot be found";
353 const rapidjson::Document& args = *method_call.
arguments();
354 auto x = args.FindMember(
kXKey);
355 auto y = args.FindMember(
kYKey);
358 if (x == args.MemberEnd() || x->value.IsNull() ||
359 y == args.MemberEnd() || y->value.IsNull() ||
360 width == args.MemberEnd() || width->value.IsNull() ||
361 height == args.MemberEnd() || height->value.IsNull()) {
363 "Composing rect values invalid.");
366 composing_rect_ = {{x->value.GetDouble(), y->value.GetDouble()},
367 {width->value.GetDouble(), height->value.GetDouble()}};
369 Rect transformed_rect = GetCursorRect();
370 view->OnCursorRectUpdated(transformed_rect);
372 FlutterWindowsView* view = engine_->
view(view_id_);
373 if (view ==
nullptr) {
374 std::stringstream ss;
375 ss <<
"Text input is not available because view with view_id=" << view_id_
376 <<
" cannot be found";
384 const rapidjson::Document& args = *method_call.
arguments();
386 if (transform == args.MemberEnd() || transform->value.IsNull() ||
387 !transform->value.IsArray() || transform->value.Size() != 16) {
389 "EditableText transform invalid.");
393 for (
auto& entry : transform->value.GetArray()) {
394 if (entry.IsNull()) {
396 "EditableText transform contains null value.");
399 editabletext_transform_[i / 4][i % 4] = entry.GetDouble();
402 Rect transformed_rect = GetCursorRect();
403 view->OnCursorRectUpdated(transformed_rect);
405 result->NotImplemented();
413 Rect TextInputPlugin::GetCursorRect()
const {
414 Point transformed_point = {
415 composing_rect_.
left() * editabletext_transform_[0][0] +
416 composing_rect_.
top() * editabletext_transform_[1][0] +
417 editabletext_transform_[3][0],
418 composing_rect_.
left() * editabletext_transform_[0][1] +
419 composing_rect_.
top() * editabletext_transform_[1][1] +
420 editabletext_transform_[3][1]};
421 return {transformed_point, composing_rect_.
size()};
424 void TextInputPlugin::SendStateUpdate(
const TextInputModel& model) {
425 auto args = std::make_unique<rapidjson::Document>(rapidjson::kArrayType);
426 auto& allocator = args->GetAllocator();
427 args->PushBack(client_id_, allocator);
429 TextRange selection = model.selection();
430 rapidjson::Value editing_state(rapidjson::kObjectType);
437 int composing_base = model.composing() ? model.composing_range().base() : -1;
438 int composing_extent =
439 model.composing() ? model.composing_range().extent() : -1;
442 editing_state.AddMember(
443 kTextKey, rapidjson::Value(model.GetText(), allocator).Move(), allocator);
444 args->PushBack(editing_state, allocator);
449 void TextInputPlugin::SendStateUpdateWithDelta(
const TextInputModel& model,
450 const TextEditingDelta* delta) {
451 auto args = std::make_unique<rapidjson::Document>(rapidjson::kArrayType);
452 auto& allocator = args->GetAllocator();
453 args->PushBack(client_id_, allocator);
455 rapidjson::Value object(rapidjson::kObjectType);
456 rapidjson::Value deltas(rapidjson::kArrayType);
457 rapidjson::Value deltaJson(rapidjson::kObjectType);
460 deltaJson.AddMember(
kDeltaTextKey, delta->delta_text(), allocator);
461 deltaJson.AddMember(
kDeltaStartKey, delta->delta_start(), allocator);
462 deltaJson.AddMember(
kDeltaEndKey, delta->delta_end(), allocator);
464 TextRange selection = model.selection();
470 int composing_base = model.composing() ? model.composing_range().base() : -1;
471 int composing_extent =
472 model.composing() ? model.composing_range().extent() : -1;
476 deltas.PushBack(deltaJson, allocator);
477 object.AddMember(
kDeltasKey, deltas, allocator);
478 args->PushBack(
object, allocator);
483 void TextInputPlugin::EnterPressed(TextInputModel* model) {
486 std::u16string text_before_change = fml::Utf8ToUtf16(model->GetText());
487 TextRange selection_before_change = model->selection();
488 model->AddText(u
"\n");
489 if (enable_delta_model) {
490 TextEditingDelta delta(text_before_change, selection_before_change,
492 SendStateUpdateWithDelta(*model, &delta);
494 SendStateUpdate(*model);
497 auto args = std::make_unique<rapidjson::Document>(rapidjson::kArrayType);
498 auto& allocator = args->GetAllocator();
499 args->PushBack(client_id_, allocator);
500 args->PushBack(rapidjson::Value(input_action_, allocator).Move(), allocator);