Unverified Commit bdb74e16 authored by LongCatIsLooong's avatar LongCatIsLooong Committed by GitHub

Fix `Action.overridable` example (#110824)

parent 3c5a0747
...@@ -11,150 +11,73 @@ void main() { ...@@ -11,150 +11,73 @@ void main() {
runApp( runApp(
const MaterialApp( const MaterialApp(
home: Scaffold( home: Scaffold(
body: Center(child: SimpleUSPhoneNumberEntry()), body: Center(child: VerificationCodeGenerator()),
), ),
), ),
); );
} }
// This implements a custom phone number input field that handles the const CopyTextIntent copyTextIntent = CopyTextIntent._();
// [DeleteCharacterIntent] intent. class CopyTextIntent extends Intent {
class DigitInput extends StatefulWidget { const CopyTextIntent._();
const DigitInput({ }
super.key,
required this.controller,
required this.focusNode,
this.maxLength,
this.textInputAction = TextInputAction.next,
});
final int? maxLength; class CopyableText extends StatelessWidget {
final TextEditingController controller; const CopyableText({ super.key, required this.text });
final TextInputAction textInputAction;
final FocusNode focusNode;
@override final String text;
DigitInputState createState() => DigitInputState();
}
class DigitInputState extends State<DigitInput> { void _copy(CopyTextIntent intent) => Clipboard.setData(ClipboardData(text: text));
late final Action<DeleteCharacterIntent> _deleteTextAction =
CallbackAction<DeleteCharacterIntent>(
onInvoke: (DeleteCharacterIntent intent) {
// For simplicity we delete everything in the section.
widget.controller.clear();
return null;
},
);
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Actions( final Action<CopyTextIntent> defaultCopyAction = CallbackAction<CopyTextIntent>(onInvoke: _copy);
actions: <Type, Action<Intent>>{ return Shortcuts(
// Make the default `DeleteCharacterIntent` handler overridable. shortcuts: const <ShortcutActivator, Intent> { SingleActivator(LogicalKeyboardKey.keyC, control: true) : copyTextIntent },
DeleteCharacterIntent: Action<DeleteCharacterIntent>.overridable( child: Actions(
defaultAction: _deleteTextAction, context: context), actions: <Type, Action<Intent>> {
/// The Action is made overridable so the VerificationCodeGenerator
/// widget can override how copying is handled.
CopyTextIntent: Action<CopyTextIntent>.overridable(defaultAction: defaultCopyAction, context: context),
}, },
child: TextField( child: Focus(
controller: widget.controller, autofocus: true,
textInputAction: TextInputAction.next, child: DefaultTextStyle.merge(
keyboardType: TextInputType.phone, style: const TextStyle(fontSize: 20, fontWeight: FontWeight.bold),
focusNode: widget.focusNode, child: Text(text),
decoration: const InputDecoration( ),
border: OutlineInputBorder(),
), ),
inputFormatters: <TextInputFormatter>[
FilteringTextInputFormatter.digitsOnly,
LengthLimitingTextInputFormatter(widget.maxLength),
],
), ),
); );
} }
} }
class SimpleUSPhoneNumberEntry extends StatefulWidget { class VerificationCodeGenerator extends StatelessWidget {
const SimpleUSPhoneNumberEntry({super.key}); const VerificationCodeGenerator({ super.key });
@override void _copy(CopyTextIntent intent) {
State<SimpleUSPhoneNumberEntry> createState() => debugPrint('Content copied');
_SimpleUSPhoneNumberEntryState(); Clipboard.setData(const ClipboardData(text: '111222333'));
}
class _DeleteDigit extends Action<DeleteCharacterIntent> {
_DeleteDigit(this.state);
final _SimpleUSPhoneNumberEntryState state;
@override
void invoke(DeleteCharacterIntent intent) {
assert(callingAction != null);
callingAction?.invoke(intent);
if (state.lineNumberController.text.isEmpty &&
state.lineNumberFocusNode.hasFocus) {
state.prefixFocusNode.requestFocus();
} }
if (state.prefixController.text.isEmpty && state.prefixFocusNode.hasFocus) {
state.areaCodeFocusNode.requestFocus();
}
}
// This action is only enabled when the `callingAction` exists and is
// enabled.
@override
bool get isActionEnabled => callingAction?.isActionEnabled ?? false;
}
class _SimpleUSPhoneNumberEntryState extends State<SimpleUSPhoneNumberEntry> {
final FocusNode areaCodeFocusNode = FocusNode();
final TextEditingController areaCodeController = TextEditingController();
final FocusNode prefixFocusNode = FocusNode();
final TextEditingController prefixController = TextEditingController();
final FocusNode lineNumberFocusNode = FocusNode();
final TextEditingController lineNumberController = TextEditingController();
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Actions( return Actions(
actions: <Type, Action<Intent>>{ actions: <Type, Action<Intent>> { CopyTextIntent: CallbackAction<CopyTextIntent>(onInvoke: _copy) },
DeleteCharacterIntent: _DeleteDigit(this), child: Column(
}, mainAxisAlignment: MainAxisAlignment.center,
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: <Widget>[ children: <Widget>[
const Expanded( const Text('Press Ctrl-C to Copy'),
child: Text('(', textAlign: TextAlign.center), const SizedBox(height: 10),
), Row(
Expanded( mainAxisAlignment: MainAxisAlignment.center,
flex: 3, children: const <Widget>[
child: DigitInput( CopyableText(text: '111'),
focusNode: areaCodeFocusNode, SizedBox(width: 5,),
controller: areaCodeController, CopyableText(text: '222'),
maxLength: 3, SizedBox(width: 5,),
), CopyableText(text: '333'),
), ],
const Expanded(
child: Text(')', textAlign: TextAlign.center),
),
Expanded(
flex: 3,
child: DigitInput(
focusNode: prefixFocusNode,
controller: prefixController,
maxLength: 3,
),
),
const Expanded(
child: Text('-', textAlign: TextAlign.center),
),
Expanded(
flex: 4,
child: DigitInput(
focusNode: lineNumberFocusNode,
controller: lineNumberController,
textInputAction: TextInputAction.done,
maxLength: 4,
),
), ),
], ],
), ),
......
// 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/services.dart';
import 'package:flutter_api_samples/widgets/actions/action.action_overridable.0.dart' as example;
import 'package:flutter_test/flutter_test.dart';
void main() {
TestWidgetsFlutterBinding.ensureInitialized();
final _MockClipboard mockClipboard = _MockClipboard();
testWidgets('Copies text on Ctrl-C', (WidgetTester tester) async {
tester.binding.defaultBinaryMessenger.setMockMethodCallHandler(SystemChannels.platform, mockClipboard.handleMethodCall);
await tester.pumpWidget(const MaterialApp(
home: Scaffold(
body: Center(child: example.VerificationCodeGenerator()),
),
),
);
expect(primaryFocus, isNotNull);
expect(mockClipboard.clipboardData, isNull);
await tester.sendKeyDownEvent(LogicalKeyboardKey.control);
await tester.sendKeyDownEvent(LogicalKeyboardKey.keyC);
await tester.sendKeyUpEvent(LogicalKeyboardKey.control);
await tester.sendKeyUpEvent(LogicalKeyboardKey.keyC);
expect(mockClipboard.clipboardData?['text'], '111222333');
});
}
class _MockClipboard {
_MockClipboard();
Map<String, dynamic>? clipboardData;
Future<Object?> handleMethodCall(MethodCall methodCall) async {
switch (methodCall.method) {
case 'Clipboard.setData':
clipboardData = methodCall.arguments as Map<String, dynamic>;
return null;
}
if (methodCall.method.startsWith('Clipboard')) {
throw StateError('unrecognized method call: ${methodCall.method}');
}
return null;
}
}
...@@ -145,12 +145,16 @@ abstract class Action<T extends Intent> with Diagnosticable { ...@@ -145,12 +145,16 @@ abstract class Action<T extends Intent> with Diagnosticable {
/// parent widgets that also support this [Intent]. /// parent widgets that also support this [Intent].
/// ///
/// {@tool dartpad} /// {@tool dartpad}
/// This sample implements a custom text input field that handles the /// This sample shows how to implement a rudimentary `CopyableText` widget
/// [DeleteCharacterIntent] intent, as well as a US telephone number input /// that responds to Ctrl-C by copying its own content to the clipboard.
/// widget that consists of multiple text fields for area code, prefix and line ///
/// number. When the backspace key is pressed, the phone number input widget /// if `CopyableText` is to be provided in a package, developers using the
/// sends the focus to the preceding text field when the currently focused /// widget may want to change how copying is handled. As the author of the
/// field becomes empty. /// package, you can enable that by making the corresponding [Action]
/// overridable. In the second part of the code sample, three `CopyableText`
/// widgets are used to build a verification code widget which overrides the
/// "copy" action by copying the combined numbers from all three `CopyableText`
/// widgets.
/// ///
/// ** See code in examples/api/lib/widgets/actions/action.action_overridable.0.dart ** /// ** See code in examples/api/lib/widgets/actions/action.action_overridable.0.dart **
/// {@end-tool} /// {@end-tool}
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment