提交 94fbf371 authored 作者: Kevin's avatar Kevin

CupertinoPopoverDirection添加了left和right

上级 ecd2f383
## [0.6.1]
* TODO: CupertinoPopoverDirection添加了left和right
## [0.6.0] ## [0.6.0]
* TODO: 重构了键盘弹出的方式,修复长按输入框报错问题 * TODO: 重构了键盘弹出的方式,修复长按输入框报错问题
......
...@@ -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.17 Flutter >=1.17
``` yaml ``` yaml
dependencies: dependencies:
cool_ui: "^0.6.0" cool_ui: "^0.6.1"
``` ```
Flutter >=1.7 Flutter >=1.7
......
...@@ -2,6 +2,7 @@ import 'package:cool_ui_example/cool_u_i_example_icons.dart'; ...@@ -2,6 +2,7 @@ import 'package:cool_ui_example/cool_u_i_example_icons.dart';
import 'package:cool_ui_example/pages/custom_keyboard.dart'; import 'package:cool_ui_example/pages/custom_keyboard.dart';
import 'package:cool_ui_example/pages/paint_event_demo.dart'; import 'package:cool_ui_example/pages/paint_event_demo.dart';
import 'package:cool_ui_example/pages/popover_demo.dart'; import 'package:cool_ui_example/pages/popover_demo.dart';
import 'package:cool_ui_example/pages/table_demo.dart';
import 'package:cool_ui_example/pages/weui_toast_demo.dart'; 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';
...@@ -113,6 +114,13 @@ class _MyHomePageState extends State<MyHomePage> { ...@@ -113,6 +114,13 @@ class _MyHomePageState extends State<MyHomePage> {
Navigator.of(context).push(MaterialPageRoute( Navigator.of(context).push(MaterialPageRoute(
builder: (context) => CustomKeyboardDemo())); builder: (context) => CustomKeyboardDemo()));
}, },
),
ListTile(
title: Text("TableEvent"),
onTap: () {
Navigator.of(context).push(MaterialPageRoute(
builder: (context) => TableDemo()));
},
) )
], ],
)); ));
......
import 'package:flutter/material.dart';
import 'package:flutter/cupertino.dart';
import 'package:cool_ui/cool_ui.dart';
class TableDemo extends StatefulWidget{
@override
State<StatefulWidget> createState() {
// TODO: implement createState
return TableDemoState();
}
}
class TableDemoState extends State<TableDemo>{
@override
Widget build(BuildContext context) {
// TODO: implement build
return Scaffold(
appBar: AppBar(
title: Text("Table Demo"),
),
body: CoolTable()
);
}
}
\ No newline at end of file
...@@ -12,6 +12,7 @@ import 'package:flutter/rendering.dart'; ...@@ -12,6 +12,7 @@ import 'package:flutter/rendering.dart';
part 'utils/widget_util.dart'; part 'utils/widget_util.dart';
part 'utils/screen_util.dart'; part 'utils/screen_util.dart';
part 'utils/scroll_utils.dart';
part 'icons/cool_ui_icons.dart'; part 'icons/cool_ui_icons.dart';
...@@ -26,4 +27,7 @@ part 'keyboards/keyboard_manager.dart'; ...@@ -26,4 +27,7 @@ part 'keyboards/keyboard_manager.dart';
part 'keyboards/number_keyboard.dart'; part 'keyboards/number_keyboard.dart';
part 'keyboards/keyboard_controller.dart'; part 'keyboards/keyboard_controller.dart';
part 'keyboards/keyboard_media_query.dart'; part 'keyboards/keyboard_media_query.dart';
part 'keyboards/keyboard_root.dart'; part 'keyboards/keyboard_root.dart';
\ No newline at end of file
part 'widgets/tables/table.dart';
\ No newline at end of file
part of cool_ui;
connectScroll(ScrollController scroll1, ScrollController scroll2){
scroll1.addListener(() {
if(scroll2.offset != scroll1.offset){
scroll2.jumpTo(scroll1.offset);
}
});
scroll2.addListener(() {
if(scroll1.offset != scroll2.offset){
scroll1.jumpTo(scroll2.offset);
}
});
}
part of cool_ui; part of cool_ui;
enum CupertinoPopoverDirection{ enum CupertinoPopoverDirection { top, bottom, left, right }
top,
bottom,
}
typedef BoolCallback = bool Function(); typedef BoolCallback = bool Function();
class CupertinoPopoverButton extends StatelessWidget{
class CupertinoPopoverButton extends StatelessWidget {
final Widget child; final Widget child;
final WidgetBuilder popoverBuild; final WidgetBuilder popoverBuild;
final double popoverWidth; final double popoverWidth;
...@@ -20,34 +18,35 @@ class CupertinoPopoverButton extends StatelessWidget{ ...@@ -20,34 +18,35 @@ class CupertinoPopoverButton extends StatelessWidget{
final Color barrierColor; final Color barrierColor;
final CupertinoPopoverDirection direction; final CupertinoPopoverDirection direction;
CupertinoPopoverButton({ CupertinoPopoverButton(
@required this.child, {@required this.child,
this.popoverBuild, this.popoverBuild,
this.popoverColor=Colors.white, this.popoverColor = Colors.white,
this.popoverBoxShadow, this.popoverBoxShadow,
this.popoverWidth, this.popoverWidth,
this.popoverHeight, this.popoverHeight,
BoxConstraints popoverConstraints, BoxConstraints popoverConstraints,
this.direction = CupertinoPopoverDirection.bottom, this.direction = CupertinoPopoverDirection.bottom,
this.onTap, this.onTap,
this.transitionDuration=const Duration(milliseconds: 200), this.transitionDuration = const Duration(milliseconds: 200),
this.barrierColor = Colors.black54, this.barrierColor = Colors.black54,
this.radius=8.0}): this.radius = 8.0})
assert(popoverBuild != null), : assert(popoverBuild != null),
this.popoverConstraints = this.popoverConstraints =
(popoverWidth != null || popoverHeight != null) (popoverWidth != null || popoverHeight != null)
? popoverConstraints?.tighten(
? popoverConstraints?.tighten(width: popoverWidth, height: popoverHeight) width: popoverWidth, height: popoverHeight) ??
?? BoxConstraints.tightFor(width: popoverWidth, height: popoverHeight) BoxConstraints.tightFor(
: popoverConstraints; width: popoverWidth, height: popoverHeight)
: popoverConstraints;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
// TODO: implement build // TODO: implement build
return GestureDetector( return GestureDetector(
behavior: HitTestBehavior.translucent, behavior: HitTestBehavior.translucent,
onTap: (){ onTap: () {
if(onTap != null && onTap()){ if (onTap != null && onTap()) {
return; return;
} }
var offset = _WidgetUtil.getWidgetLocalToGlobal(context); var offset = _WidgetUtil.getWidgetLocalToGlobal(context);
...@@ -55,20 +54,20 @@ class CupertinoPopoverButton extends StatelessWidget{ ...@@ -55,20 +54,20 @@ class CupertinoPopoverButton extends StatelessWidget{
var body; var body;
showGeneralDialog( showGeneralDialog(
context: context, context: context,
pageBuilder: (BuildContext buildContext, Animation<double> animation, Animation<double> secondaryAnimation) { pageBuilder: (BuildContext buildContext, Animation<double> animation,
return Builder( Animation<double> secondaryAnimation) {
builder: (BuildContext context) { return Builder(builder: (BuildContext context) {
return Container(); return Container();
} });
);
}, },
barrierDismissible: true, barrierDismissible: true,
barrierLabel: MaterialLocalizations.of(context).modalBarrierDismissLabel, barrierLabel:
MaterialLocalizations.of(context).modalBarrierDismissLabel,
barrierColor: this.barrierColor, barrierColor: this.barrierColor,
transitionDuration: transitionDuration, transitionDuration: transitionDuration,
transitionBuilder: (BuildContext context, Animation<double> animation, Animation<double> secondaryAnimation, Widget child) { transitionBuilder: (BuildContext context, Animation<double> animation,
if(body == null){ Animation<double> secondaryAnimation, Widget child) {
if (body == null) {
body = popoverBuild(context); body = popoverBuild(context);
} }
return FadeTransition( return FadeTransition(
...@@ -77,9 +76,10 @@ class CupertinoPopoverButton extends StatelessWidget{ ...@@ -77,9 +76,10 @@ class CupertinoPopoverButton extends StatelessWidget{
curve: Curves.easeOut, curve: Curves.easeOut,
), ),
child: CupertinoPopover( child: CupertinoPopover(
attachRect:Rect.fromLTWH(offset.dx, offset.dy, bounds.width, bounds.height), attachRect: Rect.fromLTWH(
offset.dx, offset.dy, bounds.width, bounds.height),
child: body, child: body,
constraints:popoverConstraints, constraints: popoverConstraints,
color: popoverColor, color: popoverColor,
boxShadow: popoverBoxShadow, boxShadow: popoverBoxShadow,
context: context, context: context,
...@@ -88,7 +88,8 @@ class CupertinoPopoverButton extends StatelessWidget{ ...@@ -88,7 +88,8 @@ class CupertinoPopoverButton extends StatelessWidget{
direction: direction, direction: direction,
), ),
); );
},); },
);
}, },
child: child, child: child,
); );
...@@ -106,44 +107,51 @@ class CupertinoPopover extends StatefulWidget { ...@@ -106,44 +107,51 @@ class CupertinoPopover extends StatefulWidget {
final Animation<double> doubleAnimation; final Animation<double> doubleAnimation;
BoxConstraints constraints; BoxConstraints constraints;
CupertinoPopover({ CupertinoPopover(
@required this.attachRect, {@required this.attachRect,
@required this.child, @required this.child,
BoxConstraints constraints, BoxConstraints constraints,
this.color=Colors.white, this.color = Colors.white,
this.boxShadow, this.boxShadow,
@required BuildContext context, @required BuildContext context,
this.direction = CupertinoPopoverDirection.bottom, this.direction = CupertinoPopoverDirection.bottom,
this.doubleAnimation, this.doubleAnimation,
this.radius=8.0}):super(){ this.radius = 8.0})
: super() {
BoxConstraints temp; BoxConstraints temp;
if(constraints != null){ if (constraints != null) {
temp = BoxConstraints(maxHeight:123.0,maxWidth:150.0).copyWith( temp = BoxConstraints(maxHeight: 123.0, maxWidth: 150.0).copyWith(
minWidth: constraints.minWidth.isFinite?constraints.minWidth:null, minWidth: constraints.minWidth.isFinite ? constraints.minWidth : null,
minHeight: constraints.minHeight.isFinite?constraints.minHeight:null, minHeight:
maxWidth: constraints.maxWidth.isFinite?constraints.maxWidth:null, constraints.minHeight.isFinite ? constraints.minHeight : null,
maxHeight: constraints.maxHeight.isFinite?constraints.maxHeight:null, maxWidth: constraints.maxWidth.isFinite ? constraints.maxWidth : null,
maxHeight:
constraints.maxHeight.isFinite ? constraints.maxHeight : null,
); );
}else{ } else {
temp=BoxConstraints(maxHeight:123.0,maxWidth:150.0); temp = BoxConstraints(maxHeight: 123.0, maxWidth: 150.0);
} }
this.constraints = temp.copyWith(maxHeight: temp.maxHeight + CupertinoPopoverState._arrowHeight); this.constraints = temp.copyWith(
maxHeight: temp.maxHeight + CupertinoPopoverState._arrowHeight);
} }
@override @override
CupertinoPopoverState createState() => new CupertinoPopoverState(); CupertinoPopoverState createState() => new CupertinoPopoverState();
@override @override
void debugFillProperties(DiagnosticPropertiesBuilder properties) { void debugFillProperties(DiagnosticPropertiesBuilder properties) {
super.debugFillProperties(properties); super.debugFillProperties(properties);
properties.add(DiagnosticsProperty<BoxConstraints>('constraints', constraints, showName: false)); properties.add(DiagnosticsProperty<BoxConstraints>(
'constraints', constraints,
showName: false));
properties.add(DiagnosticsProperty<Color>('color', color, showName: false)); properties.add(DiagnosticsProperty<Color>('color', color, showName: false));
properties.add(DiagnosticsProperty<double>('double', radius, showName: false)); properties
.add(DiagnosticsProperty<double>('double', radius, showName: false));
} }
} }
class CupertinoPopoverState extends State<CupertinoPopover> with TickerProviderStateMixin{ class CupertinoPopoverState extends State<CupertinoPopover>
with TickerProviderStateMixin {
static const double _arrowWidth = 12.0; static const double _arrowWidth = 12.0;
static const double _arrowHeight = 8.0; static const double _arrowHeight = 8.0;
...@@ -173,33 +181,39 @@ class CupertinoPopoverState extends State<CupertinoPopover> with TickerProvider ...@@ -173,33 +181,39 @@ class CupertinoPopoverState extends State<CupertinoPopover> with TickerProvider
color: widget.color, color: widget.color,
boxShadow: widget.boxShadow, boxShadow: widget.boxShadow,
direction: widget.direction, direction: widget.direction,
child: Material(type: MaterialType.transparency, child: widget.child), child:
Material(type: MaterialType.transparency, child: widget.child),
), ),
) )
], ],
); );
} }
} }
class _CupertionPopoverPosition extends SingleChildRenderObjectWidget {
class _CupertionPopoverPosition extends SingleChildRenderObjectWidget{
final Rect attachRect; final Rect attachRect;
final Animation<double> scale; final Animation<double> scale;
final BoxConstraints constraints; final BoxConstraints constraints;
final CupertinoPopoverDirection direction; final CupertinoPopoverDirection direction;
_CupertionPopoverPosition({Widget child,this.attachRect,this.constraints,this.scale, this.direction}):super(child:child); _CupertionPopoverPosition(
{Widget child,
this.attachRect,
this.constraints,
this.scale,
this.direction})
: super(child: child);
@override @override
RenderObject createRenderObject(BuildContext context) =>_CupertionPopoverPositionRenderObject( RenderObject createRenderObject(BuildContext context) =>
attachRect:attachRect, _CupertionPopoverPositionRenderObject(
direction: direction, attachRect: attachRect,
constraints:constraints); direction: direction,
constraints: constraints);
@override @override
void updateRenderObject(BuildContext context, _CupertionPopoverPositionRenderObject renderObject) { void updateRenderObject(BuildContext context,
_CupertionPopoverPositionRenderObject renderObject) {
renderObject renderObject
..attachRect = attachRect ..attachRect = attachRect
..direction = direction ..direction = direction
...@@ -209,18 +223,17 @@ class _CupertionPopoverPosition extends SingleChildRenderObjectWidget{ ...@@ -209,18 +223,17 @@ class _CupertionPopoverPosition extends SingleChildRenderObjectWidget{
@override @override
void debugFillProperties(DiagnosticPropertiesBuilder properties) { void debugFillProperties(DiagnosticPropertiesBuilder properties) {
super.debugFillProperties(properties); super.debugFillProperties(properties);
properties.add(DiagnosticsProperty<BoxConstraints>('constraints', constraints, showName: false)); properties.add(DiagnosticsProperty<BoxConstraints>(
'constraints', constraints,
showName: false));
} }
} }
class _CupertionPopoverPositionRenderObject extends RenderShiftedBox{ class _CupertionPopoverPositionRenderObject extends RenderShiftedBox {
CupertinoPopoverDirection get direction => _direction; CupertinoPopoverDirection get direction => _direction;
CupertinoPopoverDirection _direction; CupertinoPopoverDirection _direction;
set direction(CupertinoPopoverDirection value) { set direction(CupertinoPopoverDirection value) {
if (_direction == value) if (_direction == value) return;
return;
_direction = value; _direction = value;
markNeedsLayout(); markNeedsLayout();
} }
...@@ -228,92 +241,133 @@ class _CupertionPopoverPositionRenderObject extends RenderShiftedBox{ ...@@ -228,92 +241,133 @@ class _CupertionPopoverPositionRenderObject extends RenderShiftedBox{
Rect get attachRect => _attachRect; Rect get attachRect => _attachRect;
Rect _attachRect; Rect _attachRect;
set attachRect(Rect value) { set attachRect(Rect value) {
if (_attachRect == value) if (_attachRect == value) return;
return;
_attachRect = value; _attachRect = value;
markNeedsLayout(); markNeedsLayout();
} }
BoxConstraints get additionalConstraints => _additionalConstraints; BoxConstraints get additionalConstraints => _additionalConstraints;
BoxConstraints _additionalConstraints; BoxConstraints _additionalConstraints;
set additionalConstraints(BoxConstraints value) { set additionalConstraints(BoxConstraints value) {
if (_additionalConstraints == value) if (_additionalConstraints == value) return;
return;
_additionalConstraints = value; _additionalConstraints = value;
markNeedsLayout(); markNeedsLayout();
} }
_CupertionPopoverPositionRenderObject(
_CupertionPopoverPositionRenderObject({RenderBox child,Rect attachRect,Color color,BoxConstraints constraints,Animation<double> scale, CupertinoPopoverDirection direction}) : super(child){ {RenderBox child,
Rect attachRect,
Color color,
BoxConstraints constraints,
Animation<double> scale,
CupertinoPopoverDirection direction})
: super(child) {
this._attachRect = attachRect; this._attachRect = attachRect;
this._additionalConstraints = constraints; this._additionalConstraints = constraints;
this._direction = direction; this._direction = direction;
} }
@override @override
void performLayout() { void performLayout() {
child.layout(_additionalConstraints.enforce(constraints), parentUsesSize: true); child.layout(_additionalConstraints.enforce(constraints),
size = Size(constraints.maxWidth,constraints.maxHeight); parentUsesSize: true);
size = Size(constraints.maxWidth, constraints.maxHeight);
final BoxParentData childParentData = child.parentData; final BoxParentData childParentData = child.parentData;
childParentData.offset = calcOffset(child.size); childParentData.offset = calcOffset(child.size);
} }
Offset calcOffset(Size size){ Offset calcOffset(Size size) {
double bodyLeft = 0.0; CupertinoPopoverDirection calcDirection =
CupertinoPopoverDirection calcDirection = _calcDirection(attachRect,size, direction); _calcDirection(attachRect, size, direction);
if (calcDirection == CupertinoPopoverDirection.top ||
if(attachRect.left > size.width / 2 && calcDirection == CupertinoPopoverDirection.bottom) {
_ScreenUtil.getInstance().screenWidth - attachRect.right > size.width / 2){ //判断是否可以在中间 double bodyLeft = 0.0;
bodyLeft = attachRect.left + attachRect.width / 2 - size.width / 2; // 上下
}else if(attachRect.left < size.width / 2){ //靠左 if (attachRect.left > size.width / 2 &&
bodyLeft = 10.0; _ScreenUtil.getInstance().screenWidth - attachRect.right >
}else{ //靠右 size.width / 2) {
bodyLeft = _ScreenUtil.getInstance().screenWidth - 10.0 - size.width; //判断是否可以在中间
} bodyLeft = attachRect.left + attachRect.width / 2 - size.width / 2;
} else if (attachRect.left < size.width / 2) {
if(calcDirection == CupertinoPopoverDirection.bottom){ //靠左
return Offset(bodyLeft,attachRect.bottom); bodyLeft = 10.0;
}else{ } else {
return Offset(bodyLeft,attachRect.top - size.height - CupertinoPopoverState._arrowHeight); //靠右
bodyLeft = _ScreenUtil.getInstance().screenWidth - 10.0 - size.width;
}
if (calcDirection == CupertinoPopoverDirection.bottom) {
return Offset(bodyLeft, attachRect.bottom);
} else {
return Offset(bodyLeft,
attachRect.top - size.height);
}
} else {
double bodyTop = 0.0;
if (attachRect.top > size.height / 2 &&
_ScreenUtil.getInstance().screenHeight - attachRect.bottom >
size.height / 2) {
//判断是否可以在中间
bodyTop = attachRect.top + attachRect.height / 2 - size.height / 2;
} else if (attachRect.top < size.height / 2) {
//靠左
bodyTop = 10.0;
} else {
//靠右
bodyTop = _ScreenUtil.getInstance().screenHeight - 10.0 - size.height;
}
if (calcDirection == CupertinoPopoverDirection.right) {
return Offset(attachRect.right, bodyTop);
} else {
return Offset(
attachRect.left - size.width,
bodyTop);
}
} }
} }
@override @override
void debugFillProperties(DiagnosticPropertiesBuilder properties) { void debugFillProperties(DiagnosticPropertiesBuilder properties) {
super.debugFillProperties(properties); super.debugFillProperties(properties);
properties.add(DiagnosticsProperty<BoxConstraints>('additionalConstraints', additionalConstraints)); properties.add(DiagnosticsProperty<BoxConstraints>(
'additionalConstraints', additionalConstraints));
} }
} }
class _CupertionPopoverContext extends SingleChildRenderObjectWidget{ class _CupertionPopoverContext extends SingleChildRenderObjectWidget {
final Rect attachRect; final Rect attachRect;
final Color color; final Color color;
final List<BoxShadow> boxShadow; final List<BoxShadow> boxShadow;
final Animation<double> scale; final Animation<double> scale;
final double radius; final double radius;
final CupertinoPopoverDirection direction; final CupertinoPopoverDirection direction;
_CupertionPopoverContext({Widget child,this.attachRect,this.color, this.boxShadow,this.scale,this.radius, this.direction}):super(child:child); _CupertionPopoverContext(
{Widget child,
this.attachRect,
this.color,
this.boxShadow,
this.scale,
this.radius,
this.direction})
: super(child: child);
@override @override
RenderObject createRenderObject(BuildContext context) => _CupertionPopoverContextRenderObject( RenderObject createRenderObject(BuildContext context) =>
attachRect: attachRect, _CupertionPopoverContextRenderObject(
color: color, attachRect: attachRect,
boxShadow: boxShadow, color: color,
scale: scale.value, boxShadow: boxShadow,
direction: direction, scale: scale.value,
radius: radius direction: direction,
); radius: radius);
@override @override
void updateRenderObject(BuildContext context, _CupertionPopoverContextRenderObject renderObject) { void updateRenderObject(
BuildContext context, _CupertionPopoverContextRenderObject renderObject) {
renderObject renderObject
..attachRect = attachRect ..attachRect = attachRect
..color = color ..color = color
...@@ -322,35 +376,29 @@ class _CupertionPopoverContext extends SingleChildRenderObjectWidget{ ...@@ -322,35 +376,29 @@ class _CupertionPopoverContext extends SingleChildRenderObjectWidget{
..direction = direction ..direction = direction
..radius = radius; ..radius = radius;
} }
} }
class _CupertionPopoverContextRenderObject extends RenderShiftedBox{ class _CupertionPopoverContextRenderObject extends RenderShiftedBox {
CupertinoPopoverDirection get direction => _direction; CupertinoPopoverDirection get direction => _direction;
CupertinoPopoverDirection _direction; CupertinoPopoverDirection _direction;
set direction(CupertinoPopoverDirection value) { set direction(CupertinoPopoverDirection value) {
if (_direction == value) if (_direction == value) return;
return;
_direction = value; _direction = value;
markNeedsLayout(); markNeedsLayout();
} }
Rect get attachRect => _attachRect; Rect get attachRect => _attachRect;
Rect _attachRect; Rect _attachRect;
set attachRect(Rect value) { set attachRect(Rect value) {
if (_attachRect == value) if (_attachRect == value) return;
return;
_attachRect = value; _attachRect = value;
markNeedsLayout(); markNeedsLayout();
} }
Color get color => _color; Color get color => _color;
Color _color; Color _color;
set color(Color value) { set color(Color value) {
if (_color == value) if (_color == value) return;
return;
_color = value; _color = value;
markNeedsLayout(); markNeedsLayout();
} }
...@@ -358,13 +406,11 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{ ...@@ -358,13 +406,11 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{
List<BoxShadow> get boxShadow => _boxShadow; List<BoxShadow> get boxShadow => _boxShadow;
List<BoxShadow> _boxShadow; List<BoxShadow> _boxShadow;
set boxShadow(List<BoxShadow> value) { set boxShadow(List<BoxShadow> value) {
if (_boxShadow == value) if (_boxShadow == value) return;
return;
_boxShadow = value; _boxShadow = value;
markNeedsLayout(); markNeedsLayout();
} }
double get scale => _scale; double get scale => _scale;
double _scale; double _scale;
set scale(double value) { set scale(double value) {
...@@ -375,18 +421,23 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{ ...@@ -375,18 +421,23 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{
markNeedsLayout(); markNeedsLayout();
} }
double get radius => _radius; double get radius => _radius;
double _radius; double _radius;
set radius(double value) { set radius(double value) {
if (_radius == value) if (_radius == value) return;
return;
_radius = value; _radius = value;
markNeedsLayout(); markNeedsLayout();
} }
_CupertionPopoverContextRenderObject(
_CupertionPopoverContextRenderObject({RenderBox child,Rect attachRect,Color color, List<BoxShadow> boxShadow,double scale,double radius, CupertinoPopoverDirection direction}) : super(child){ {RenderBox child,
Rect attachRect,
Color color,
List<BoxShadow> boxShadow,
double scale,
double radius,
CupertinoPopoverDirection direction})
: super(child) {
this._attachRect = attachRect; this._attachRect = attachRect;
this._color = color; this._color = color;
this._boxShadow = boxShadow; this._boxShadow = boxShadow;
...@@ -395,19 +446,42 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{ ...@@ -395,19 +446,42 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{
this._direction = direction; this._direction = direction;
} }
@override @override
void performLayout() { void performLayout() {
assert(constraints.maxHeight.isFinite); assert(constraints.maxHeight.isFinite);
BoxConstraints childConstraints = BoxConstraints(maxHeight: constraints.maxHeight - CupertinoPopoverState._arrowHeight).enforce(constraints); BoxConstraints childConstraints;
if (direction == CupertinoPopoverDirection.top ||
direction == CupertinoPopoverDirection.bottom) {
childConstraints = BoxConstraints(
maxHeight:
constraints.maxHeight - CupertinoPopoverState._arrowHeight)
.enforce(constraints);
} else {
childConstraints = BoxConstraints(
maxWidth:
constraints.maxWidth - CupertinoPopoverState._arrowHeight)
.enforce(constraints);
}
child.layout(childConstraints, parentUsesSize: true); child.layout(childConstraints, parentUsesSize: true);
size = Size(child.size.width,child.size.height + CupertinoPopoverState._arrowHeight);
if (direction == CupertinoPopoverDirection.top ||
direction == CupertinoPopoverDirection.bottom) {
size = Size(child.size.width,
child.size.height + CupertinoPopoverState._arrowHeight);
} else {
size = Size(child.size.width + CupertinoPopoverState._arrowHeight,
child.size.height);
}
CupertinoPopoverDirection calcDirection =
_calcDirection(attachRect, size, direction);
final BoxParentData childParentData = child.parentData; final BoxParentData childParentData = child.parentData;
CupertinoPopoverDirection calcDirection = _calcDirection(attachRect,size, direction); if (calcDirection == CupertinoPopoverDirection.bottom) {
if(calcDirection == CupertinoPopoverDirection.bottom)
{
childParentData.offset = Offset(0.0, CupertinoPopoverState._arrowHeight); childParentData.offset = Offset(0.0, CupertinoPopoverState._arrowHeight);
} else if (calcDirection == CupertinoPopoverDirection.right) {
childParentData.offset = Offset(CupertinoPopoverState._arrowHeight, 0.0);
} }
} }
...@@ -417,120 +491,223 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{ ...@@ -417,120 +491,223 @@ class _CupertionPopoverContextRenderObject extends RenderShiftedBox{
Matrix4 transform = Matrix4.identity(); Matrix4 transform = Matrix4.identity();
// //
CupertinoPopoverDirection calcDirection = _calcDirection(attachRect,size, direction); CupertinoPopoverDirection calcDirection =
var isArrowUp = calcDirection == CupertinoPopoverDirection.bottom; _calcDirection(attachRect, size, direction);
// var isArrowUp = calcDirection == CupertinoPopoverDirection.bottom;
Rect arrowRect;
Offset translation;
Rect bodyRect;
final BoxParentData childParentData = child.parentData;
bodyRect = childParentData.offset & child.size;
var arrowLeft = attachRect.left + // 用于 Top和Bottom
attachRect.width / 2 -
CupertinoPopoverState._arrowWidth / 2 -
offset.dx;
var arrowTop = attachRect.top + // 用于 Left和Right
attachRect.height / 2 -
CupertinoPopoverState._arrowWidth / 2 -
offset.dy;
switch (calcDirection) {
case CupertinoPopoverDirection.top:
arrowRect = Rect.fromLTWH(
arrowLeft,
child.size.height,
CupertinoPopoverState._arrowWidth,
CupertinoPopoverState._arrowHeight);
translation = Offset(
arrowLeft + CupertinoPopoverState._arrowWidth / 2, size.height);
break;
case CupertinoPopoverDirection.left:
arrowRect = Rect.fromLTWH(
child.size.width,
arrowTop,
CupertinoPopoverState._arrowHeight,
CupertinoPopoverState._arrowWidth);
translation = Offset(
size.width, arrowTop + CupertinoPopoverState._arrowWidth / 2);
break;
case CupertinoPopoverDirection.bottom:
arrowRect = Rect.fromLTWH(
arrowLeft,
0,
CupertinoPopoverState._arrowWidth,
CupertinoPopoverState._arrowHeight);
translation =
Offset(arrowLeft + CupertinoPopoverState._arrowWidth / 2, 0);
break;
case CupertinoPopoverDirection.right:
arrowRect = Rect.fromLTWH(
0,
arrowTop,
CupertinoPopoverState._arrowHeight,
CupertinoPopoverState._arrowWidth);
translation = Offset(
0, arrowTop + CupertinoPopoverState._arrowWidth / 2);
break;
default:
}
var arrowLeft =attachRect.left + attachRect.width / 2 - CupertinoPopoverState._arrowWidth / 2 - offset.dx;
var translation = Offset(arrowLeft + CupertinoPopoverState._arrowWidth / 2,isArrowUp?0.0:size.height);
transform.translate(translation.dx, translation.dy); transform.translate(translation.dx, translation.dy);
transform.scale(scale, scale, 1.0); transform.scale(scale, scale, 1.0);
transform.translate(-translation.dx, -translation.dy); transform.translate(-translation.dx, -translation.dy);
Rect arrowRect = Rect.fromLTWH(
arrowLeft,
isArrowUp?0.0:child.size.height,
CupertinoPopoverState._arrowWidth,
CupertinoPopoverState._arrowHeight);
Rect bodyRect = Offset(0.0, isArrowUp?CupertinoPopoverState._arrowHeight:0.0) & child.size;
_paintShadows(context, transform, offset, isArrowUp,arrowRect,bodyRect);
Path clipPath = _getClip(isArrowUp,arrowRect,bodyRect);
context.pushClipPath(needsCompositing,
offset,offset & size,
clipPath,(context,offset){
context.pushTransform(needsCompositing, offset, transform,(context,offset){
final Paint backgroundPaint = Paint();
backgroundPaint.color = color;
context.canvas.drawRect(offset & size, backgroundPaint);
super.paint(context,offset);
});
});
}
_paintShadows(
context, transform, offset, calcDirection, arrowRect, bodyRect);
Path clipPath = _getClip(calcDirection, arrowRect, bodyRect);
context.pushClipPath(needsCompositing, offset, offset & size, clipPath,
(context, offset) {
context.pushTransform(needsCompositing, offset, transform,
(context, offset) {
final Paint backgroundPaint = Paint();
backgroundPaint.color = color;
context.canvas.drawRect(offset & size, backgroundPaint);
super.paint(context, offset);
});
});
}
void _paintShadows(PaintingContext context, Matrix4 transform, Offset offset,
void _paintShadows(PaintingContext context, Matrix4 transform, Offset offset, bool isArrowUp,Rect arrowRect,Rect bodyRect) { CupertinoPopoverDirection direction, Rect arrowRect, Rect bodyRect) {
if (boxShadow == null) if (boxShadow == null) return;
return;
for (final BoxShadow boxShadow in boxShadow) { for (final BoxShadow boxShadow in boxShadow) {
final Paint paint = boxShadow.toPaint(); final Paint paint = boxShadow.toPaint();
arrowRect = arrowRect.shift(offset).shift(boxShadow.offset).inflate(boxShadow.spreadRadius); arrowRect = arrowRect
bodyRect = bodyRect.shift(offset).shift(boxShadow.offset).inflate(boxShadow.spreadRadius); .shift(offset)
Path path = _getClip(isArrowUp, arrowRect, bodyRect); .shift(boxShadow.offset)
.inflate(boxShadow.spreadRadius);
context.pushTransform(needsCompositing, offset, transform,(context,offset){ bodyRect = bodyRect
context.canvas.drawPath(path,paint); .shift(offset)
.shift(boxShadow.offset)
.inflate(boxShadow.spreadRadius);
Path path = _getClip(direction, arrowRect, bodyRect);
context.pushTransform(needsCompositing, offset, transform,
(context, offset) {
context.canvas.drawPath(path, paint);
}); });
} }
} }
Path _getClip(bool isArrowUp,Rect arrowRect,Rect bodyRect) {
Path path = new Path();
if(isArrowUp) Path _getClip(
{ CupertinoPopoverDirection direction, Rect arrowRect, Rect bodyRect) {
Path path = new Path();
path.moveTo(arrowRect.left,arrowRect.bottom); //箭头 if (direction == CupertinoPopoverDirection.top) {
path.lineTo(arrowRect.left + arrowRect.width / 2, arrowRect.top); path.moveTo(arrowRect.left, arrowRect.top); //箭头
path.lineTo(arrowRect.left + arrowRect.width / 2, arrowRect.bottom);
path.lineTo(arrowRect.right, arrowRect.top);
path.lineTo(bodyRect.right - radius, bodyRect.bottom); //右下角
path.conicTo(bodyRect.right, bodyRect.bottom, bodyRect.right,
bodyRect.bottom - radius, 1.0);
path.lineTo(bodyRect.right, bodyRect.top + radius); //右上角
path.conicTo(bodyRect.right, bodyRect.top, bodyRect.right - radius,
bodyRect.top, 1.0);
path.lineTo(bodyRect.left + radius, bodyRect.top); //左上角
path.conicTo(bodyRect.left, bodyRect.top, bodyRect.left,
bodyRect.top + radius, 1.0);
path.lineTo(bodyRect.left, bodyRect.bottom - radius); //左下角
path.conicTo(bodyRect.left, bodyRect.bottom, bodyRect.left + radius,
bodyRect.bottom, 1.0);
} else if (direction == CupertinoPopoverDirection.right) {
path.moveTo(arrowRect.right, arrowRect.top); //箭头
path.lineTo(arrowRect.left, arrowRect.top + arrowRect.height / 2);
path.lineTo(arrowRect.right, arrowRect.bottom); path.lineTo(arrowRect.right, arrowRect.bottom);
path.lineTo(bodyRect.right - radius,bodyRect.top); //右上角 path.lineTo(bodyRect.left, bodyRect.bottom - radius); //左下角
path.conicTo(bodyRect.right,bodyRect.top path.conicTo(bodyRect.left, bodyRect.bottom, bodyRect.left + radius,
,bodyRect.right,bodyRect.top + radius,1.0); bodyRect.bottom, 1.0);
path.lineTo(bodyRect.right - radius, bodyRect.bottom); //右下角
path.conicTo(bodyRect.right, bodyRect.bottom, bodyRect.right,
bodyRect.bottom - radius, 1.0);
path.lineTo(bodyRect.right,bodyRect.bottom - radius); //右下 path.lineTo(bodyRect.right, bodyRect.top + radius); //右上
path.conicTo(bodyRect.right,bodyRect.bottom path.conicTo(bodyRect.right, bodyRect.top, bodyRect.right - radius,
,bodyRect.right -radius ,bodyRect.bottom,1.0); bodyRect.top, 1.0);
path.lineTo(bodyRect.left + radius, bodyRect.top); //左上角
path.conicTo(bodyRect.left, bodyRect.top, bodyRect.left,
bodyRect.top + radius, 1.0);
} else if (direction == CupertinoPopoverDirection.left) {
path.moveTo(arrowRect.left, arrowRect.top); //箭头
path.lineTo(arrowRect.right, arrowRect.top + arrowRect.height / 2);
path.lineTo(arrowRect.left, arrowRect.bottom);
path.lineTo(bodyRect.right, bodyRect.bottom - radius); //右下角
path.conicTo(bodyRect.right, bodyRect.bottom, bodyRect.right - radius,
bodyRect.bottom, 1.0);
path.lineTo(bodyRect.left + radius, bodyRect.bottom); //左下角 path.lineTo(bodyRect.left + radius, bodyRect.bottom); //左下角
path.conicTo(bodyRect.left,bodyRect.bottom path.conicTo(bodyRect.left, bodyRect.bottom, bodyRect.left,
,bodyRect.left ,bodyRect.bottom - radius,1.0); bodyRect.bottom - radius, 1.0);
path.lineTo(bodyRect.left, bodyRect.top + radius); //左上角 path.lineTo(bodyRect.left, bodyRect.top + radius); //左上角
path.conicTo(bodyRect.left,bodyRect.top path.conicTo(bodyRect.left, bodyRect.top, bodyRect.left + radius,
,bodyRect.left + radius,bodyRect.top,1.0); bodyRect.top, 1.0);
}else{
path.lineTo(bodyRect.right - radius, bodyRect.top); //右上角
path.moveTo(bodyRect.left + radius,bodyRect.top); path.conicTo(bodyRect.right, bodyRect.top, bodyRect.right,
bodyRect.top + radius, 1.0);
} else {
path.moveTo(arrowRect.left, arrowRect.bottom); //箭头
path.lineTo(arrowRect.left + arrowRect.width / 2, arrowRect.top);
path.lineTo(arrowRect.right, arrowRect.bottom);
path.lineTo(bodyRect.right - radius,bodyRect.top); //右上角 path.lineTo(bodyRect.right - radius, bodyRect.top); //右上角
path.conicTo(bodyRect.right,bodyRect.top path.conicTo(bodyRect.right, bodyRect.top, bodyRect.right,
,bodyRect.right,bodyRect.top + radius,1.0); bodyRect.top + radius, 1.0);
path.lineTo(bodyRect.right,bodyRect.bottom - radius); //右下角 path.lineTo(bodyRect.right, bodyRect.bottom - radius); //右下角
path.conicTo(bodyRect.right,bodyRect.bottom path.conicTo(bodyRect.right, bodyRect.bottom, bodyRect.right - radius,
,bodyRect.right -radius ,bodyRect.bottom,1.0); bodyRect.bottom, 1.0);
path.lineTo(arrowRect.right, arrowRect.top); //箭头
path.lineTo(arrowRect.left + arrowRect.width / 2, arrowRect.bottom);
path.lineTo(arrowRect.left,arrowRect.top);
path.lineTo(bodyRect.left + radius, bodyRect.bottom); //左下角 path.lineTo(bodyRect.left + radius, bodyRect.bottom); //左下角
path.conicTo(bodyRect.left,bodyRect.bottom path.conicTo(bodyRect.left, bodyRect.bottom, bodyRect.left,
,bodyRect.left ,bodyRect.bottom - radius,1.0); bodyRect.bottom - radius, 1.0);
path.lineTo(bodyRect.left, bodyRect.top + radius); //左上角 path.lineTo(bodyRect.left, bodyRect.top + radius); //左上角
path.conicTo(bodyRect.left,bodyRect.top path.conicTo(bodyRect.left, bodyRect.top, bodyRect.left + radius,
,bodyRect.left + radius,bodyRect.top,1.0); bodyRect.top, 1.0);
} }
path.close(); path.close();
return path; return path;
} }
} }
CupertinoPopoverDirection _calcDirection(Rect attachRect,Size size, CupertinoPopoverDirection direction) { CupertinoPopoverDirection _calcDirection(
bool isArrowUp; Rect attachRect, Size size, CupertinoPopoverDirection direction) {
switch(direction){ switch (direction) {
case CupertinoPopoverDirection.top: case CupertinoPopoverDirection.top:
isArrowUp = attachRect.top < size.height + CupertinoPopoverState._arrowHeight; // 判断顶部位置够不够 return (attachRect.top < size.height + CupertinoPopoverState._arrowHeight)
break; ? CupertinoPopoverDirection.bottom
: CupertinoPopoverDirection.top; // 判断顶部位置够不够
case CupertinoPopoverDirection.bottom: case CupertinoPopoverDirection.bottom:
isArrowUp = _ScreenUtil.getInstance().screenHeight > attachRect.bottom + size.height + CupertinoPopoverState._arrowHeight; return _ScreenUtil.getInstance().screenHeight >
break; attachRect.bottom +
} size.height +
return isArrowUp ? CupertinoPopoverDirection.bottom : CupertinoPopoverDirection.top; CupertinoPopoverState._arrowHeight
} ? CupertinoPopoverDirection.bottom
\ No newline at end of file : CupertinoPopoverDirection.top;
case CupertinoPopoverDirection.left:
return (attachRect.left < size.width + CupertinoPopoverState._arrowHeight)
? CupertinoPopoverDirection.right
: CupertinoPopoverDirection.left; // 判断顶部位置够不够
case CupertinoPopoverDirection.right:
return _ScreenUtil.getInstance().screenWidth >
attachRect.right +
size.width +
CupertinoPopoverState._arrowHeight
? CupertinoPopoverDirection.right
: CupertinoPopoverDirection.left;
}
}
import 'package:flutter/material.dart'; part of cool_ui;
class CoolTable extends StatefulWidget{ class CoolTable extends StatefulWidget{
@override @override
State<StatefulWidget> createState() { State<StatefulWidget> createState() {
// TODO: implement createState // TODO: implement createState
return null; return CoolTableState();
} }
} }
class CoolTableState extends State<CoolTable>{ class CoolTableState extends State<CoolTable>{
ScrollController hTitle = ScrollController();
ScrollController hBody = ScrollController();
@override
void initState() {
// TODO: implement initState
super.initState();
connectScroll(hTitle, hBody);
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
// TODO: implement build // TODO: implement build
return null; return Column(
mainAxisSize: MainAxisSize.min,
children: <Widget>[
SizedBox(
height: 50,
child: ListView(
scrollDirection: Axis.horizontal,
shrinkWrap: true,
physics: BouncingScrollPhysics(),
controller: hTitle,
children: <Widget>[
Text('1111111111111111111111111111111111111111111111111111111111111111111111111111')
],
),
width: 375,),
SizedBox(
height: 50,
child: ListView(
physics: BouncingScrollPhysics(),
scrollDirection: Axis.horizontal,
controller: hBody,
shrinkWrap: true,
children: <Widget>[
Text('1111111111111111111111111111111111111111111111111111111111111111111111111111')
],
)),
],
);
} }
} }
class CoolColumnInfo{ class CoolColumnInfo{
final double flex; final double flex;
final double width; final double width;
final double minWidth; final Widget title;
final double maxWidth;
const CoolColumnInfo({this.flex, this.width, this.minWidth, this.maxWidth}); const CoolColumnInfo({this.flex, this.width, this.title});
} }
\ No newline at end of file
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.6.0 version: 0.6.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 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论