parent
80ee3f4e30
commit
40e18b2706
11 changed files with 388 additions and 55 deletions
@ -0,0 +1,68 @@ |
|||||||
|
# Search |
||||||
|
|
||||||
|
You can search the text of your document using the search toolbar button. |
||||||
|
Enter the text and use the up/down buttons to move to the previous/next selection. |
||||||
|
Use the 3 vertical dots icon to turn on case-sensitivity or whole word constraints. |
||||||
|
|
||||||
|
## Search configuration options |
||||||
|
|
||||||
|
By default, the content of Embed objects are not searched. |
||||||
|
You can enable search by setting the [searchEmbedMode] in searchConfigurations: |
||||||
|
|
||||||
|
```dart |
||||||
|
MyQuillEditor( |
||||||
|
controller: _controller, |
||||||
|
configurations: QuillEditorConfigurations( |
||||||
|
searchConfigurations: const QuillSearchConfigurations( |
||||||
|
searchEmbedMode: SearchEmbedMode.plainText, |
||||||
|
), |
||||||
|
), |
||||||
|
... |
||||||
|
), |
||||||
|
``` |
||||||
|
|
||||||
|
### SearchEmbedMode.none (default option) |
||||||
|
|
||||||
|
Embed objects will not be included in searches. |
||||||
|
|
||||||
|
### SearchEmbedMode.rawData |
||||||
|
|
||||||
|
This is the simplest search option when your Embed objects use simple text that is also displayed to the user. |
||||||
|
This option allows searching within custom Embed objects using the node's raw data [Embeddable.data]. |
||||||
|
|
||||||
|
### SearchEmbedMode.plainText |
||||||
|
|
||||||
|
This option is best for complex Embeds where the raw data contains text that is not visible to the user and/or contains textual data that is not suitable for searching. |
||||||
|
For example, searching for '2024' would not be meaningful if the raw data is the full path of an image object (such as /user/temp/20240125/myimage.png). |
||||||
|
In this case the image would be shown as a search hit but the user would not know why. |
||||||
|
|
||||||
|
This option allows searching within custom Embed objects using an override to the [toPlainText] method. |
||||||
|
|
||||||
|
```dart |
||||||
|
class MyEmbedBuilder extends EmbedBuilder { |
||||||
|
|
||||||
|
@override |
||||||
|
String toPlainText(Embed node) { |
||||||
|
/// Convert [node] to the text that can be searched. |
||||||
|
/// For example: convert to MyEmbeddable and use the |
||||||
|
/// properties to return the searchable text. |
||||||
|
final m = MyEmbeddable(node.value.data); |
||||||
|
return '${m.property1}\t${m.property2}'; |
||||||
|
} |
||||||
|
... |
||||||
|
``` |
||||||
|
If [toPlainText] is not overridden, the base class implementation returns [Embed.kObjectReplacementCharacter] which is not searchable. |
||||||
|
|
||||||
|
### Strategy for mixed complex and simple Embed objects |
||||||
|
|
||||||
|
Select option [SearchEmbedMode.plainText] and override [toPlainText] to provide the searchable text. For your simple Embed objects provide the following override: |
||||||
|
|
||||||
|
```dart |
||||||
|
class MySimpleEmbedBuilder extends EmbedBuilder { |
||||||
|
|
||||||
|
@override |
||||||
|
String toPlainText(Embed node) { |
||||||
|
return node.value.data; |
||||||
|
} |
||||||
|
... |
||||||
|
``` |
@ -0,0 +1,39 @@ |
|||||||
|
import 'package:flutter/foundation.dart' show immutable; |
||||||
|
|
||||||
|
enum SearchEmbedMode { |
||||||
|
/// No search within Embed nodes. |
||||||
|
none, |
||||||
|
// Searches within Embed nodes using the nodes raw data [Embeddable.data.toString()] |
||||||
|
rawData, |
||||||
|
|
||||||
|
/// Searches within Embed nodes using override to [EmbedBuilder.toPlainText] |
||||||
|
plainText, |
||||||
|
} |
||||||
|
|
||||||
|
/// The configurations for the quill editor widget of flutter quill |
||||||
|
@immutable |
||||||
|
class QuillSearchConfigurations { |
||||||
|
const QuillSearchConfigurations({ |
||||||
|
this.searchEmbedMode = SearchEmbedMode.none, |
||||||
|
}); |
||||||
|
|
||||||
|
/// Search options for embed objects |
||||||
|
/// |
||||||
|
/// [SearchEmbedMode.none] disables searching within embed objects. |
||||||
|
/// [SearchEmbedMode.rawData] searches the Embed node using the raw data. |
||||||
|
/// [SearchEmbedMode.plainText] searches the Embed node using the [EmbedBuilder.toPlainText] override. |
||||||
|
final SearchEmbedMode searchEmbedMode; |
||||||
|
|
||||||
|
/// Future search options |
||||||
|
/// |
||||||
|
/// [rememberLastSearch] - would recall the last search text used. |
||||||
|
/// [enableSearchHistory] - would allow selection of previous searches. |
||||||
|
|
||||||
|
QuillSearchConfigurations copyWith({ |
||||||
|
SearchEmbedMode? searchEmbedMode, |
||||||
|
}) { |
||||||
|
return QuillSearchConfigurations( |
||||||
|
searchEmbedMode: searchEmbedMode ?? this.searchEmbedMode, |
||||||
|
); |
||||||
|
} |
||||||
|
} |
@ -1,5 +1,6 @@ |
|||||||
export '../controller/quill_controller_configurations.dart'; |
export '../controller/quill_controller_configurations.dart'; |
||||||
export '../editor/config/editor_configurations.dart'; |
export '../editor/config/editor_configurations.dart'; |
||||||
|
export '../editor/config/search_configurations.dart'; |
||||||
export '../editor_toolbar_shared/config/quill_shared_configurations.dart'; |
export '../editor_toolbar_shared/config/quill_shared_configurations.dart'; |
||||||
export '../toolbar/config/simple_toolbar_configurations.dart'; |
export '../toolbar/config/simple_toolbar_configurations.dart'; |
||||||
export '../toolbar/config/toolbar_configurations.dart'; |
export '../toolbar/config/toolbar_configurations.dart'; |
||||||
|
@ -0,0 +1,146 @@ |
|||||||
|
import 'package:flutter/widgets.dart'; |
||||||
|
import 'package:flutter_quill/flutter_quill.dart'; |
||||||
|
import 'package:flutter_quill/quill_delta.dart'; |
||||||
|
import 'package:test/test.dart'; |
||||||
|
|
||||||
|
class TestTimeStampEmbed extends Embeddable { |
||||||
|
const TestTimeStampEmbed( |
||||||
|
String value, |
||||||
|
) : super(timeStampType, value); |
||||||
|
|
||||||
|
static const String timeStampType = 'timeStamp'; |
||||||
|
} |
||||||
|
|
||||||
|
class TestTimeStampEmbedBuilderWidget extends EmbedBuilder { |
||||||
|
const TestTimeStampEmbedBuilderWidget(); |
||||||
|
|
||||||
|
@override |
||||||
|
String get key => 'timeStamp'; |
||||||
|
|
||||||
|
@override |
||||||
|
String toPlainText(Embed node) { |
||||||
|
return node.value.data.split(' ')[0]; // return date component |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build( |
||||||
|
BuildContext context, |
||||||
|
QuillController controller, |
||||||
|
Embed node, |
||||||
|
bool readOnly, |
||||||
|
bool inline, |
||||||
|
TextStyle textStyle, |
||||||
|
) { |
||||||
|
return Text(node.value.data); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
class TestUnknownEmbedBuilder extends EmbedBuilder { |
||||||
|
const TestUnknownEmbedBuilder(); |
||||||
|
|
||||||
|
@override |
||||||
|
String get key => 'unknown'; |
||||||
|
|
||||||
|
@override |
||||||
|
String toPlainText(Embed node) { |
||||||
|
return node.value.data.toString().substring(0, 5); |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build( |
||||||
|
BuildContext context, |
||||||
|
QuillController controller, |
||||||
|
Embed node, |
||||||
|
bool readOnly, |
||||||
|
bool inline, |
||||||
|
TextStyle textStyle, |
||||||
|
) { |
||||||
|
return Text(node.value.data); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void main() { |
||||||
|
group('search plain', () { |
||||||
|
test('search plain', () { |
||||||
|
final delta = Delta() |
||||||
|
..insert('Abc de\nfGhi') |
||||||
|
..insert('kl', {'bold': true}) |
||||||
|
..insert('demnoDe\n match whole word who\n'); |
||||||
|
final document = Document.fromDelta(delta); |
||||||
|
|
||||||
|
expect(document.search('de'), [4, 13, 18]); |
||||||
|
expect(document.search('lde'), [12]); |
||||||
|
expect(document.search('a'), [0, 23]); |
||||||
|
|
||||||
|
expect(document.search('de', caseSensitive: true), [4, 13]); |
||||||
|
expect(document.search('De', caseSensitive: true), [18]); |
||||||
|
|
||||||
|
expect(document.search('who'), [28, 39]); |
||||||
|
expect(document.search('whole', wholeWord: true), [28]); |
||||||
|
expect(document.search('who', wholeWord: true), [39]); |
||||||
|
}); |
||||||
|
|
||||||
|
test('search embed', () { |
||||||
|
final delta = Delta() |
||||||
|
..insert('Test ') |
||||||
|
..insert({ |
||||||
|
'image': 'https://unknown08.com/7900d52.png' |
||||||
|
}, { |
||||||
|
'width': '230', |
||||||
|
'style': {'display': 'block', 'margin': 'auto'} |
||||||
|
}) |
||||||
|
..insert('\n') |
||||||
|
..insert({'timeStamp': '2024-08-03 18:03:37.790068'}) |
||||||
|
..insert('\n'); |
||||||
|
final document = Document.fromDelta(delta); |
||||||
|
|
||||||
|
/// Default does not search embeds |
||||||
|
expect(document.search('2024'), [], reason: 'Does not search embeds'); |
||||||
|
|
||||||
|
/// Test rawData mode |
||||||
|
document.editorConfigurations = const QuillEditorConfigurations( |
||||||
|
searchConfigurations: QuillSearchConfigurations( |
||||||
|
searchEmbedMode: SearchEmbedMode.rawData)); |
||||||
|
expect(document.search('18'), [7], reason: 'raw data finds timeStamp'); |
||||||
|
expect(document.search('d52'), [5], reason: 'raw data finds image'); |
||||||
|
expect(document.search('08'), [5, 7], |
||||||
|
reason: 'raw data finds both embeds'); |
||||||
|
// |
||||||
|
document.editorConfigurations = const QuillEditorConfigurations( |
||||||
|
searchConfigurations: QuillSearchConfigurations( |
||||||
|
searchEmbedMode: SearchEmbedMode.plainText)); |
||||||
|
expect(document.search('2024'), [], reason: 'No embed builders'); |
||||||
|
|
||||||
|
/// Test plainText mode |
||||||
|
document.editorConfigurations = const QuillEditorConfigurations( |
||||||
|
searchConfigurations: QuillSearchConfigurations( |
||||||
|
searchEmbedMode: SearchEmbedMode.plainText), |
||||||
|
embedBuilders: [ |
||||||
|
TestTimeStampEmbedBuilderWidget(), |
||||||
|
], |
||||||
|
); |
||||||
|
expect(document.search('2024'), [7], |
||||||
|
reason: 'timeStamp embed builder overrides toPlainText'); |
||||||
|
expect(document.search('18'), [], |
||||||
|
reason: 'timeStamp overrides toPlainText returns date not time'); |
||||||
|
expect(document.search('08'), [7], |
||||||
|
reason: 'image does not override toPlainText'); |
||||||
|
|
||||||
|
/// Test unknownEmbedBuilder |
||||||
|
document.editorConfigurations = const QuillEditorConfigurations( |
||||||
|
searchConfigurations: QuillSearchConfigurations( |
||||||
|
searchEmbedMode: SearchEmbedMode.plainText), |
||||||
|
embedBuilders: [ |
||||||
|
TestTimeStampEmbedBuilderWidget(), |
||||||
|
], |
||||||
|
unknownEmbedBuilder: TestUnknownEmbedBuilder()); |
||||||
|
expect(document.search('7900'), [], |
||||||
|
reason: |
||||||
|
'image not found because unknown returns first 5 chars of rawData'); |
||||||
|
expect(document.search('https'), [5], |
||||||
|
reason: |
||||||
|
'image found because unknown returns first 5 chars of rawData'); |
||||||
|
expect(document.search('http'), [5]); |
||||||
|
}); |
||||||
|
}); |
||||||
|
} |
Loading…
Reference in new issue