Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
7db25c36
Unverified
Commit
7db25c36
authored
Aug 17, 2022
by
Camille Simon
Committed by
GitHub
Aug 17, 2022
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Re-land Add Spell Check to EditableText (#109643)
parent
caafc893
Changes
32
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
32 changed files
with
1928 additions
and
3 deletions
+1928
-3
.ci.yaml
.ci.yaml
+11
-0
TESTOWNERS
TESTOWNERS
+1
-0
spell_check_test.dart
dev/devicelab/bin/tasks/spell_check_test.dart
+12
-0
integration_tests.dart
dev/devicelab/lib/tasks/integration_tests.dart
+7
-0
README.md
dev/integration_tests/spell_check/README.md
+3
-0
build.gradle
dev/integration_tests/spell_check/android/app/build.gradle
+75
-0
AndroidManifest.xml
...sts/spell_check/android/app/src/debug/AndroidManifest.xml
+12
-0
AndroidManifest.xml
...ests/spell_check/android/app/src/main/AndroidManifest.xml
+38
-0
MainActivity.kt
...p/src/main/kotlin/com/example/sc_int_test/MainActivity.kt
+6
-0
launch_background.xml
...k/android/app/src/main/res/drawable/launch_background.xml
+16
-0
ic_launcher.png
...heck/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
+0
-0
styles.xml
...ts/spell_check/android/app/src/main/res/values/styles.xml
+22
-0
AndroidManifest.xml
...s/spell_check/android/app/src/profile/AndroidManifest.xml
+12
-0
build.gradle
dev/integration_tests/spell_check/android/build.gradle
+35
-0
gradle.properties
dev/integration_tests/spell_check/android/gradle.properties
+3
-0
gradle-wrapper.properties
...ll_check/android/gradle/wrapper/gradle-wrapper.properties
+6
-0
settings.gradle
dev/integration_tests/spell_check/android/settings.gradle
+15
-0
integration_test.dart
..._tests/spell_check/integration_test/integration_test.dart
+188
-0
main.dart
dev/integration_tests/spell_check/lib/main.dart
+57
-0
pubspec.yaml
dev/integration_tests/spell_check/pubspec.yaml
+108
-0
integration_test.dart
...ation_tests/spell_check/test_driver/integration_test.dart
+7
-0
services.dart
packages/flutter/lib/services.dart
+1
-0
text_field.dart
packages/flutter/lib/src/cupertino/text_field.dart
+35
-0
text_field.dart
packages/flutter/lib/src/material/text_field.dart
+35
-0
spell_check.dart
packages/flutter/lib/src/services/spell_check.dart
+220
-0
system_channels.dart
packages/flutter/lib/src/services/system_channels.dart
+22
-0
editable_text.dart
packages/flutter/lib/src/widgets/editable_text.dart
+129
-2
spell_check.dart
packages/flutter/lib/src/widgets/spell_check.dart
+330
-0
widgets.dart
packages/flutter/lib/widgets.dart
+1
-0
editable_text_test.dart
packages/flutter/test/widgets/editable_text_test.dart
+161
-1
spell_check_test.dart
packages/flutter/test/widgets/spell_check_test.dart
+341
-0
window.dart
packages/flutter_test/lib/src/window.dart
+19
-0
No files found.
.ci.yaml
View file @
7db25c36
...
...
@@ -2102,6 +2102,17 @@ targets:
["devicelab", "android", "linux"]
task_name
:
routing_test
-
name
:
Linux_android spell_check_test
bringup
:
true
recipe
:
devicelab/devicelab_drone
presubmit
:
false
timeout
:
60
properties
:
tags
:
>
["devicelab", "android", "linux"]
task_name
:
spell_check_test
scheduler
:
luci
-
name
:
Linux_android service_extensions_test
recipe
:
devicelab/devicelab_drone
presubmit
:
false
...
...
TESTOWNERS
View file @
7db25c36
...
...
@@ -86,6 +86,7 @@
/dev/devicelab/bin/tasks/gradient_static_perf__e2e_summary.dart @flar @flutter/engine
/dev/devicelab/bin/tasks/animated_complex_opacity_perf__e2e_summary.dart @jonahwilliams @flutter/engine
/dev/devicelab/bin/tasks/openpay_benchmarks__scroll_perf.dart @iskakaushik @flutter/engine
/dev/devicelab/bin/tasks/spell_check_test.dart @camsim99 @flutter/android
## Windows Android DeviceLab tests
/dev/devicelab/bin/tasks/basic_material_app_win__compile.dart @zanderso @flutter/tool
...
...
dev/devicelab/bin/tasks/spell_check_test.dart
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import
'package:flutter_devicelab/framework/devices.dart'
;
import
'package:flutter_devicelab/framework/framework.dart'
;
import
'package:flutter_devicelab/tasks/integration_tests.dart'
;
Future
<
void
>
main
()
async
{
deviceOperatingSystem
=
DeviceOperatingSystem
.
android
;
await
task
(
createSpellCheckIntegrationTest
());
}
dev/devicelab/lib/tasks/integration_tests.dart
View file @
7db25c36
...
...
@@ -142,6 +142,13 @@ TaskFunction createEndToEndIntegrationTest() {
);
}
TaskFunction
createSpellCheckIntegrationTest
(
)
{
return
IntegrationTest
(
'
${flutterDirectory.path}
/dev/integration_tests/spell_check'
,
'integration_test/integration_test.dart'
,
);
}
class
DriverTest
{
DriverTest
(
this
.
testDirectory
,
...
...
dev/integration_tests/spell_check/README.md
0 → 100644
View file @
7db25c36
# spell_check
A Flutter project for testing spell check for
[
EditableText
]
.
\ No newline at end of file
dev/integration_tests/spell_check/android/app/build.gradle
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
def
localProperties
=
new
Properties
()
def
localPropertiesFile
=
rootProject
.
file
(
'local.properties'
)
if
(
localPropertiesFile
.
exists
())
{
localPropertiesFile
.
withReader
(
'UTF-8'
)
{
reader
->
localProperties
.
load
(
reader
)
}
}
def
flutterRoot
=
localProperties
.
getProperty
(
'flutter.sdk'
)
if
(
flutterRoot
==
null
)
{
throw
new
GradleException
(
"Flutter SDK not found. Define location with flutter.sdk in the local.properties file."
)
}
def
flutterVersionCode
=
localProperties
.
getProperty
(
'flutter.versionCode'
)
if
(
flutterVersionCode
==
null
)
{
flutterVersionCode
=
'1'
}
def
flutterVersionName
=
localProperties
.
getProperty
(
'flutter.versionName'
)
if
(
flutterVersionName
==
null
)
{
flutterVersionName
=
'1.0'
}
apply
plugin:
'com.android.application'
apply
plugin:
'kotlin-android'
apply
from:
"$flutterRoot/packages/flutter_tools/gradle/flutter.gradle"
android
{
compileSdkVersion
flutter
.
compileSdkVersion
ndkVersion
flutter
.
ndkVersion
compileOptions
{
sourceCompatibility
JavaVersion
.
VERSION_1_8
targetCompatibility
JavaVersion
.
VERSION_1_8
}
kotlinOptions
{
jvmTarget
=
'1.8'
}
sourceSets
{
main
.
java
.
srcDirs
+=
'src/main/kotlin'
}
defaultConfig
{
// TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html).
applicationId
"com.example.spell_check"
// You can update the following values to match your application needs.
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-build-configuration.
minSdkVersion
flutter
.
minSdkVersion
targetSdkVersion
flutter
.
targetSdkVersion
versionCode
flutterVersionCode
.
toInteger
()
versionName
flutterVersionName
}
buildTypes
{
release
{
// TODO: Add your own signing config for the release build.
// Signing with the debug keys for now, so `flutter run --release` works.
signingConfig
signingConfigs
.
debug
}
}
}
flutter
{
source
'../..'
}
dependencies
{
implementation
"org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
}
dev/integration_tests/spell_check/android/app/src/debug/AndroidManifest.xml
0 → 100644
View file @
7db25c36
<!-- Copyright 2014 The Flutter Authors. All rights reserved.
Use of this source code is governed by a BSD-style license that can be
found in the LICENSE file. -->
<manifest
xmlns:android=
"http://schemas.android.com/apk/res/android"
package=
"com.example.spell_check"
>
<!-- The INTERNET permission is required for development. Specifically,
the Flutter tool needs it to communicate with the running application
to allow setting breakpoints, to provide hot reload, etc.
-->
<uses-permission
android:name=
"android.permission.INTERNET"
/>
</manifest>
dev/integration_tests/spell_check/android/app/src/main/AndroidManifest.xml
0 → 100644
View file @
7db25c36
<!-- Copyright 2014 The Flutter Authors. All rights reserved.
Use of this source code is governed by a BSD-style license that can be
found in the LICENSE file. -->
<manifest
xmlns:android=
"http://schemas.android.com/apk/res/android"
package=
"com.example.spell_check"
>
<application
android:label=
"spell_check"
android:name=
"${applicationName}"
android:icon=
"@mipmap/ic_launcher"
>
<activity
android:name=
".MainActivity"
android:exported=
"true"
android:launchMode=
"singleTop"
android:theme=
"@style/LaunchTheme"
android:configChanges=
"orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode"
android:hardwareAccelerated=
"true"
android:windowSoftInputMode=
"adjustResize"
>
<!-- Specifies an Android theme to apply to this Activity as soon as
the Android process has started. This theme is visible to the user
while the Flutter UI initializes. After that, this theme continues
to determine the Window background behind the Flutter UI. -->
<meta-data
android:name=
"io.flutter.embedding.android.NormalTheme"
android:resource=
"@style/NormalTheme"
/>
<intent-filter>
<action
android:name=
"android.intent.action.MAIN"
/>
<category
android:name=
"android.intent.category.LAUNCHER"
/>
</intent-filter>
</activity>
<!-- Don't delete the meta-data below.
This is used by the Flutter tool to generate GeneratedPluginRegistrant.java -->
<meta-data
android:name=
"flutterEmbedding"
android:value=
"2"
/>
</application>
</manifest>
dev/integration_tests/spell_check/android/app/src/main/kotlin/com/example/sc_int_test/MainActivity.kt
0 → 100644
View file @
7db25c36
package
com.example.spell_check
import
io.flutter.embedding.android.FlutterActivity
class
MainActivity
:
FlutterActivity
()
{
}
dev/integration_tests/spell_check/android/app/src/main/res/drawable/launch_background.xml
0 → 100644
View file @
7db25c36
<!-- Copyright 2014 The Flutter Authors. All rights reserved.
Use of this source code is governed by a BSD-style license that can be
found in the LICENSE file. -->
<?xml version="1.0" encoding="utf-8"?>
<!-- Modify this file to customize your launch splash screen -->
<layer-list
xmlns:android=
"http://schemas.android.com/apk/res/android"
>
<item
android:drawable=
"@android:color/white"
/>
<!-- You can insert your own image assets here -->
<!-- <item>
<bitmap
android:gravity="center"
android:src="@mipmap/launch_image" />
</item> -->
</layer-list>
dev/integration_tests/spell_check/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
0 → 100644
View file @
7db25c36
This diff was suppressed by a .gitattributes entry.
dev/integration_tests/spell_check/android/app/src/main/res/values/styles.xml
0 → 100644
View file @
7db25c36
<!-- Copyright 2014 The Flutter Authors. All rights reserved.
Use of this source code is governed by a BSD-style license that can be
found in the LICENSE file. -->
<?xml version="1.0" encoding="utf-8"?>
<resources>
<!-- Theme applied to the Android Window while the process is starting when the OS's Dark Mode setting is off -->
<style
name=
"LaunchTheme"
parent=
"@android:style/Theme.Light.NoTitleBar"
>
<!-- Show a splash screen on the activity. Automatically removed when
Flutter draws its first frame -->
<item
name=
"android:windowBackground"
>
@drawable/launch_background
</item>
</style>
<!-- Theme applied to the Android Window as soon as the process has started.
This theme determines the color of the Android Window while your
Flutter UI initializes, as well as behind your Flutter UI while its
running.
This Theme is only used starting with V2 of Flutter's Android embedding. -->
<style
name=
"NormalTheme"
parent=
"@android:style/Theme.Light.NoTitleBar"
>
<item
name=
"android:windowBackground"
>
?android:colorBackground
</item>
</style>
</resources>
dev/integration_tests/spell_check/android/app/src/profile/AndroidManifest.xml
0 → 100644
View file @
7db25c36
<!-- Copyright 2014 The Flutter Authors. All rights reserved.
Use of this source code is governed by a BSD-style license that can be
found in the LICENSE file. -->
<manifest
xmlns:android=
"http://schemas.android.com/apk/res/android"
package=
"com.example.spell_check"
>
<!-- The INTERNET permission is required for development. Specifically,
the Flutter tool needs it to communicate with the running application
to allow setting breakpoints, to provide hot reload, etc.
-->
<uses-permission
android:name=
"android.permission.INTERNET"
/>
</manifest>
dev/integration_tests/spell_check/android/build.gradle
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
buildscript
{
ext
.
kotlin_version
=
'1.6.10'
repositories
{
google
()
mavenCentral
()
}
dependencies
{
classpath
'com.android.tools.build:gradle:7.1.2'
classpath
"org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
}
}
allprojects
{
repositories
{
google
()
mavenCentral
()
}
}
rootProject
.
buildDir
=
'../build'
subprojects
{
project
.
buildDir
=
"${rootProject.buildDir}/${project.name}"
}
subprojects
{
project
.
evaluationDependsOn
(
':app'
)
}
task
clean
(
type:
Delete
)
{
delete
rootProject
.
buildDir
}
dev/integration_tests/spell_check/android/gradle.properties
0 → 100644
View file @
7db25c36
org.gradle.jvmargs
=
-Xmx1536M
android.useAndroidX
=
true
android.enableJetifier
=
true
dev/integration_tests/spell_check/android/gradle/wrapper/gradle-wrapper.properties
0 → 100644
View file @
7db25c36
#Fri Jun 23 08:50:38 CEST 2017
distributionBase
=
GRADLE_USER_HOME
distributionPath
=
wrapper/dists
zipStoreBase
=
GRADLE_USER_HOME
zipStorePath
=
wrapper/dists
distributionUrl
=
https
\:
//services.gradle.org/distributions/gradle-7.4-all.zip
dev/integration_tests/spell_check/android/settings.gradle
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
include
':app'
def
localPropertiesFile
=
new
File
(
rootProject
.
projectDir
,
"local.properties"
)
def
properties
=
new
Properties
()
assert
localPropertiesFile
.
exists
()
localPropertiesFile
.
withReader
(
"UTF-8"
)
{
reader
->
properties
.
load
(
reader
)
}
def
flutterSdkPath
=
properties
.
getProperty
(
"flutter.sdk"
)
assert
flutterSdkPath
!=
null
,
"flutter.sdk not set in local.properties"
apply
from:
"$flutterSdkPath/packages/flutter_tools/gradle/app_plugin_loader.gradle"
dev/integration_tests/spell_check/integration_test/integration_test.dart
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import
'package:flutter/material.dart'
;
import
'package:flutter/rendering.dart'
;
import
'package:flutter/services.dart'
;
import
'package:flutter_test/flutter_test.dart'
;
import
'package:integration_test/integration_test.dart'
;
import
'package:spell_check/main.dart'
;
late
DefaultSpellCheckService
defaultSpellCheckService
;
late
Locale
locale
;
/// Copy from flutter/test/widgets/editable_text_utils.dart.
RenderEditable
findRenderEditable
(
WidgetTester
tester
,
Type
type
)
{
final
RenderObject
root
=
tester
.
renderObject
(
find
.
byType
(
type
));
expect
(
root
,
isNotNull
);
late
RenderEditable
renderEditable
;
void
recursiveFinder
(
RenderObject
child
)
{
if
(
child
is
RenderEditable
)
{
renderEditable
=
child
;
return
;
}
child
.
visitChildren
(
recursiveFinder
);
}
root
.
visitChildren
(
recursiveFinder
);
expect
(
renderEditable
,
isNotNull
);
return
renderEditable
;
}
Future
<
void
>
main
()
async
{
IntegrationTestWidgetsFlutterBinding
.
ensureInitialized
();
setUp
(()
{
defaultSpellCheckService
=
DefaultSpellCheckService
();
locale
=
const
Locale
(
'en'
,
'us'
);
});
test
(
'fetchSpellCheckSuggestions returns null with no misspelled words'
,
()
async
{
const
String
text
=
'Hello, world!'
;
final
List
<
SuggestionSpan
>?
spellCheckSuggestionSpans
=
await
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
text
);
expect
(
spellCheckSuggestionSpans
!.
length
,
equals
(
0
));
expect
(
defaultSpellCheckService
.
lastSavedResults
!.
spellCheckedText
,
equals
(
text
)
);
expect
(
defaultSpellCheckService
.
lastSavedResults
!.
suggestionSpans
,
equals
(
spellCheckSuggestionSpans
)
);
});
test
(
'fetchSpellCheckSuggestions returns correct ranges with misspelled words'
,
()
async
{
const
String
text
=
'Hlelo, world! Yuou are magnificente'
;
const
List
<
TextRange
>
misspelledWordRanges
=
<
TextRange
>[
TextRange
(
start:
0
,
end:
5
),
TextRange
(
start:
14
,
end:
18
),
TextRange
(
start:
23
,
end:
35
)
];
final
List
<
SuggestionSpan
>?
spellCheckSuggestionSpans
=
await
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
text
);
expect
(
spellCheckSuggestionSpans
,
isNotNull
);
expect
(
spellCheckSuggestionSpans
!.
length
,
equals
(
misspelledWordRanges
.
length
)
);
for
(
int
i
=
0
;
i
<
misspelledWordRanges
.
length
;
i
+=
1
)
{
expect
(
spellCheckSuggestionSpans
[
i
].
range
,
equals
(
misspelledWordRanges
[
i
])
);
}
expect
(
defaultSpellCheckService
.
lastSavedResults
!.
spellCheckedText
,
equals
(
text
)
);
expect
(
defaultSpellCheckService
.
lastSavedResults
!.
suggestionSpans
,
equals
(
spellCheckSuggestionSpans
)
);
});
test
(
'fetchSpellCheckSuggestions does not correct results when Gboard not ignoring composing region'
,
()
async
{
const
String
text
=
'Wwow, whaaett a beautiful day it is!'
;
final
List
<
SuggestionSpan
>?
spellCheckSpansWithComposingRegion
=
await
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
text
);
expect
(
spellCheckSpansWithComposingRegion
,
isNotNull
);
expect
(
spellCheckSpansWithComposingRegion
!.
length
,
equals
(
2
));
final
List
<
SuggestionSpan
>?
spellCheckSuggestionSpans
=
await
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
text
);
expect
(
spellCheckSuggestionSpans
,
equals
(
spellCheckSpansWithComposingRegion
)
);
});
test
(
'fetchSpellCheckSuggestions merges results when Gboard ignoring composing region'
,
()
async
{
const
String
text
=
'Wooahha it is an amazzinng dayyebf!'
;
final
List
<
SuggestionSpan
>?
modifiedSpellCheckSuggestionSpans
=
await
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
text
);
final
List
<
SuggestionSpan
>
expectedSpellCheckSuggestionSpans
=
List
<
SuggestionSpan
>.
from
(
modifiedSpellCheckSuggestionSpans
!);
expect
(
modifiedSpellCheckSuggestionSpans
,
isNotNull
);
expect
(
modifiedSpellCheckSuggestionSpans
.
length
,
equals
(
3
));
// Remove one span to simulate Gboard attempting to un-ignore the composing region, after tapping away from "Yuou".
modifiedSpellCheckSuggestionSpans
.
removeAt
(
1
);
defaultSpellCheckService
.
lastSavedResults
=
SpellCheckResults
(
text
,
modifiedSpellCheckSuggestionSpans
);
final
List
<
SuggestionSpan
>?
spellCheckSuggestionSpans
=
await
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
text
);
expect
(
spellCheckSuggestionSpans
,
isNotNull
);
expect
(
spellCheckSuggestionSpans
,
equals
(
expectedSpellCheckSuggestionSpans
)
);
});
testWidgets
(
'EditableText spell checks when text is entered and spell check enabled'
,
(
WidgetTester
tester
)
async
{
const
TextStyle
style
=
TextStyle
();
const
TextStyle
misspelledTextStyle
=
TextField
.
materialMisspelledTextStyle
;
await
tester
.
pumpWidget
(
const
MyApp
());
await
tester
.
enterText
(
find
.
byType
(
EditableText
),
'Hey wrororld! Hey!'
);
await
tester
.
pumpAndSettle
();
final
RenderEditable
renderEditable
=
findRenderEditable
(
tester
,
EditableText
);
final
TextSpan
textSpanTree
=
renderEditable
.
text
!
as
TextSpan
;
const
TextSpan
expectedTextSpanTree
=
TextSpan
(
style:
style
,
children:
<
TextSpan
>[
TextSpan
(
style:
style
,
text:
'Hey '
),
TextSpan
(
style:
misspelledTextStyle
,
text:
'wrororld'
),
TextSpan
(
style:
style
,
text:
'! Hey!'
),
]);
expect
(
textSpanTree
,
equals
(
expectedTextSpanTree
));
});
test
(
'fetchSpellCheckSuggestions returns null when there is a pending request'
,
()
async
{
final
String
text
=
'neaf niofenaifn iofn iefnaoeifn ifneoa finoiafn inf ionfieaon ienf ifn ieonfaiofneionf oieafn oifnaioe nioenfio nefaion oifan'
*
10
;
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
text
);
final
String
modifiedText
=
text
.
substring
(
5
);
final
List
<
SuggestionSpan
>?
spellCheckSuggestionSpans
=
await
defaultSpellCheckService
.
fetchSpellCheckSuggestions
(
locale
,
modifiedText
);
expect
(
spellCheckSuggestionSpans
,
isNull
);
// We expect it to be rare for the first request to complete before the
// second, so no text should be saved as of now.
expect
(
defaultSpellCheckService
.
lastSavedResults
,
null
);
});
}
dev/integration_tests/spell_check/lib/main.dart
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import
'package:flutter/material.dart'
;
void
main
(
)
{
runApp
(
const
MyApp
());
}
class
MyApp
extends
StatelessWidget
{
const
MyApp
({
super
.
key
});
@override
Widget
build
(
BuildContext
context
)
{
return
MaterialApp
(
title:
'Spellcheck Demo'
,
theme:
ThemeData
(
primarySwatch:
Colors
.
blue
,
),
home:
const
MyHomePage
(
title:
'Spellcheck Demo'
),
);
}
}
class
MyHomePage
extends
StatefulWidget
{
const
MyHomePage
({
super
.
key
,
required
this
.
title
});
final
String
title
;
@override
State
<
MyHomePage
>
createState
()
=>
_MyHomePageState
();
}
class
_MyHomePageState
extends
State
<
MyHomePage
>
{
@override
Widget
build
(
BuildContext
context
)
{
return
Scaffold
(
appBar:
AppBar
(
title:
Text
(
widget
.
title
),
),
body:
Center
(
child:
EditableText
(
controller:
TextEditingController
(),
backgroundCursorColor:
Colors
.
grey
,
focusNode:
FocusNode
(),
style:
const
TextStyle
(),
cursorColor:
Colors
.
red
,
spellCheckConfiguration:
const
SpellCheckConfiguration
(
misspelledTextStyle:
TextField
.
materialMisspelledTextStyle
,
)
)
),
);
}
}
dev/integration_tests/spell_check/pubspec.yaml
0 → 100644
View file @
7db25c36
name
:
spell_check
description
:
Integration test for spell check.
# The following line prevents the package from being accidentally published to
# pub.dev using `flutter pub publish`. This is preferred for private packages.
publish_to
:
'
none'
# Remove this line if you wish to publish to pub.dev
# The following defines the version and build number for your application.
# A version number is three numbers separated by dots, like 1.2.43
# followed by an optional build number separated by a +.
# Both the version and the builder number may be overridden in flutter
# build by specifying --build-name and --build-number, respectively.
# In Android, build-name is used as versionName while build-number used as versionCode.
# Read more about Android versioning at https://developer.android.com/studio/publish/versioning
# In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion.
# Read more about iOS versioning at
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
version
:
1.0.0+1
environment
:
sdk
:
'
>=2.18.0-149.0.dev
<3.0.0'
# Dependencies specify other packages that your package needs in order to work.
# To automatically upgrade your package dependencies to the latest versions
# consider running `flutter pub upgrade --major-versions`. Alternatively,
# dependencies can be manually updated by changing the version numbers below to
# the latest version available on pub.dev. To see which dependencies have newer
# versions available, run `flutter pub outdated`.
dependencies
:
flutter
:
sdk
:
flutter
# The following adds the Cupertino Icons font to your application.
# Use with the CupertinoIcons class for iOS style icons.
cupertino_icons
:
1.0.5
characters
:
1.2.1
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
collection
:
1.16.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
material_color_utilities
:
0.2.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
meta
:
1.8.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
vector_math
:
2.1.2
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
dev_dependencies
:
flutter_test
:
sdk
:
flutter
# Used to run the integration tests in this app:
integration_test
:
sdk
:
flutter
# For information on the generic Dart part of this file, see the
# following page: https://dart.dev/tools/pub/pubspec
# The following section is specific to Flutter packages.
async
:
2.9.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
boolean_selector
:
2.1.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
clock
:
1.1.1
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
fake_async
:
1.3.1
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
matcher
:
0.12.12
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
path
:
1.8.2
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
source_span
:
1.9.1
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
stack_trace
:
1.10.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
stream_channel
:
2.1.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
string_scanner
:
1.1.1
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
term_glyph
:
1.2.1
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
test_api
:
0.4.12
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
vm_service
:
9.3.0
# THIS LINE IS AUTOGENERATED - TO UPDATE USE "flutter update-packages --force-upgrade"
flutter
:
# The following line ensures that the Material Icons font is
# included with your application, so that you can use the icons in
# the material Icons class.
uses-material-design
:
true
# To add assets to your application, add an assets section, like this:
# assets:
# - images/a_dot_burr.jpeg
# - images/a_dot_ham.jpeg
# An image asset can refer to one or more resolution-specific "variants", see
# https://flutter.dev/assets-and-images/#resolution-aware
# For details regarding adding assets from package dependencies, see
# https://flutter.dev/assets-and-images/#from-packages
# To add custom fonts to your application, add a fonts section here,
# in this "flutter" section. Each entry in this list should have a
# "family" key with the font family name, and a "fonts" key with a
# list giving the asset and other descriptors for the font. For
# example:
# fonts:
# - family: Schyler
# fonts:
# - asset: fonts/Schyler-Regular.ttf
# - asset: fonts/Schyler-Italic.ttf
# style: italic
# - family: Trajan Pro
# fonts:
# - asset: fonts/TrajanPro.ttf
# - asset: fonts/TrajanPro_Bold.ttf
# weight: 700
#
# For details regarding fonts from package dependencies,
# see https://flutter.dev/custom-fonts/#from-packages
# PUBSPEC CHECKSUM: 53ec
dev/integration_tests/spell_check/test_driver/integration_test.dart
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import
'package:integration_test/integration_test_driver.dart'
;
Future
<
void
>
main
()
=>
integrationDriver
();
packages/flutter/lib/services.dart
View file @
7db25c36
...
...
@@ -37,6 +37,7 @@ export 'src/services/raw_keyboard_macos.dart';
export
'src/services/raw_keyboard_web.dart'
;
export
'src/services/raw_keyboard_windows.dart'
;
export
'src/services/restoration.dart'
;
export
'src/services/spell_check.dart'
;
export
'src/services/system_channels.dart'
;
export
'src/services/system_chrome.dart'
;
export
'src/services/system_navigator.dart'
;
...
...
packages/flutter/lib/src/cupertino/text_field.dart
View file @
7db25c36
...
...
@@ -273,6 +273,7 @@ class CupertinoTextField extends StatefulWidget {
this
.
restorationId
,
this
.
scribbleEnabled
=
true
,
this
.
enableIMEPersonalizedLearning
=
true
,
this
.
spellCheckConfiguration
,
this
.
magnifierConfiguration
,
})
:
assert
(
textAlign
!=
null
),
assert
(
readOnly
!=
null
),
...
...
@@ -435,6 +436,7 @@ class CupertinoTextField extends StatefulWidget {
this
.
restorationId
,
this
.
scribbleEnabled
=
true
,
this
.
enableIMEPersonalizedLearning
=
true
,
this
.
spellCheckConfiguration
,
this
.
magnifierConfiguration
,
})
:
assert
(
textAlign
!=
null
),
assert
(
readOnly
!=
null
),
...
...
@@ -800,6 +802,26 @@ class CupertinoTextField extends StatefulWidget {
// docs with images of what a magnifier is.
final
TextMagnifierConfiguration
?
magnifierConfiguration
;
/// {@macro flutter.widgets.EditableText.spellCheckConfiguration}
///
/// If [SpellCheckConfiguration.misspelledTextStyle] is not specified in this
/// configuration, then [cupertinoMisspelledTextStyle] is used by default.
final
SpellCheckConfiguration
?
spellCheckConfiguration
;
/// The [TextStyle] used to indicate misspelled words in the Cupertino style.
///
/// See also:
/// * [SpellCheckConfiguration.misspelledTextStyle], the style configured to
/// mark misspelled words with.
/// * [TextField.materialMisspelledTextStyle], the style configured
/// to mark misspelled words with in the Material style.
static
const
TextStyle
cupertinoMisspelledTextStyle
=
TextStyle
(
decoration:
TextDecoration
.
underline
,
decorationColor:
CupertinoColors
.
systemRed
,
decorationStyle:
TextDecorationStyle
.
dotted
,
);
@override
State
<
CupertinoTextField
>
createState
()
=>
_CupertinoTextFieldState
();
...
...
@@ -843,6 +865,7 @@ class CupertinoTextField extends StatefulWidget {
properties
.
add
(
DiagnosticsProperty
<
Clip
>(
'clipBehavior'
,
clipBehavior
,
defaultValue:
Clip
.
hardEdge
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'scribbleEnabled'
,
scribbleEnabled
,
defaultValue:
true
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'enableIMEPersonalizedLearning'
,
enableIMEPersonalizedLearning
,
defaultValue:
true
));
properties
.
add
(
DiagnosticsProperty
<
SpellCheckConfiguration
>(
'spellCheckConfiguration'
,
spellCheckConfiguration
,
defaultValue:
null
));
}
static
final
TextMagnifierConfiguration
_iosMagnifierConfiguration
=
TextMagnifierConfiguration
(
...
...
@@ -1282,6 +1305,17 @@ class _CupertinoTextFieldState extends State<CupertinoTextField> with Restoratio
context
,
)
??
CupertinoTheme
.
of
(
context
).
primaryColor
.
withOpacity
(
0.2
);
// Set configuration as disabled if not otherwise specified. If specified,
// ensure that configuration uses Cupertino text style for misspelled words
// unless a custom style is specified.
final
SpellCheckConfiguration
spellCheckConfiguration
=
widget
.
spellCheckConfiguration
!=
null
&&
widget
.
spellCheckConfiguration
!=
const
SpellCheckConfiguration
.
disabled
()
?
widget
.
spellCheckConfiguration
!.
copyWith
(
misspelledTextStyle:
widget
.
spellCheckConfiguration
!.
misspelledTextStyle
??
CupertinoTextField
.
cupertinoMisspelledTextStyle
)
:
const
SpellCheckConfiguration
.
disabled
();
final
Widget
paddedEditable
=
Padding
(
padding:
widget
.
padding
,
child:
RepaintBoundary
(
...
...
@@ -1346,6 +1380,7 @@ class _CupertinoTextFieldState extends State<CupertinoTextField> with Restoratio
restorationId:
'editable'
,
scribbleEnabled:
widget
.
scribbleEnabled
,
enableIMEPersonalizedLearning:
widget
.
enableIMEPersonalizedLearning
,
spellCheckConfiguration:
spellCheckConfiguration
,
),
),
),
...
...
packages/flutter/lib/src/material/text_field.dart
View file @
7db25c36
...
...
@@ -10,6 +10,7 @@ import 'package:flutter/gestures.dart';
import
'package:flutter/rendering.dart'
;
import
'package:flutter/services.dart'
;
import
'colors.dart'
;
import
'debug.dart'
;
import
'desktop_text_selection.dart'
;
import
'feedback.dart'
;
...
...
@@ -333,6 +334,7 @@ class TextField extends StatefulWidget {
this
.
restorationId
,
this
.
scribbleEnabled
=
true
,
this
.
enableIMEPersonalizedLearning
=
true
,
this
.
spellCheckConfiguration
,
this
.
magnifierConfiguration
,
})
:
assert
(
textAlign
!=
null
),
assert
(
readOnly
!=
null
),
...
...
@@ -800,6 +802,26 @@ class TextField extends StatefulWidget {
/// {@macro flutter.services.TextInputConfiguration.enableIMEPersonalizedLearning}
final
bool
enableIMEPersonalizedLearning
;
/// {@macro flutter.widgets.EditableText.spellCheckConfiguration}
///
/// If [SpellCheckConfiguration.misspelledTextStyle] is not specified in this
/// configuration, then [materialMisspelledTextStyle] is used by default.
final
SpellCheckConfiguration
?
spellCheckConfiguration
;
/// The [TextStyle] used to indicate misspelled words in the Material style.
///
/// See also:
/// * [SpellCheckConfiguration.misspelledTextStyle], the style configured to
/// mark misspelled words with.
/// * [CupertinoTextField.cupertinoMisspelledTextStyle], the style configured
/// to mark misspelled words with in the Cupertino style.
static
const
TextStyle
materialMisspelledTextStyle
=
TextStyle
(
decoration:
TextDecoration
.
underline
,
decorationColor:
Colors
.
red
,
decorationStyle:
TextDecorationStyle
.
wavy
,
);
@override
State
<
TextField
>
createState
()
=>
_TextFieldState
();
...
...
@@ -842,6 +864,7 @@ class TextField extends StatefulWidget {
properties
.
add
(
DiagnosticsProperty
<
Clip
>(
'clipBehavior'
,
clipBehavior
,
defaultValue:
Clip
.
hardEdge
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'scribbleEnabled'
,
scribbleEnabled
,
defaultValue:
true
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'enableIMEPersonalizedLearning'
,
enableIMEPersonalizedLearning
,
defaultValue:
true
));
properties
.
add
(
DiagnosticsProperty
<
SpellCheckConfiguration
>(
'spellCheckConfiguration'
,
spellCheckConfiguration
,
defaultValue:
null
));
}
}
...
...
@@ -1187,6 +1210,17 @@ class _TextFieldState extends State<TextField> with RestorationMixin implements
),
];
// Set configuration as disabled if not otherwise specified. If specified,
// ensure that configuration uses Material text style for misspelled words
// unless a custom style is specified.
final
SpellCheckConfiguration
spellCheckConfiguration
=
widget
.
spellCheckConfiguration
!=
null
&&
widget
.
spellCheckConfiguration
!=
const
SpellCheckConfiguration
.
disabled
()
?
widget
.
spellCheckConfiguration
!.
copyWith
(
misspelledTextStyle:
widget
.
spellCheckConfiguration
!.
misspelledTextStyle
??
TextField
.
materialMisspelledTextStyle
)
:
const
SpellCheckConfiguration
.
disabled
();
TextSelectionControls
?
textSelectionControls
=
widget
.
selectionControls
;
final
bool
paintCursorAboveText
;
final
bool
cursorOpacityAnimates
;
...
...
@@ -1327,6 +1361,7 @@ class _TextFieldState extends State<TextField> with RestorationMixin implements
restorationId:
'editable'
,
scribbleEnabled:
widget
.
scribbleEnabled
,
enableIMEPersonalizedLearning:
widget
.
enableIMEPersonalizedLearning
,
spellCheckConfiguration:
spellCheckConfiguration
,
magnifierConfiguration:
widget
.
magnifierConfiguration
??
TextMagnifier
.
adaptiveMagnifierConfiguration
,
),
),
...
...
packages/flutter/lib/src/services/spell_check.dart
0 → 100644
View file @
7db25c36
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import
'dart:ui'
;
import
'package:flutter/foundation.dart'
;
import
'system_channels.dart'
;
/// A data structure representing a range of misspelled text and the suggested
/// replacements for this range.
///
/// For example, one [SuggestionSpan] of the
/// [List<SuggestionSpan>] suggestions of the [SpellCheckResults] corresponding
/// to "Hello, wrold!" may be:
/// ```dart
/// SuggestionSpan suggestionSpan =
/// SuggestionSpan(
/// const TextRange(start: 7, end: 12),
/// List<String>.of(<String>['word', 'world', 'old']),
/// );
/// ```
@immutable
class
SuggestionSpan
{
/// Creates a span representing a misspelled range of text and the replacements
/// suggested by a spell checker.
///
/// The [range] and replacement [suggestions] must all not
/// be null.
const
SuggestionSpan
(
this
.
range
,
this
.
suggestions
)
:
assert
(
range
!=
null
),
assert
(
suggestions
!=
null
);
/// The misspelled range of text.
final
TextRange
range
;
/// The alternate suggestions for the misspelled range of text.
final
List
<
String
>
suggestions
;
@override
bool
operator
==(
Object
other
)
{
if
(
identical
(
this
,
other
))
{
return
true
;
}
return
other
is
SuggestionSpan
&&
other
.
range
.
start
==
range
.
start
&&
other
.
range
.
end
==
range
.
end
&&
listEquals
<
String
>(
other
.
suggestions
,
suggestions
);
}
@override
int
get
hashCode
=>
Object
.
hash
(
range
.
start
,
range
.
end
,
Object
.
hashAll
(
suggestions
));
}
/// A data structure grouping together the [SuggestionSpan]s and related text of
/// results returned by a spell checker.
@immutable
class
SpellCheckResults
{
/// Creates results based off those received by spell checking some text input.
const
SpellCheckResults
(
this
.
spellCheckedText
,
this
.
suggestionSpans
)
:
assert
(
spellCheckedText
!=
null
),
assert
(
suggestionSpans
!=
null
);
/// The text that the [suggestionSpans] correspond to.
final
String
spellCheckedText
;
/// The spell check results of the [spellCheckedText].
///
/// See also:
///
/// * [SuggestionSpan], the ranges of misspelled text and corresponding
/// replacement suggestions.
final
List
<
SuggestionSpan
>
suggestionSpans
;
@override
bool
operator
==(
Object
other
)
{
if
(
identical
(
this
,
other
))
{
return
true
;
}
return
other
is
SpellCheckResults
&&
other
.
spellCheckedText
==
spellCheckedText
&&
listEquals
<
SuggestionSpan
>(
other
.
suggestionSpans
,
suggestionSpans
);
}
@override
int
get
hashCode
=>
Object
.
hash
(
spellCheckedText
,
Object
.
hashAll
(
suggestionSpans
));
}
/// Determines how spell check results are received for text input.
abstract
class
SpellCheckService
{
/// Facilitates a spell check request.
///
/// Returns a [Future] that resolves with a [List] of [SuggestionSpan]s for
/// all misspelled words in the given [String] for the given [Locale].
Future
<
List
<
SuggestionSpan
>?>
fetchSpellCheckSuggestions
(
Locale
locale
,
String
text
);
}
/// The service used by default to fetch spell check results for text input.
///
/// Any widget may use this service to spell check text by calling
/// `fetchSpellCheckSuggestions(locale, text)` with an instance of this class.
/// This is currently only supported by Android.
///
/// See also:
///
/// * [SpellCheckService], the service that this implements that may be
/// overriden for use by [EditableText].
/// * [EditableText], which may use this service to fetch results.
class
DefaultSpellCheckService
implements
SpellCheckService
{
/// Creates service to spell check text input by default via communcication
/// over the spell check [MethodChannel].
DefaultSpellCheckService
()
{
spellCheckChannel
=
SystemChannels
.
spellCheck
;
}
/// The last received results from the shell side.
SpellCheckResults
?
lastSavedResults
;
/// The channel used to communicate with the shell side to complete spell
/// check requests.
late
MethodChannel
spellCheckChannel
;
/// Merges two lists of spell check [SuggestionSpan]s.
///
/// Used in cases where the text has not changed, but the spell check results
/// received from the shell side have. This case is caused by IMEs (GBoard,
/// for instance) that ignore the composing region when spell checking text.
///
/// Assumes that the lists provided as parameters are sorted by range start
/// and that both list of [SuggestionSpan]s apply to the same text.
static
List
<
SuggestionSpan
>
mergeResults
(
List
<
SuggestionSpan
>
oldResults
,
List
<
SuggestionSpan
>
newResults
)
{
final
List
<
SuggestionSpan
>
mergedResults
=
<
SuggestionSpan
>[];
SuggestionSpan
oldSpan
;
SuggestionSpan
newSpan
;
int
oldSpanPointer
=
0
;
int
newSpanPointer
=
0
;
while
(
oldSpanPointer
<
oldResults
.
length
&&
newSpanPointer
<
newResults
.
length
)
{
oldSpan
=
oldResults
[
oldSpanPointer
];
newSpan
=
newResults
[
newSpanPointer
];
if
(
oldSpan
.
range
.
start
==
newSpan
.
range
.
start
)
{
mergedResults
.
add
(
oldSpan
);
oldSpanPointer
++;
newSpanPointer
++;
}
else
{
if
(
oldSpan
.
range
.
start
<
newSpan
.
range
.
start
)
{
mergedResults
.
add
(
oldSpan
);
oldSpanPointer
++;
}
else
{
mergedResults
.
add
(
newSpan
);
newSpanPointer
++;
}
}
}
mergedResults
.
addAll
(
oldResults
.
sublist
(
oldSpanPointer
));
mergedResults
.
addAll
(
newResults
.
sublist
(
newSpanPointer
));
return
mergedResults
;
}
@override
Future
<
List
<
SuggestionSpan
>?>
fetchSpellCheckSuggestions
(
Locale
locale
,
String
text
)
async
{
assert
(
locale
!=
null
);
assert
(
text
!=
null
);
final
List
<
dynamic
>
rawResults
;
final
String
languageTag
=
locale
.
toLanguageTag
();
try
{
rawResults
=
await
spellCheckChannel
.
invokeMethod
(
'SpellCheck.initiateSpellCheck'
,
<
String
>[
languageTag
,
text
],
)
as
List
<
dynamic
>;
}
catch
(
e
)
{
// Spell check request canceled due to pending request.
return
null
;
}
List
<
SuggestionSpan
>
suggestionSpans
=
<
SuggestionSpan
>[];
for
(
final
dynamic
result
in
rawResults
)
{
final
Map
<
String
,
dynamic
>
resultMap
=
Map
<
String
,
dynamic
>.
from
(
result
as
Map
<
dynamic
,
dynamic
>);
suggestionSpans
.
add
(
SuggestionSpan
(
TextRange
(
start:
resultMap
[
'startIndex'
]
as
int
,
end:
resultMap
[
'endIndex'
]
as
int
),
(
resultMap
[
'suggestions'
]
as
List
<
dynamic
>).
cast
<
String
>(),
)
);
}
if
(
lastSavedResults
!=
null
)
{
// Merge current and previous spell check results if between requests,
// the text has not changed but the spell check results have.
final
bool
textHasNotChanged
=
lastSavedResults
!.
spellCheckedText
==
text
;
final
bool
spansHaveChanged
=
listEquals
(
lastSavedResults
!.
suggestionSpans
,
suggestionSpans
);
if
(
textHasNotChanged
&&
spansHaveChanged
)
{
suggestionSpans
=
mergeResults
(
lastSavedResults
!.
suggestionSpans
,
suggestionSpans
);
}
lastSavedResults
=
SpellCheckResults
(
text
,
suggestionSpans
);
}
return
suggestionSpans
;
}
}
packages/flutter/lib/src/services/system_channels.dart
View file @
7db25c36
...
...
@@ -222,6 +222,28 @@ class SystemChannels {
JSONMethodCodec
(),
);
/// A [MethodChannel] for handling spell check for text input.
///
/// This channel exposes the spell check framework for supported platforms.
/// Currently supported on Android only.
///
/// Spell check requests are intiated by `SpellCheck.initiateSpellCheck`.
/// These requests may either be completed or canceled. If the request is
/// completed, the shell side will respond with the results of the request.
/// Otherwise, the shell side will respond with null.
///
/// The following outgoing methods are defined for this channel (invoked by
/// [OptionalMethodChannel.invokeMethod]):
///
/// * `SpellCheck.initiateSpellCheck`: Sends request for specified text to be
/// spell checked and returns the result, either a [List<SuggestionSpan>]
/// representing the spell check results of the text or null if the request
/// was cancelled. The arguments are the [String] to be spell checked
/// and the [Locale] for the text to be spell checked with.
static
const
MethodChannel
spellCheck
=
OptionalMethodChannel
(
'flutter/spellcheck'
,
);
/// A JSON [BasicMessageChannel] for keyboard events.
///
/// Each incoming message received on this channel (registered using
...
...
packages/flutter/lib/src/widgets/editable_text.dart
View file @
7db25c36
...
...
@@ -34,6 +34,7 @@ import 'scroll_controller.dart';
import
'scroll_physics.dart'
;
import
'scrollable.dart'
;
import
'shortcuts.dart'
;
import
'spell_check.dart'
;
import
'tap_region.dart'
;
import
'text.dart'
;
import
'text_editing_intents.dart'
;
...
...
@@ -171,9 +172,12 @@ class TextEditingController extends ValueNotifier<TextEditingValue> {
// If the composing range is out of range for the current text, ignore it to
// preserve the tree integrity, otherwise in release mode a RangeError will
// be thrown and this EditableText will be built with a broken subtree.
if
(!
value
.
isComposingRangeValid
||
!
withComposing
)
{
final
bool
composingRegionOutOfRange
=
!
value
.
isComposingRangeValid
||
!
withComposing
;
if
(
composingRegionOutOfRange
)
{
return
TextSpan
(
style:
style
,
text:
text
);
}
final
TextStyle
composingStyle
=
style
?.
merge
(
const
TextStyle
(
decoration:
TextDecoration
.
underline
))
??
const
TextStyle
(
decoration:
TextDecoration
.
underline
);
return
TextSpan
(
...
...
@@ -643,6 +647,7 @@ class EditableText extends StatefulWidget {
this
.
scrollBehavior
,
this
.
scribbleEnabled
=
true
,
this
.
enableIMEPersonalizedLearning
=
true
,
this
.
spellCheckConfiguration
,
this
.
magnifierConfiguration
=
TextMagnifierConfiguration
.
disabled
,
})
:
assert
(
controller
!=
null
),
assert
(
focusNode
!=
null
),
...
...
@@ -706,6 +711,12 @@ class EditableText extends StatefulWidget {
))),
assert
(
clipBehavior
!=
null
),
assert
(
enableIMEPersonalizedLearning
!=
null
),
assert
(
spellCheckConfiguration
==
null
||
spellCheckConfiguration
==
const
SpellCheckConfiguration
.
disabled
()
||
spellCheckConfiguration
.
misspelledTextStyle
!=
null
,
'spellCheckConfiguration must specify a misspelledTextStyle if spell check behavior is desired'
,
),
_strutStyle
=
strutStyle
,
keyboardType
=
keyboardType
??
_inferKeyboardType
(
autofillHints:
autofillHints
,
maxLines:
maxLines
),
inputFormatters
=
maxLines
==
1
...
...
@@ -1555,6 +1566,20 @@ class EditableText extends StatefulWidget {
/// {@macro flutter.services.TextInputConfiguration.enableIMEPersonalizedLearning}
final
bool
enableIMEPersonalizedLearning
;
/// {@template flutter.widgets.EditableText.spellCheckConfiguration}
/// Configuration that details how spell check should be performed.
///
/// Specifies the [SpellCheckService] used to spell check text input and the
/// [TextStyle] used to style text with misspelled words.
///
/// If the [SpellCheckService] is left null, spell check is disabled by
/// default unless the [DefaultSpellCheckService] is supported, in which case
/// it is used. It is currently supported only on Android.
///
/// If this configuration is left null, then spell check is disabled by default.
/// {@endtemplate}
final
SpellCheckConfiguration
?
spellCheckConfiguration
;
/// {@macro flutter.widgets.magnifier.TextMagnifierConfiguration.intro}
///
/// {@macro flutter.widgets.magnifier.intro}
...
...
@@ -1738,6 +1763,7 @@ class EditableText extends StatefulWidget {
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'scribbleEnabled'
,
scribbleEnabled
,
defaultValue:
true
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'enableIMEPersonalizedLearning'
,
enableIMEPersonalizedLearning
,
defaultValue:
true
));
properties
.
add
(
DiagnosticsProperty
<
bool
>(
'enableInteractiveSelection'
,
enableInteractiveSelection
,
defaultValue:
true
));
properties
.
add
(
DiagnosticsProperty
<
SpellCheckConfiguration
>(
'spellCheckConfiguration'
,
spellCheckConfiguration
,
defaultValue:
null
));
}
}
...
...
@@ -1774,6 +1800,31 @@ class EditableTextState extends State<EditableText> with AutomaticKeepAliveClien
AutofillClient
get
_effectiveAutofillClient
=>
widget
.
autofillClient
??
this
;
late
SpellCheckConfiguration
_spellCheckConfiguration
;
/// Configuration that determines how spell check will be performed.
///
/// If possible, this configuration will contain a default for the
/// [SpellCheckService] if it is not otherwise specified.
///
/// See also:
/// * [DefaultSpellCheckService], the spell check service used by default.
@visibleForTesting
SpellCheckConfiguration
get
spellCheckConfiguration
=>
_spellCheckConfiguration
;
/// Whether or not spell check is enabled.
///
/// Spell check is enabled when a [SpellCheckConfiguration] has been specified
/// for the widget.
bool
get
spellCheckEnabled
=>
_spellCheckConfiguration
.
spellCheckEnabled
;
/// The most up-to-date spell check results for text input.
///
/// These results will be updated via calls to spell check through a
/// [SpellCheckService] and used by this widget to build the [TextSpan] tree
/// for text input and menus for replacement suggestions of misspelled words.
SpellCheckResults
?
_spellCheckResults
;
/// Whether to create an input connection with the platform for text editing
/// or not.
///
...
...
@@ -1960,6 +2011,28 @@ class EditableTextState extends State<EditableText> with AutomaticKeepAliveClien
}
}
/// Infers the [SpellCheckConfiguration] used to perform spell check.
///
/// If spell check is enabled, this will try to infer a value for
/// the [SpellCheckService] if left unspecified.
static
SpellCheckConfiguration
_inferSpellCheckConfiguration
(
SpellCheckConfiguration
?
configuration
)
{
if
(
configuration
==
null
||
configuration
==
const
SpellCheckConfiguration
.
disabled
())
{
return
const
SpellCheckConfiguration
.
disabled
();
}
SpellCheckService
?
spellCheckService
=
configuration
.
spellCheckService
;
assert
(
spellCheckService
!=
null
||
WidgetsBinding
.
instance
.
platformDispatcher
.
nativeSpellCheckServiceDefined
,
'spellCheckService must be specified for this platform because no default service available'
,
);
spellCheckService
=
spellCheckService
??
DefaultSpellCheckService
();
return
configuration
.
copyWith
(
spellCheckService:
spellCheckService
);
}
// State lifecycle:
@override
...
...
@@ -1970,6 +2043,7 @@ class EditableTextState extends State<EditableText> with AutomaticKeepAliveClien
widget
.
focusNode
.
addListener
(
_handleFocusChanged
);
_scrollController
.
addListener
(
_updateSelectionOverlayForScroll
);
_cursorVisibilityNotifier
.
value
=
widget
.
showCursor
;
_spellCheckConfiguration
=
_inferSpellCheckConfiguration
(
widget
.
spellCheckConfiguration
);
}
// Whether `TickerMode.of(context)` is true and animations (like blinking the
...
...
@@ -2817,6 +2891,37 @@ class EditableTextState extends State<EditableText> with AutomaticKeepAliveClien
_lastBottomViewInset
=
WidgetsBinding
.
instance
.
window
.
viewInsets
.
bottom
;
}
Future
<
void
>
_performSpellCheck
(
final
String
text
)
async
{
try
{
final
Locale
?
localeForSpellChecking
=
widget
.
locale
??
Localizations
.
maybeLocaleOf
(
context
);
assert
(
localeForSpellChecking
!=
null
,
'Locale must be specified in widget or Localization widget must be in scope'
,
);
final
List
<
SuggestionSpan
>?
spellCheckResults
=
await
_spellCheckConfiguration
.
spellCheckService
!
.
fetchSpellCheckSuggestions
(
localeForSpellChecking
!,
text
);
if
(
spellCheckResults
==
null
)
{
// The request to fetch spell check suggestions was canceled due to ongoing request.
return
;
}
_spellCheckResults
=
SpellCheckResults
(
text
,
spellCheckResults
);
renderEditable
.
text
=
buildTextSpan
();
}
catch
(
exception
,
stack
)
{
FlutterError
.
reportError
(
FlutterErrorDetails
(
exception:
exception
,
stack:
stack
,
library
:
'widgets'
,
context:
ErrorDescription
(
'while performing spell check'
),
));
}
}
@pragma
(
'vm:notify-debugger-on-exception'
)
void
_formatAndSetValue
(
TextEditingValue
value
,
SelectionChangedCause
?
cause
,
{
bool
userInteraction
=
false
})
{
// Only apply input formatters if the text has changed (including uncommitted
...
...
@@ -2837,6 +2942,10 @@ class EditableTextState extends State<EditableText> with AutomaticKeepAliveClien
value
,
(
TextEditingValue
newValue
,
TextInputFormatter
formatter
)
=>
formatter
.
formatEditUpdate
(
_value
,
newValue
),
)
??
value
;
if
(
spellCheckEnabled
&&
value
.
text
.
isNotEmpty
&&
_value
.
text
!=
value
.
text
)
{
_performSpellCheck
(
value
.
text
);
}
}
catch
(
exception
,
stack
)
{
FlutterError
.
reportError
(
FlutterErrorDetails
(
exception:
exception
,
...
...
@@ -3732,12 +3841,30 @@ class EditableTextState extends State<EditableText> with AutomaticKeepAliveClien
],
);
}
final
bool
spellCheckResultsReceived
=
spellCheckEnabled
&&
_spellCheckResults
!=
null
;
final
bool
withComposing
=
!
widget
.
readOnly
&&
_hasFocus
;
if
(
spellCheckResultsReceived
)
{
// If the composing range is out of range for the current text, ignore it to
// preserve the tree integrity, otherwise in release mode a RangeError will
// be thrown and this EditableText will be built with a broken subtree.
assert
(!
_value
.
composing
.
isValid
||
!
withComposing
||
_value
.
isComposingRangeValid
);
final
bool
composingRegionOutOfRange
=
!
_value
.
isComposingRangeValid
||
!
withComposing
;
return
buildTextSpanWithSpellCheckSuggestions
(
_value
,
composingRegionOutOfRange
,
widget
.
style
,
_spellCheckConfiguration
.
misspelledTextStyle
!,
_spellCheckResults
!,
);
}
// Read only mode should not paint text composing.
return
widget
.
controller
.
buildTextSpan
(
context:
context
,
style:
widget
.
style
,
withComposing:
!
widget
.
readOnly
&&
_hasFocus
,
withComposing:
withComposing
,
);
}
}
...
...
packages/flutter/lib/src/widgets/spell_check.dart
0 → 100644
View file @
7db25c36
This diff is collapsed.
Click to expand it.
packages/flutter/lib/widgets.dart
View file @
7db25c36
...
...
@@ -128,6 +128,7 @@ export 'src/widgets/sliver_persistent_header.dart';
export
'src/widgets/sliver_prototype_extent_list.dart'
;
export
'src/widgets/slotted_render_object_widget.dart'
;
export
'src/widgets/spacer.dart'
;
export
'src/widgets/spell_check.dart'
;
export
'src/widgets/status_transitions.dart'
;
export
'src/widgets/table.dart'
;
export
'src/widgets/tap_region.dart'
;
...
...
packages/flutter/test/widgets/editable_text_test.dart
View file @
7db25c36
...
...
@@ -12643,6 +12643,164 @@ void main() {
});
});
group
(
'Spell check'
,
()
{
testWidgets
(
'Spell check configured properly when spell check disabled by default'
,
(
WidgetTester
tester
)
async
{
await
tester
.
pumpWidget
(
MaterialApp
(
home:
EditableText
(
controller:
TextEditingController
(
text:
'A'
),
focusNode:
FocusNode
(),
style:
const
TextStyle
(),
cursorColor:
Colors
.
blue
,
backgroundCursorColor:
Colors
.
grey
,
cursorOpacityAnimates:
true
,
autofillHints:
null
,
),
),
);
final
EditableTextState
state
=
tester
.
state
<
EditableTextState
>(
find
.
byType
(
EditableText
));
expect
(
state
.
spellCheckEnabled
,
isFalse
);
});
testWidgets
(
'Spell check configured properly when spell check disabled manually'
,
(
WidgetTester
tester
)
async
{
await
tester
.
pumpWidget
(
MaterialApp
(
home:
EditableText
(
controller:
TextEditingController
(
text:
'A'
),
focusNode:
FocusNode
(),
style:
const
TextStyle
(),
cursorColor:
Colors
.
blue
,
backgroundCursorColor:
Colors
.
grey
,
cursorOpacityAnimates:
true
,
autofillHints:
null
,
spellCheckConfiguration:
const
SpellCheckConfiguration
.
disabled
(),
),
),
);
final
EditableTextState
state
=
tester
.
state
<
EditableTextState
>(
find
.
byType
(
EditableText
));
expect
(
state
.
spellCheckEnabled
,
isFalse
);
});
testWidgets
(
'Error thrown when spell check configuration defined without specifying misspelled text style'
,
(
WidgetTester
tester
)
async
{
expect
(
()
{
EditableText
(
controller:
TextEditingController
(
text:
'A'
),
focusNode:
FocusNode
(),
style:
const
TextStyle
(),
cursorColor:
Colors
.
blue
,
backgroundCursorColor:
Colors
.
grey
,
cursorOpacityAnimates:
true
,
autofillHints:
null
,
spellCheckConfiguration:
const
SpellCheckConfiguration
(),
);
},
throwsAssertionError
,
);
});
testWidgets
(
'Spell check configured properly when spell check enabled without specified spell check service and native spell check service defined'
,
(
WidgetTester
tester
)
async
{
tester
.
binding
.
platformDispatcher
.
nativeSpellCheckServiceDefinedTestValue
=
true
;
await
tester
.
pumpWidget
(
MaterialApp
(
home:
EditableText
(
controller:
TextEditingController
(
text:
'A'
),
focusNode:
FocusNode
(),
style:
const
TextStyle
(),
cursorColor:
Colors
.
blue
,
backgroundCursorColor:
Colors
.
grey
,
cursorOpacityAnimates:
true
,
autofillHints:
null
,
spellCheckConfiguration:
const
SpellCheckConfiguration
(
misspelledTextStyle:
TextField
.
materialMisspelledTextStyle
,
),
),
),
);
final
EditableTextState
state
=
tester
.
state
<
EditableTextState
>(
find
.
byType
(
EditableText
));
expect
(
state
.
spellCheckEnabled
,
isTrue
);
expect
(
state
.
spellCheckConfiguration
.
spellCheckService
.
runtimeType
,
equals
(
DefaultSpellCheckService
),
);
tester
.
binding
.
platformDispatcher
.
clearNativeSpellCheckServiceDefined
();
});
testWidgets
(
'Spell check configured properly with specified spell check service'
,
(
WidgetTester
tester
)
async
{
final
FakeSpellCheckService
fakeSpellCheckService
=
FakeSpellCheckService
();
await
tester
.
pumpWidget
(
MaterialApp
(
home:
EditableText
(
controller:
TextEditingController
(
text:
'A'
),
focusNode:
FocusNode
(),
style:
const
TextStyle
(),
cursorColor:
Colors
.
blue
,
backgroundCursorColor:
Colors
.
grey
,
cursorOpacityAnimates:
true
,
autofillHints:
null
,
spellCheckConfiguration:
SpellCheckConfiguration
(
spellCheckService:
fakeSpellCheckService
,
misspelledTextStyle:
TextField
.
materialMisspelledTextStyle
,
),
),
),
);
final
EditableTextState
state
=
tester
.
state
<
EditableTextState
>(
find
.
byType
(
EditableText
));
expect
(
state
.
spellCheckConfiguration
.
spellCheckService
.
runtimeType
,
equals
(
FakeSpellCheckService
),
);
});
testWidgets
(
'Error thrown when spell check enabled but no default spell check service available'
,
(
WidgetTester
tester
)
async
{
tester
.
binding
.
platformDispatcher
.
nativeSpellCheckServiceDefinedTestValue
=
false
;
await
tester
.
pumpWidget
(
EditableText
(
controller:
TextEditingController
(
text:
'A'
),
focusNode:
FocusNode
(),
style:
const
TextStyle
(),
cursorColor:
Colors
.
blue
,
backgroundCursorColor:
Colors
.
grey
,
cursorOpacityAnimates:
true
,
autofillHints:
null
,
spellCheckConfiguration:
const
SpellCheckConfiguration
(
misspelledTextStyle:
TextField
.
materialMisspelledTextStyle
,
),
));
expect
(
tester
.
takeException
(),
isA
<
AssertionError
>());
tester
.
binding
.
platformDispatcher
.
clearNativeSpellCheckServiceDefined
();
});
});
group
(
'magnifier'
,
()
{
testWidgets
(
'should build nothing by default'
,
(
WidgetTester
tester
)
async
{
final
EditableText
editableText
=
EditableText
(
...
...
@@ -13032,7 +13190,7 @@ class _AccentColorTextEditingController extends TextEditingController {
_AccentColorTextEditingController
(
String
text
)
:
super
(
text:
text
);
@override
TextSpan
buildTextSpan
({
required
BuildContext
context
,
TextStyle
?
style
,
required
bool
withComposing
})
{
TextSpan
buildTextSpan
({
required
BuildContext
context
,
TextStyle
?
style
,
required
bool
withComposing
,
SpellCheckConfiguration
?
spellCheckConfiguration
})
{
final
Color
color
=
Theme
.
of
(
context
).
colorScheme
.
secondary
;
return
super
.
buildTextSpan
(
context:
context
,
style:
TextStyle
(
color:
color
),
withComposing:
withComposing
);
}
...
...
@@ -13041,3 +13199,5 @@ class _AccentColorTextEditingController extends TextEditingController {
class
_TestScrollController
extends
ScrollController
{
bool
get
attached
=>
hasListeners
;
}
class
FakeSpellCheckService
extends
DefaultSpellCheckService
{}
packages/flutter/test/widgets/spell_check_test.dart
0 → 100644
View file @
7db25c36
This diff is collapsed.
Click to expand it.
packages/flutter_test/lib/src/window.dart
View file @
7db25c36
...
...
@@ -310,6 +310,12 @@ class TestWindow implements ui.SingletonFlutterWindow {
platformDispatcher
.
onTextScaleFactorChanged
=
callback
;
}
@override
bool
get
nativeSpellCheckServiceDefined
=>
platformDispatcher
.
nativeSpellCheckServiceDefined
;
set
nativeSpellCheckServiceDefinedTestValue
(
bool
nativeSpellCheckServiceDefinedTestValue
)
{
// ignore: avoid_setters_without_getters
platformDispatcher
.
nativeSpellCheckServiceDefinedTestValue
=
nativeSpellCheckServiceDefinedTestValue
;
}
@override
bool
get
brieflyShowPassword
=>
platformDispatcher
.
brieflyShowPassword
;
/// Hides the real [brieflyShowPassword] and reports the given
...
...
@@ -721,6 +727,18 @@ class TestPlatformDispatcher implements ui.PlatformDispatcher {
_platformDispatcher
.
onTextScaleFactorChanged
=
callback
;
}
@override
bool
get
nativeSpellCheckServiceDefined
=>
_nativeSpellCheckServiceDefinedTestValue
??
_platformDispatcher
.
nativeSpellCheckServiceDefined
;
bool
?
_nativeSpellCheckServiceDefinedTestValue
;
set
nativeSpellCheckServiceDefinedTestValue
(
bool
nativeSpellCheckServiceDefinedTestValue
)
{
// ignore: avoid_setters_without_getters
_nativeSpellCheckServiceDefinedTestValue
=
nativeSpellCheckServiceDefinedTestValue
;
}
/// Deletes existing value that determines whether or not a native spell check
/// service is defined and returns to the real value.
void
clearNativeSpellCheckServiceDefined
()
{
_nativeSpellCheckServiceDefinedTestValue
=
null
;
}
@override
bool
get
brieflyShowPassword
=>
_brieflyShowPasswordTestValue
??
_platformDispatcher
.
brieflyShowPassword
;
bool
?
_brieflyShowPasswordTestValue
;
...
...
@@ -882,6 +900,7 @@ class TestPlatformDispatcher implements ui.PlatformDispatcher {
clearLocalesTestValue
();
clearSemanticsEnabledTestValue
();
clearTextScaleFactorTestValue
();
clearNativeSpellCheckServiceDefined
();
}
@override
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment