2022-12-21 19:31:22 +00:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:lightmeter/res/dimens.dart';
|
|
|
|
|
2023-01-26 09:10:23 +00:00
|
|
|
class CenteredSlider extends StatefulWidget {
|
|
|
|
final Icon? icon;
|
2022-12-21 19:31:22 +00:00
|
|
|
final double value;
|
|
|
|
final double min;
|
|
|
|
final double max;
|
|
|
|
final ValueChanged<double> onChanged;
|
|
|
|
final bool isVertical;
|
|
|
|
|
2023-01-26 09:10:23 +00:00
|
|
|
const CenteredSlider({
|
|
|
|
this.icon,
|
2022-12-21 19:31:22 +00:00
|
|
|
required this.value,
|
|
|
|
required this.min,
|
|
|
|
required this.max,
|
|
|
|
required this.onChanged,
|
|
|
|
this.isVertical = false,
|
|
|
|
super.key,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
2023-01-26 09:10:23 +00:00
|
|
|
State<CenteredSlider> createState() => _CenteredSliderState();
|
2022-12-21 19:31:22 +00:00
|
|
|
}
|
|
|
|
|
2023-01-26 09:10:23 +00:00
|
|
|
class _CenteredSliderState extends State<CenteredSlider> {
|
2022-12-21 19:31:22 +00:00
|
|
|
double relativeValue = 0.0;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
relativeValue = (widget.value - widget.min) / (widget.max - widget.min);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2023-01-26 09:10:23 +00:00
|
|
|
void didUpdateWidget(CenteredSlider oldWidget) {
|
2022-12-21 19:31:22 +00:00
|
|
|
super.didUpdateWidget(oldWidget);
|
|
|
|
relativeValue = (widget.value - widget.min) / (widget.max - widget.min);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2023-01-26 09:10:23 +00:00
|
|
|
return SizedBox(
|
2024-04-07 08:54:57 +00:00
|
|
|
height: widget.isVertical ? double.maxFinite : Dimens.cameraSliderHandleArea,
|
|
|
|
width: !widget.isVertical ? double.maxFinite : Dimens.cameraSliderHandleArea,
|
2023-01-26 09:10:23 +00:00
|
|
|
child: LayoutBuilder(
|
|
|
|
builder: (context, constraints) {
|
|
|
|
final biggestSize = widget.isVertical ? constraints.maxHeight : constraints.maxWidth;
|
2024-04-07 08:54:57 +00:00
|
|
|
final handleDistance = biggestSize - Dimens.cameraSliderHandleArea;
|
2023-01-26 09:10:23 +00:00
|
|
|
return RotatedBox(
|
|
|
|
quarterTurns: widget.isVertical ? -1 : 0,
|
|
|
|
child: GestureDetector(
|
|
|
|
behavior: HitTestBehavior.translucent,
|
2023-01-26 15:03:48 +00:00
|
|
|
onTapUp: (details) => _updateHandlePosition(
|
|
|
|
details.localPosition.dx,
|
|
|
|
handleDistance,
|
|
|
|
),
|
|
|
|
onHorizontalDragUpdate: (details) => _updateHandlePosition(
|
|
|
|
details.localPosition.dx,
|
|
|
|
handleDistance,
|
|
|
|
),
|
2023-01-26 09:10:23 +00:00
|
|
|
child: SizedBox(
|
2024-04-07 08:54:57 +00:00
|
|
|
height: Dimens.cameraSliderHandleArea,
|
2023-01-26 09:10:23 +00:00
|
|
|
width: biggestSize,
|
|
|
|
child: _Slider(
|
|
|
|
handleDistance: handleDistance,
|
2024-04-07 08:54:57 +00:00
|
|
|
handleSize: Dimens.cameraSliderHandleArea,
|
2023-01-26 09:10:23 +00:00
|
|
|
trackThickness: Dimens.cameraSliderTrackHeight,
|
|
|
|
value: relativeValue,
|
|
|
|
icon: RotatedBox(
|
|
|
|
quarterTurns: widget.isVertical ? 1 : 0,
|
|
|
|
child: widget.icon,
|
|
|
|
),
|
2022-12-21 19:31:22 +00:00
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
2023-01-26 09:10:23 +00:00
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
2022-12-21 19:31:22 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _updateHandlePosition(double offset, double handleDistance) {
|
2024-04-07 08:54:57 +00:00
|
|
|
if (offset <= Dimens.cameraSliderHandleArea / 2) {
|
2022-12-21 19:31:22 +00:00
|
|
|
relativeValue = 0;
|
2024-04-07 08:54:57 +00:00
|
|
|
} else if (offset >= handleDistance + Dimens.cameraSliderHandleArea / 2) {
|
2022-12-21 19:31:22 +00:00
|
|
|
relativeValue = 1;
|
|
|
|
} else {
|
2024-04-07 08:54:57 +00:00
|
|
|
relativeValue = (offset - Dimens.cameraSliderHandleArea / 2) / handleDistance;
|
2022-12-21 19:31:22 +00:00
|
|
|
}
|
|
|
|
setState(() {});
|
2023-01-26 15:03:48 +00:00
|
|
|
widget.onChanged(relativeValue * (widget.max - widget.min) + widget.min);
|
2022-12-21 19:31:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _Slider extends StatelessWidget {
|
|
|
|
final double handleSize;
|
|
|
|
final double trackThickness;
|
|
|
|
final double handleDistance;
|
|
|
|
final double value;
|
|
|
|
final Widget icon;
|
|
|
|
|
|
|
|
const _Slider({
|
|
|
|
required this.handleSize,
|
|
|
|
required this.trackThickness,
|
|
|
|
required this.handleDistance,
|
|
|
|
required this.value,
|
|
|
|
required this.icon,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Stack(
|
|
|
|
alignment: Alignment.center,
|
|
|
|
children: [
|
|
|
|
Positioned(
|
|
|
|
height: trackThickness,
|
2023-01-26 15:03:48 +00:00
|
|
|
|
|
|
|
/// add thickness to maintain radius overlap with handle
|
|
|
|
width: handleDistance + trackThickness,
|
2022-12-21 19:31:22 +00:00
|
|
|
child: ClipRRect(
|
|
|
|
borderRadius: BorderRadius.circular(trackThickness / 2),
|
2025-01-06 14:20:18 +00:00
|
|
|
child: ColoredBox(color: Theme.of(context).colorScheme.surfaceContainerHighest),
|
2022-12-21 19:31:22 +00:00
|
|
|
),
|
|
|
|
),
|
|
|
|
AnimatedPositioned.fromRect(
|
2024-04-07 08:54:57 +00:00
|
|
|
duration: Dimens.durationS,
|
2022-12-21 19:31:22 +00:00
|
|
|
rect: Rect.fromCenter(
|
|
|
|
center: Offset(
|
|
|
|
handleSize / 2 + handleDistance * value,
|
|
|
|
handleSize / 2,
|
|
|
|
),
|
|
|
|
width: handleSize,
|
|
|
|
height: handleSize,
|
|
|
|
),
|
2024-04-07 08:54:57 +00:00
|
|
|
child: Center(
|
|
|
|
child: _Handle(
|
|
|
|
color: Theme.of(context).colorScheme.primary,
|
|
|
|
size: Dimens.cameraSliderHandleSize,
|
|
|
|
child: IconTheme(
|
|
|
|
data: Theme.of(context).iconTheme.copyWith(
|
|
|
|
color: Theme.of(context).colorScheme.onPrimary,
|
|
|
|
size: Dimens.cameraSliderHandleIconSize,
|
|
|
|
),
|
|
|
|
child: icon,
|
|
|
|
),
|
2022-12-21 19:31:22 +00:00
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _Handle extends StatelessWidget {
|
|
|
|
final Color color;
|
|
|
|
final double size;
|
|
|
|
final Widget? child;
|
|
|
|
|
|
|
|
const _Handle({
|
|
|
|
required this.color,
|
|
|
|
required this.size,
|
|
|
|
this.child,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2024-04-07 08:54:57 +00:00
|
|
|
return SizedBox(
|
|
|
|
height: size,
|
|
|
|
width: size,
|
|
|
|
child: DecoratedBox(
|
|
|
|
decoration: BoxDecoration(
|
2022-12-21 19:31:22 +00:00
|
|
|
color: color,
|
2024-04-07 08:54:57 +00:00
|
|
|
shape: BoxShape.circle,
|
2022-12-21 19:31:22 +00:00
|
|
|
),
|
2024-04-07 08:54:57 +00:00
|
|
|
child: child,
|
2022-12-21 19:31:22 +00:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|