提交 b28fad3f authored 作者: Kevin's avatar Kevin

修复了多个自定义键盘无法切换的问题

上级 bbc3d000
## [0.2.1]
* TODO: 修复了多个自定义键盘无法切换的问题
## [0.2.0] ## [0.2.0]
* TODO: 优化了WeuiToast的效果,添加了WeuiToastConfig,可以全局配置默认设置 * TODO: 优化了WeuiToast的效果,添加了WeuiToastConfig,可以全局配置默认设置
* TODO: 添加了Keyboard对android 返回按钮的监听 * TODO: 添加了Keyboard对android 返回按钮的监听
......
...@@ -9,7 +9,7 @@ Usage Add this to your package's pubspec.yaml file: ...@@ -9,7 +9,7 @@ Usage Add this to your package's pubspec.yaml file:
Flutter >=1.7 Flutter >=1.7
``` yaml ``` yaml
dependencies: dependencies:
cool_ui: "^0.2.0" cool_ui: "^0.2.1"
``` ```
Flutter < 1.7 Flutter < 1.7
......
import 'package:cool_ui/cool_ui.dart';
import 'package:flutter/material.dart';
class TestKeyboard extends StatelessWidget{
static const CKTextInputType inputType = const CKTextInputType(name:'CKTestKeyboard');
static double getHeight(BuildContext ctx){
MediaQueryData mediaQuery = MediaQuery.of(ctx);
return mediaQuery.size.width / 3 / 2 * 4;
}
final KeyboardController controller ;
const TestKeyboard({this.controller});
static register(){
CoolKeyboard.addKeyboard(TestKeyboard.inputType,KeyboardConfig(builder: (context,controller){
return TestKeyboard(controller: controller);
},getHeight: TestKeyboard.getHeight));
}
@override
Widget build(BuildContext context) {
MediaQueryData mediaQuery = MediaQuery.of(context);
return Material(
child: DefaultTextStyle(style: TextStyle(fontWeight: FontWeight.w500,color: Colors.black,fontSize: 23.0), child: Container(
height:getHeight(context),
width: mediaQuery.size.width,
decoration: BoxDecoration(
color: Color(0xffafafaf),
),
child: GridView.count(
childAspectRatio: 2/1,
mainAxisSpacing:0.5,
crossAxisSpacing:0.5,
padding: EdgeInsets.all(0.0),
crossAxisCount: 3,
children: <Widget>[
buildButton('A'),
buildButton('B'),
buildButton('C'),
buildButton('D'),
buildButton('E'),
buildButton('F'),
buildButton('G'),
buildButton('H'),
buildButton('J'),
Container(
color: Color(0xFFd3d6dd),
child: GestureDetector(
behavior: HitTestBehavior.translucent,
child: Center(child: Icon(Icons.expand_more),),
onTap: (){
controller.doneAction();
},
),
),
buildButton('0'),
Container(
color: Color(0xFFd3d6dd),
child: GestureDetector(
behavior: HitTestBehavior.translucent,
child: Center(child: Text('X'),),
onTap: (){
controller.deleteOne();
},
),
),
]),
)),
);
}
Widget buildButton(String title,{String value}){
if(value == null){
value = title;
}
return Container(
color: Colors.white,
child: GestureDetector(
behavior: HitTestBehavior.translucent,
child: Center(child: Text(title),),
onTap: (){
controller.addText(value);
},
),
);
}
}
...@@ -6,8 +6,11 @@ import 'package:cool_ui_example/pages/weui_toast_demo.dart'; ...@@ -6,8 +6,11 @@ import 'package:cool_ui_example/pages/weui_toast_demo.dart';
import 'package:cool_ui/cool_ui.dart'; import 'package:cool_ui/cool_ui.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'keyboards/test_keyboard.dart';
void main(){ void main(){
NumberKeyboard.register(); NumberKeyboard.register();
TestKeyboard.register();
runApp(MyApp()); runApp(MyApp());
} }
......
import 'package:cool_ui_example/keyboards/test_keyboard.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:cool_ui/cool_ui.dart'; import 'package:cool_ui/cool_ui.dart';
...@@ -28,7 +29,7 @@ class CustomKeyboardDemoState extends State<CustomKeyboardDemo>{ ...@@ -28,7 +29,7 @@ class CustomKeyboardDemoState extends State<CustomKeyboardDemo>{
body: ListView( body: ListView(
children: <Widget>[ children: <Widget>[
TextField( TextField(
keyboardType: NumberKeyboard.inputType, keyboardType: TextInputType.text,
), ),
Container( Container(
height: 300.0, height: 300.0,
...@@ -36,6 +37,10 @@ class CustomKeyboardDemoState extends State<CustomKeyboardDemo>{ ...@@ -36,6 +37,10 @@ class CustomKeyboardDemoState extends State<CustomKeyboardDemo>{
TextField( TextField(
decoration: InputDecoration(labelText: '演示键盘弹出后滚动'), decoration: InputDecoration(labelText: '演示键盘弹出后滚动'),
keyboardType: NumberKeyboard.inputType, keyboardType: NumberKeyboard.inputType,
),
TextField(
decoration: InputDecoration(labelText: '多个键盘演示'),
keyboardType: TestKeyboard.inputType,
) )
], ],
) )
......
part of cool_ui; part of cool_ui;
typedef GetKeyboardHeight = double Function(BuildContext context); typedef GetKeyboardHeight = double Function(BuildContext context);
typedef KeyboardBuilder = Widget Function(BuildContext context,KeyboardController controller); typedef KeyboardBuilder = Widget Function(
BuildContext context, KeyboardController controller);
class CoolKeyboard { class CoolKeyboard {
static JSONMethodCodec _codec = const JSONMethodCodec(); static JSONMethodCodec _codec = const JSONMethodCodec();
static KeyboardConfig _currentKeyboard; static KeyboardConfig _currentKeyboard;
static Map<CKTextInputType,KeyboardConfig> _keyboards = {}; static Map<CKTextInputType, KeyboardConfig> _keyboards = {};
static BuildContext _context; static BuildContext _context;
static OverlayEntry _keyboardEntry; static OverlayEntry _keyboardEntry;
static KeyboardController _keyboardController; static KeyboardController _keyboardController;
static GlobalKey<KeyboardPageState> _pageKey; static GlobalKey<KeyboardPageState> _pageKey;
static bool isInterceptor = false; static bool isInterceptor = false;
static double get keyboardHeight =>_keyboardHeight; static double get keyboardHeight => _keyboardHeight;
static double _keyboardHeight; static double _keyboardHeight;
static init(BuildContext context) {
static init(BuildContext context){
_context = context; _context = context;
interceptorInput(); interceptorInput();
} }
static interceptorInput(){ static interceptorInput() {
if(isInterceptor) if (isInterceptor) return;
return;
isInterceptor = true; isInterceptor = true;
defaultBinaryMessenger.setMockMessageHandler("flutter/textinput", (ByteData data) async{ defaultBinaryMessenger.setMockMessageHandler("flutter/textinput",
(ByteData data) async {
var methodCall = _codec.decodeMethodCall(data); var methodCall = _codec.decodeMethodCall(data);
switch(methodCall.method){ switch (methodCall.method) {
case 'TextInput.show': case 'TextInput.show':
if(_currentKeyboard != null){ if (_currentKeyboard != null) {
openKeyboard(); openKeyboard();
return _codec.encodeSuccessEnvelope(null); return _codec.encodeSuccessEnvelope(null);
}else{ } else {
return await _sendPlatformMessage("flutter/textinput", data); return await _sendPlatformMessage("flutter/textinput", data);
} }
break; break;
case 'TextInput.hide': case 'TextInput.hide':
if(_currentKeyboard != null){ if (_currentKeyboard != null) {
hideKeyboard(); hideKeyboard();
return _codec.encodeSuccessEnvelope(null); return _codec.encodeSuccessEnvelope(null);
}else{ } else {
return await _sendPlatformMessage("flutter/textinput", data); return await _sendPlatformMessage("flutter/textinput", data);
} }
break; break;
case 'TextInput.setEditingState': case 'TextInput.setEditingState':
var editingState = TextEditingValue.fromJSON(methodCall.arguments); var editingState = TextEditingValue.fromJSON(methodCall.arguments);
if(editingState != null && _keyboardController != null){ if (editingState != null && _keyboardController != null) {
_keyboardController.value = editingState; _keyboardController.value = editingState;
return _codec.encodeSuccessEnvelope(null); return _codec.encodeSuccessEnvelope(null);
} }
break; break;
case 'TextInput.clearClient': case 'TextInput.clearClient':
hideKeyboard(animation:true); hideKeyboard(animation: true);
clearKeyboard(); clearKeyboard();
break; break;
case 'TextInput.setClient': case 'TextInput.setClient':
var setInputType = methodCall.arguments[1]['inputType']; var setInputType = methodCall.arguments[1]['inputType'];
InputClient client; InputClient client;
_keyboards.forEach((inputType,keyboardConfig){ _keyboards.forEach((inputType, keyboardConfig) {
if(inputType.name == setInputType['name']){ if (inputType.name == setInputType['name']) {
client = InputClient.fromJSON(methodCall.arguments); client = InputClient.fromJSON(methodCall.arguments);
clearKeyboard(); clearKeyboard();
_currentKeyboard = keyboardConfig; _currentKeyboard = keyboardConfig;
_keyboardController = KeyboardController(client:client)..addListener((){ _keyboardController = KeyboardController(client: client)
var callbackMethodCall = MethodCall("TextInputClient.updateEditingState",[ ..addListener(() {
_keyboardController.client.connectionId, var callbackMethodCall = MethodCall(
_keyboardController.value.toJSON()]); "TextInputClient.updateEditingState", [
defaultBinaryMessenger.handlePlatformMessage("flutter/textinput", _codec.encodeMethodCall(callbackMethodCall), (data){ _keyboardController.client.connectionId,
_keyboardController.value.toJSON()
]);
defaultBinaryMessenger.handlePlatformMessage(
"flutter/textinput",
_codec.encodeMethodCall(callbackMethodCall),
(data) {});
}); });
});
} }
}); });
if(client != null){ if (client != null) {
await _sendPlatformMessage("flutter/textinput", _codec.encodeMethodCall(MethodCall('TextInput.hide'))); await _sendPlatformMessage("flutter/textinput",
_codec.encodeMethodCall(MethodCall('TextInput.hide')));
return _codec.encodeSuccessEnvelope(null); return _codec.encodeSuccessEnvelope(null);
}else{ } else {
hideKeyboard(animation:false); hideKeyboard(animation: false);
clearKeyboard(); clearKeyboard();
} }
break; break;
} }
ByteData response = await _sendPlatformMessage("flutter/textinput", data); ByteData response = await _sendPlatformMessage("flutter/textinput", data);
return response; return response;
}); });
} }
static Future<ByteData> _sendPlatformMessage(String channel, ByteData message) { static Future<ByteData> _sendPlatformMessage(
String channel, ByteData message) {
final Completer<ByteData> completer = Completer<ByteData>(); final Completer<ByteData> completer = Completer<ByteData>();
ui.window.sendPlatformMessage(channel, message, (ByteData reply) { ui.window.sendPlatformMessage(channel, message, (ByteData reply) {
try { try {
...@@ -99,54 +104,55 @@ class CoolKeyboard { ...@@ -99,54 +104,55 @@ class CoolKeyboard {
exception: exception, exception: exception,
stack: stack, stack: stack,
library: 'services library', library: 'services library',
context: ErrorDescription('during a platform message response callback'), context:
ErrorDescription('during a platform message response callback'),
)); ));
} }
}); });
return completer.future; return completer.future;
} }
static addKeyboard(CKTextInputType inputType,KeyboardConfig config){ static addKeyboard(CKTextInputType inputType, KeyboardConfig config) {
_keyboards[inputType] = config; _keyboards[inputType] = config;
} }
static openKeyboard(){ static openKeyboard() {
if(_keyboardEntry != null) if (_keyboardEntry != null) return;
return;
_pageKey = GlobalKey<KeyboardPageState>(); _pageKey = GlobalKey<KeyboardPageState>();
_keyboardHeight = _currentKeyboard.getHeight(_context); _keyboardHeight = _currentKeyboard.getHeight(_context);
KeyboardMediaQueryState queryState = _context.ancestorStateOfType(const TypeMatcher<KeyboardMediaQueryState>()) as KeyboardMediaQueryState; KeyboardMediaQueryState queryState = _context
.ancestorStateOfType(const TypeMatcher<KeyboardMediaQueryState>())
as KeyboardMediaQueryState;
queryState.update(); queryState.update();
var tempKey = _pageKey; var tempKey = _pageKey;
_keyboardEntry = OverlayEntry(builder: (ctx) { _keyboardEntry = OverlayEntry(builder: (ctx) {
if(_currentKeyboard != null && _keyboardHeight != null) if (_currentKeyboard != null && _keyboardHeight != null) {
{ return KeyboardPage(
return KeyboardPage(
key: tempKey, key: tempKey,
child: Builder(builder: (ctx){ child: Builder(builder: (ctx) {
return _currentKeyboard.builder(ctx,_keyboardController); return _currentKeyboard.builder(ctx, _keyboardController);
}), }),
height:_keyboardHeight height: _keyboardHeight);
); } else {
}else{
return Container(); return Container();
} }
}); });
Overlay.of(_context).insert(_keyboardEntry); Overlay.of(_context).insert(_keyboardEntry);
BackButtonInterceptor.add((_){ BackButtonInterceptor.add((_) {
CoolKeyboard.sendPerformAction(TextInputAction.done); CoolKeyboard.sendPerformAction(TextInputAction.done);
return true; return true;
}, zIndex: 1, name:'CustomKeyboard'); }, zIndex: 1, name: 'CustomKeyboard');
} }
static hideKeyboard({bool animation=true}){ static hideKeyboard({bool animation = true}) {
BackButtonInterceptor.removeByName('CustomKeyboard'); BackButtonInterceptor.removeByName('CustomKeyboard');
if(_keyboardEntry != null && _pageKey != null) { if (_keyboardEntry != null && _pageKey != null) {
_keyboardHeight = null; _keyboardHeight = null;
_pageKey.currentState.animationController.addStatusListener((AnimationStatus status) { _pageKey.currentState.animationController
.addStatusListener((AnimationStatus status) {
if (status == AnimationStatus.dismissed || if (status == AnimationStatus.dismissed ||
status == AnimationStatus.completed) { status == AnimationStatus.completed) {
if (_keyboardEntry != null) { if (_keyboardEntry != null) {
...@@ -155,67 +161,63 @@ class CoolKeyboard { ...@@ -155,67 +161,63 @@ class CoolKeyboard {
} }
} }
}); });
if (animation) if (animation) {
{
_pageKey.currentState.exitKeyboard(); _pageKey.currentState.exitKeyboard();
} } else {
else{
_keyboardEntry.remove(); _keyboardEntry.remove();
_keyboardEntry = null; _keyboardEntry = null;
} }
} }
_pageKey = null; _pageKey = null;
KeyboardMediaQueryState queryState = _context.ancestorStateOfType(const TypeMatcher<KeyboardMediaQueryState>()) as KeyboardMediaQueryState; KeyboardMediaQueryState queryState = _context
.ancestorStateOfType(const TypeMatcher<KeyboardMediaQueryState>())
as KeyboardMediaQueryState;
queryState.update(); queryState.update();
} }
static clearKeyboard(){ static clearKeyboard() {
_currentKeyboard = null; _currentKeyboard = null;
if(_keyboardController != null){ if (_keyboardController != null) {
_keyboardController.dispose(); _keyboardController.dispose();
_keyboardController = null; _keyboardController = null;
} }
} }
static sendPerformAction(TextInputAction action){ static sendPerformAction(TextInputAction action) {
var callbackMethodCall = MethodCall("TextInputClient.performAction", var callbackMethodCall = MethodCall("TextInputClient.performAction",
[ [_keyboardController.client.connectionId, action.toString()]);
_keyboardController.client.connectionId, defaultBinaryMessenger.handlePlatformMessage("flutter/textinput",
action.toString() _codec.encodeMethodCall(callbackMethodCall), (data) {});
]);
defaultBinaryMessenger.handlePlatformMessage(
"flutter/textinput", _codec.encodeMethodCall(callbackMethodCall), (
data) {});
} }
} }
class KeyboardConfig{ class KeyboardConfig {
final KeyboardBuilder builder; final KeyboardBuilder builder;
final GetKeyboardHeight getHeight; final GetKeyboardHeight getHeight;
const KeyboardConfig({this.builder,this.getHeight}); const KeyboardConfig({this.builder, this.getHeight});
} }
class InputClient{ class InputClient {
final int connectionId; final int connectionId;
final TextInputConfiguration configuration; final TextInputConfiguration configuration;
const InputClient({this.connectionId,this.configuration}); const InputClient({this.connectionId, this.configuration});
factory InputClient.fromJSON(List<dynamic> encoded) { factory InputClient.fromJSON(List<dynamic> encoded) {
return InputClient(connectionId: encoded[0],configuration: TextInputConfiguration( return InputClient(
inputType:CKTextInputType.fromJSON(encoded[1]['inputType']), connectionId: encoded[0],
obscureText:encoded[1]['obscureText'], configuration: TextInputConfiguration(
autocorrect:encoded[1]['autocorrect'], inputType: CKTextInputType.fromJSON(encoded[1]['inputType']),
actionLabel:encoded[1]['actionLabel'], obscureText: encoded[1]['obscureText'],
inputAction:_toTextInputAction(encoded[1]['inputAction']), autocorrect: encoded[1]['autocorrect'],
textCapitalization:_toTextCapitalization(encoded[1]['textCapitalization']), actionLabel: encoded[1]['actionLabel'],
keyboardAppearance:_toBrightness(encoded[1]['keyboardAppearance']) inputAction: _toTextInputAction(encoded[1]['inputAction']),
textCapitalization:
)); _toTextCapitalization(encoded[1]['textCapitalization']),
keyboardAppearance:
_toBrightness(encoded[1]['keyboardAppearance'])));
} }
static TextInputAction _toTextInputAction(String action) { static TextInputAction _toTextInputAction(String action) {
switch (action) { switch (action) {
case 'TextInputAction.none': case 'TextInputAction.none':
...@@ -248,9 +250,8 @@ class InputClient{ ...@@ -248,9 +250,8 @@ class InputClient{
throw FlutterError('Unknown text input action: $action'); throw FlutterError('Unknown text input action: $action');
} }
static TextCapitalization _toTextCapitalization(String capitalization) {
static TextCapitalization _toTextCapitalization(String capitalization){ switch (capitalization) {
switch(capitalization){
case 'TextCapitalization.none': case 'TextCapitalization.none':
return TextCapitalization.none; return TextCapitalization.none;
case 'TextCapitalization.characters': case 'TextCapitalization.characters':
...@@ -264,8 +265,8 @@ class InputClient{ ...@@ -264,8 +265,8 @@ class InputClient{
throw FlutterError('Unknown text capitalization: $capitalization'); throw FlutterError('Unknown text capitalization: $capitalization');
} }
static Brightness _toBrightness(String brightness){ static Brightness _toBrightness(String brightness) {
switch(brightness){ switch (brightness) {
case 'Brightness.dark': case 'Brightness.dark':
return Brightness.dark; return Brightness.dark;
case 'Brightness.light': case 'Brightness.light':
...@@ -276,10 +277,11 @@ class InputClient{ ...@@ -276,10 +277,11 @@ class InputClient{
} }
} }
class CKTextInputType extends TextInputType{ class CKTextInputType extends TextInputType {
final String name; final String name;
const CKTextInputType({this.name,bool signed,bool decimal}) : super.numberWithOptions(signed:signed,decimal:decimal); const CKTextInputType({this.name, bool signed, bool decimal})
: super.numberWithOptions(signed: signed, decimal: decimal);
@override @override
Map<String, dynamic> toJson() { Map<String, dynamic> toJson() {
...@@ -290,26 +292,42 @@ class CKTextInputType extends TextInputType{ ...@@ -290,26 +292,42 @@ class CKTextInputType extends TextInputType{
}; };
} }
factory CKTextInputType.fromJSON(Map<String,dynamic> encoded) { @override
String toString() {
return '$runtimeType('
'name: $name, '
'signed: $signed, '
'decimal: $decimal)';
}
bool operator ==(Object target) {
if (target is CKTextInputType) {
if (this.toString() == target.toString()) {
return true;
}
}
return false;
}
factory CKTextInputType.fromJSON(Map<String, dynamic> encoded) {
return CKTextInputType( return CKTextInputType(
name: encoded['name'], name: encoded['name'],
signed: encoded['signed'], signed: encoded['signed'],
decimal: encoded['decimal'] decimal: encoded['decimal']);
);
} }
} }
class KeyboardPage extends StatefulWidget{ class KeyboardPage extends StatefulWidget {
final Widget child; final Widget child;
final double height; final double height;
const KeyboardPage({this.child,this.height,Key key}):super(key:key); const KeyboardPage({this.child, this.height, Key key}) : super(key: key);
@override @override
State<StatefulWidget> createState() =>KeyboardPageState(); State<StatefulWidget> createState() => KeyboardPageState();
} }
class KeyboardPageState extends State<KeyboardPage> with SingleTickerProviderStateMixin{ class KeyboardPageState extends State<KeyboardPage>
with SingleTickerProviderStateMixin {
AnimationController animationController; AnimationController animationController;
Animation<double> doubleAnimation; Animation<double> doubleAnimation;
double bottom; double bottom;
...@@ -318,37 +336,33 @@ class KeyboardPageState extends State<KeyboardPage> with SingleTickerProviderSta ...@@ -318,37 +336,33 @@ class KeyboardPageState extends State<KeyboardPage> with SingleTickerProviderSta
void initState() { void initState() {
// TODO: implement initState // TODO: implement initState
super.initState(); super.initState();
animationController = new AnimationController(duration: new Duration(milliseconds: 100),vsync: this) animationController = new AnimationController(
..addListener(()=>setState((){})); duration: new Duration(milliseconds: 100), vsync: this)
doubleAnimation = ..addListener(() => setState(() {}));
new Tween(begin: 0.0, end: widget.height).animate(animationController)..addListener(()=>setState((){})); doubleAnimation = new Tween(begin: 0.0, end: widget.height)
animationController.forward(from:0.0); .animate(animationController)
..addListener(() => setState(() {}));
animationController.forward(from: 0.0);
} }
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Positioned( return Positioned(
child: IntrinsicHeight( child: IntrinsicHeight(child: widget.child),
child: widget.child bottom: (widget.height - doubleAnimation.value) * -1);
),
bottom: (widget.height - doubleAnimation.value) * -1
);
} }
@override @override
void dispose() { void dispose() {
super.dispose(); if (animationController.status == AnimationStatus.forward ||
if(animationController.status == AnimationStatus.forward || animationController.status == AnimationStatus.reverse) animationController.status == AnimationStatus.reverse) {
{ animationController.notifyStatusListeners(AnimationStatus.dismissed);
animationController.notifyStatusListeners(AnimationStatus.dismissed); }
}
animationController.dispose(); animationController.dispose();
super.dispose();
} }
exitKeyboard(){ exitKeyboard() {
animationController.reverse(); animationController.reverse();
} }
} }
name: cool_ui name: cool_ui
description: Some practical Widget for flutter,Popover,Weui,Custom Keyboard description: Some practical Widget for flutter,Popover,Weui,Custom Keyboard
version: 0.2.0 version: 0.2.1
author: Kevin <liangkaikevin@gmail.com> author: Kevin <liangkaikevin@gmail.com>
homepage: https://github.com/Im-Kevin/cool_ui homepage: https://github.com/Im-Kevin/cool_ui
......
Markdown 格式
0%
您添加了 0 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论