mirror of
https://github.com/venera-app/venera.git
synced 2025-09-27 07:47:24 +00:00
initial commit
This commit is contained in:
315
lib/components/flyout.dart
Normal file
315
lib/components/flyout.dart
Normal file
@@ -0,0 +1,315 @@
|
||||
part of "components.dart";
|
||||
|
||||
const minFlyoutWidth = 256.0;
|
||||
const minFlyoutHeight = 128.0;
|
||||
|
||||
class FlyoutController {
|
||||
Function? _show;
|
||||
|
||||
void show() {
|
||||
if (_show == null) {
|
||||
throw "FlyoutController is not attached to a Flyout";
|
||||
}
|
||||
_show!();
|
||||
}
|
||||
}
|
||||
|
||||
class Flyout extends StatefulWidget {
|
||||
const Flyout(
|
||||
{super.key,
|
||||
required this.flyoutBuilder,
|
||||
required this.child,
|
||||
this.enableTap = false,
|
||||
this.enableDoubleTap = false,
|
||||
this.enableLongPress = false,
|
||||
this.enableSecondaryTap = false,
|
||||
this.withInkWell = false,
|
||||
this.borderRadius = 0,
|
||||
this.controller,
|
||||
this.navigator});
|
||||
|
||||
final WidgetBuilder flyoutBuilder;
|
||||
|
||||
final Widget child;
|
||||
|
||||
final bool enableTap;
|
||||
|
||||
final bool enableDoubleTap;
|
||||
|
||||
final bool enableLongPress;
|
||||
|
||||
final bool enableSecondaryTap;
|
||||
|
||||
final bool withInkWell;
|
||||
|
||||
final double borderRadius;
|
||||
|
||||
final NavigatorState? navigator;
|
||||
|
||||
final FlyoutController? controller;
|
||||
|
||||
@override
|
||||
State<Flyout> createState() => _FlyoutState();
|
||||
}
|
||||
|
||||
class _FlyoutState extends State<Flyout> {
|
||||
@override
|
||||
void initState() {
|
||||
if (widget.controller != null) {
|
||||
widget.controller?._show = show;
|
||||
}
|
||||
super.initState();
|
||||
}
|
||||
|
||||
@override
|
||||
void didChangeDependencies() {
|
||||
if (widget.controller != null) {
|
||||
widget.controller?._show = show;
|
||||
}
|
||||
super.didChangeDependencies();
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
if (widget.withInkWell) {
|
||||
return InkWell(
|
||||
borderRadius: BorderRadius.circular(widget.borderRadius),
|
||||
onTap: widget.enableTap ? show : null,
|
||||
onDoubleTap: widget.enableDoubleTap ? show : null,
|
||||
onLongPress: widget.enableLongPress ? show : null,
|
||||
onSecondaryTap: widget.enableSecondaryTap ? show : null,
|
||||
child: widget.child,
|
||||
);
|
||||
}
|
||||
return GestureDetector(
|
||||
onTap: widget.enableTap ? show : null,
|
||||
onDoubleTap: widget.enableDoubleTap ? show : null,
|
||||
onLongPress: widget.enableLongPress ? show : null,
|
||||
onSecondaryTap: widget.enableSecondaryTap ? show : null,
|
||||
child: widget.child,
|
||||
);
|
||||
}
|
||||
|
||||
void show() {
|
||||
var renderBox = context.findRenderObject() as RenderBox;
|
||||
var rect = renderBox.localToGlobal(Offset.zero) & renderBox.size;
|
||||
var navigator = widget.navigator ?? Navigator.of(context);
|
||||
navigator.push(PageRouteBuilder(
|
||||
fullscreenDialog: true,
|
||||
barrierDismissible: true,
|
||||
opaque: false,
|
||||
transitionDuration: _fastAnimationDuration,
|
||||
reverseTransitionDuration: _fastAnimationDuration,
|
||||
pageBuilder: (context, animation, secondaryAnimation) {
|
||||
var left = rect.left;
|
||||
var top = rect.bottom;
|
||||
|
||||
if (left + minFlyoutWidth > MediaQuery.of(context).size.width) {
|
||||
left = MediaQuery.of(context).size.width - minFlyoutWidth;
|
||||
}
|
||||
if (top + minFlyoutHeight > MediaQuery.of(context).size.height) {
|
||||
top = MediaQuery.of(context).size.height - minFlyoutHeight;
|
||||
}
|
||||
|
||||
Widget transition(BuildContext context, Animation<double> animation,
|
||||
Animation<double> secondaryAnimation, Widget flyout) {
|
||||
return SlideTransition(
|
||||
position: Tween<Offset>(
|
||||
begin: const Offset(0, -0.05),
|
||||
end: const Offset(0, 0),
|
||||
).animate(animation),
|
||||
child: flyout,
|
||||
);
|
||||
}
|
||||
|
||||
return Stack(
|
||||
children: [
|
||||
Positioned.fill(
|
||||
child: GestureDetector(
|
||||
behavior: HitTestBehavior.opaque,
|
||||
onTap: navigator.pop,
|
||||
child: AnimatedBuilder(
|
||||
animation: animation,
|
||||
builder: (context, builder) {
|
||||
return ColoredBox(
|
||||
color: Colors.black.withOpacity(0.3 * animation.value),
|
||||
);
|
||||
},
|
||||
),
|
||||
),
|
||||
),
|
||||
Positioned(
|
||||
left: left,
|
||||
right: 0,
|
||||
top: top,
|
||||
bottom: 0,
|
||||
child: transition(
|
||||
context,
|
||||
animation,
|
||||
secondaryAnimation,
|
||||
Align(
|
||||
alignment: Alignment.topLeft,
|
||||
child: widget.flyoutBuilder(context),
|
||||
)),
|
||||
)
|
||||
],
|
||||
);
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
class FlyoutContent extends StatelessWidget {
|
||||
const FlyoutContent(
|
||||
{super.key, required this.title, required this.actions, this.content});
|
||||
|
||||
final String title;
|
||||
|
||||
final String? content;
|
||||
|
||||
final List<Widget> actions;
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return IntrinsicWidth(
|
||||
child: Material(
|
||||
borderRadius: BorderRadius.circular(16),
|
||||
type: MaterialType.card,
|
||||
elevation: 1,
|
||||
surfaceTintColor: Theme.of(context).colorScheme.surfaceTint,
|
||||
child: Container(
|
||||
constraints: const BoxConstraints(
|
||||
minWidth: minFlyoutWidth,
|
||||
),
|
||||
padding: const EdgeInsets.symmetric(vertical: 8, horizontal: 16),
|
||||
child: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(title,
|
||||
style: const TextStyle(
|
||||
fontWeight: FontWeight.bold, fontSize: 16)),
|
||||
if (content != null)
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8),
|
||||
child: Text(content!, style: const TextStyle(fontSize: 12)),
|
||||
),
|
||||
const SizedBox(
|
||||
height: 12,
|
||||
),
|
||||
Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
mainAxisAlignment: MainAxisAlignment.end,
|
||||
children: [const Spacer(), ...actions],
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
).paddingAll(4),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class FlyoutTextButton extends StatefulWidget {
|
||||
const FlyoutTextButton(
|
||||
{super.key,
|
||||
required this.child,
|
||||
required this.flyoutBuilder,
|
||||
this.navigator});
|
||||
|
||||
final Widget child;
|
||||
|
||||
final WidgetBuilder flyoutBuilder;
|
||||
|
||||
final NavigatorState? navigator;
|
||||
|
||||
@override
|
||||
State<FlyoutTextButton> createState() => _FlyoutTextButtonState();
|
||||
}
|
||||
|
||||
class _FlyoutTextButtonState extends State<FlyoutTextButton> {
|
||||
final FlyoutController _controller = FlyoutController();
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Flyout(
|
||||
controller: _controller,
|
||||
flyoutBuilder: widget.flyoutBuilder,
|
||||
navigator: widget.navigator,
|
||||
child: TextButton(
|
||||
onPressed: () {
|
||||
_controller.show();
|
||||
},
|
||||
child: widget.child,
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
class FlyoutIconButton extends StatefulWidget {
|
||||
const FlyoutIconButton(
|
||||
{super.key,
|
||||
required this.icon,
|
||||
required this.flyoutBuilder,
|
||||
this.navigator});
|
||||
|
||||
final Widget icon;
|
||||
|
||||
final WidgetBuilder flyoutBuilder;
|
||||
|
||||
final NavigatorState? navigator;
|
||||
|
||||
@override
|
||||
State<FlyoutIconButton> createState() => _FlyoutIconButtonState();
|
||||
}
|
||||
|
||||
class _FlyoutIconButtonState extends State<FlyoutIconButton> {
|
||||
final FlyoutController _controller = FlyoutController();
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Flyout(
|
||||
controller: _controller,
|
||||
flyoutBuilder: widget.flyoutBuilder,
|
||||
navigator: widget.navigator,
|
||||
child: IconButton(
|
||||
onPressed: () {
|
||||
_controller.show();
|
||||
},
|
||||
icon: widget.icon,
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
class FlyoutFilledButton extends StatefulWidget {
|
||||
const FlyoutFilledButton(
|
||||
{super.key,
|
||||
required this.child,
|
||||
required this.flyoutBuilder,
|
||||
this.navigator});
|
||||
|
||||
final Widget child;
|
||||
|
||||
final WidgetBuilder flyoutBuilder;
|
||||
|
||||
final NavigatorState? navigator;
|
||||
|
||||
@override
|
||||
State<FlyoutFilledButton> createState() => _FlyoutFilledButtonState();
|
||||
}
|
||||
|
||||
class _FlyoutFilledButtonState extends State<FlyoutFilledButton> {
|
||||
final FlyoutController _controller = FlyoutController();
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Flyout(
|
||||
controller: _controller,
|
||||
flyoutBuilder: widget.flyoutBuilder,
|
||||
navigator: widget.navigator,
|
||||
child: ElevatedButton(
|
||||
onPressed: () {
|
||||
_controller.show();
|
||||
},
|
||||
child: widget.child,
|
||||
));
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user