refa: video progress indicator

Signed-off-by: dom <githubaccount56556@proton.me>
This commit is contained in:
dom
2026-01-18 22:21:48 +08:00
parent f5657d2d4c
commit 395893fc7d
7 changed files with 193 additions and 35 deletions

View File

@@ -181,7 +181,7 @@ class ProgressBar extends LeafRenderObjectWidget {
@override @override
RenderObject createRenderObject(BuildContext context) { RenderObject createRenderObject(BuildContext context) {
return _RenderProgressBar( return RenderProgressBar(
progress: progress, progress: progress,
total: total, total: total,
buffered: buffered ?? Duration.zero, buffered: buffered ?? Duration.zero,
@@ -203,8 +203,11 @@ class ProgressBar extends LeafRenderObjectWidget {
} }
@override @override
void updateRenderObject(BuildContext context, RenderObject renderObject) { void updateRenderObject(
(renderObject as _RenderProgressBar) BuildContext context,
RenderProgressBar renderObject,
) {
renderObject
..total = total ..total = total
..progress = progress ..progress = progress
..buffered = buffered ?? Duration.zero ..buffered = buffered ?? Duration.zero
@@ -327,8 +330,8 @@ class _EagerHorizontalDragGestureRecognizer
String get debugDescription => '_EagerHorizontalDragGestureRecognizer'; String get debugDescription => '_EagerHorizontalDragGestureRecognizer';
} }
class _RenderProgressBar extends RenderBox { class RenderProgressBar extends RenderBox {
_RenderProgressBar({ RenderProgressBar({
required Duration progress, required Duration progress,
required Duration total, required Duration total,
required Duration buffered, required Duration buffered,

View File

@@ -1,28 +1,162 @@
import 'package:PiliPlus/common/constants.dart'; /*
import 'package:flutter/material.dart'; * This file is part of PiliPlus
*
* PiliPlus is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* PiliPlus is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with PiliPlus. If not, see <https://www.gnu.org/licenses/>.
*/
Widget videoProgressIndicator(double progress) => ClipRect( import 'package:flutter/widgets.dart';
clipper: ProgressClipper(),
child: ClipRRect( class VideoProgressIndicator extends LeafRenderObjectWidget {
borderRadius: const BorderRadius.vertical(bottom: StyleString.imgRadius), const VideoProgressIndicator({
child: LinearProgressIndicator( super.key,
minHeight: 10, required this.color,
value: progress, required this.backgroundColor,
// ignore: deprecated_member_use this.radius = 10,
year2023: true, this.height = 4,
stopIndicatorColor: Colors.transparent, required this.progress,
), }) : assert(progress >= 0 && progress <= 1);
),
); final Color color;
final Color backgroundColor;
final double radius;
final double height;
final double progress;
class ProgressClipper extends CustomClipper<Rect> {
@override @override
Rect getClip(Size size) { RenderObject createRenderObject(BuildContext context) {
return Rect.fromLTWH(0, 6, size.width, size.height - 6); return RenderProgressBar(
color: color,
backgroundColor: backgroundColor,
radius: radius,
height: height,
progress: progress,
);
} }
@override @override
bool shouldReclip(CustomClipper<Rect> oldClipper) { void updateRenderObject(
return false; BuildContext context,
RenderProgressBar renderObject,
) {
renderObject
..color = color
..backgroundColor = backgroundColor
..radius = radius
..height = height
..progress = progress;
} }
} }
class RenderProgressBar extends RenderBox {
RenderProgressBar({
required Color color,
required Color backgroundColor,
required double radius,
required double height,
required double progress,
}) : _color = color,
_backgroundColor = backgroundColor,
_radius = radius,
_height = height,
_progress = progress;
Color _color;
Color get color => _color;
set color(Color value) {
if (_color == value) return;
_color = value;
markNeedsPaint();
}
Color _backgroundColor;
Color get backgroundColor => _backgroundColor;
set backgroundColor(Color value) {
if (_backgroundColor == value) return;
_backgroundColor = value;
markNeedsPaint();
}
double _progress;
double get progress => _progress;
set progress(double value) {
if (_progress == value) return;
_progress = value;
markNeedsPaint();
}
double _radius;
double get borderRadius => _radius;
set radius(double value) {
if (_radius == value) return;
_radius = value;
markNeedsLayout();
}
double _height;
double get height => _height;
set height(double value) {
if (_height == value) return;
_height = value;
markNeedsPaint();
}
@override
void performLayout() {
size = computeDryLayout(constraints);
}
@override
Size computeDryLayout(BoxConstraints constraints) {
return constraints.constrain(
Size(constraints.maxWidth, _radius),
);
}
@override
void paint(PaintingContext context, Offset offset) {
final size = this.size;
final canvas = context.canvas;
final paint = Paint()..style = .fill;
canvas.clipRect(
.fromLTWH(0, size.height - height, size.width, height),
);
final radius = Radius.circular(_radius);
final rect = Rect.fromLTWH(0, 0, size.width, size.height);
if (progress == 0) {
canvas.drawRRect(
.fromRectAndCorners(rect, bottomLeft: radius, bottomRight: radius),
paint..color = _backgroundColor,
);
} else if (progress == 1) {
canvas.drawRRect(
.fromRectAndCorners(rect, bottomLeft: radius, bottomRight: radius),
paint..color = _color,
);
} else {
final w = size.width * progress;
final left = Rect.fromLTWH(0, 0, w, size.height);
final right = Rect.fromLTWH(w, 0, size.width - w, size.height);
canvas
..clipRRect(.fromRectAndRadius(rect, radius))
..drawRect(left, paint..color = _color)
..drawRect(right, paint..color = _backgroundColor);
}
}
@override
bool get isRepaintBoundary => true;
}

View File

@@ -63,6 +63,7 @@ class VideoCardH extends StatelessWidget {
title: videoItem.title, title: videoItem.title,
cover: videoItem.cover, cover: videoItem.cover,
); );
final colorScheme = ColorScheme.of(context);
return Material( return Material(
type: MaterialType.transparency, type: MaterialType.transparency,
child: Stack( child: Stack(
@@ -166,8 +167,11 @@ class VideoCardH extends StatelessWidget {
left: 0, left: 0,
bottom: 0, bottom: 0,
right: 0, right: 0,
child: videoProgressIndicator( child: VideoProgressIndicator(
progress == -1 color: colorScheme.primary,
backgroundColor:
colorScheme.secondaryContainer,
progress: progress == -1
? 1 ? 1
: progress / videoItem.duration, : progress / videoItem.duration,
), ),

View File

@@ -224,8 +224,11 @@ class DetailItem extends StatelessWidget {
child: Stack( child: Stack(
clipBehavior: Clip.none, clipBehavior: Clip.none,
children: [ children: [
videoProgressIndicator( VideoProgressIndicator(
progress / entry.totalTimeMilli, color: theme.colorScheme.primary,
backgroundColor:
theme.colorScheme.secondaryContainer,
progress: progress / entry.totalTimeMilli,
), ),
PBadge( PBadge(
text: progress >= entry.totalTimeMilli - 400 text: progress >= entry.totalTimeMilli - 400

View File

@@ -157,8 +157,11 @@ class HistoryItem extends StatelessWidget {
left: 0, left: 0,
right: 0, right: 0,
bottom: 0, bottom: 0,
child: videoProgressIndicator( child: VideoProgressIndicator(
item.progress == -1 color: theme.colorScheme.primary,
backgroundColor:
theme.colorScheme.secondaryContainer,
progress: item.progress == -1
? 1 ? 1
: item.progress! / item.duration!, : item.progress! / item.duration!,
), ),

View File

@@ -136,8 +136,11 @@ class VideoCardHLater extends StatelessWidget {
left: 0, left: 0,
bottom: 0, bottom: 0,
right: 0, right: 0,
child: videoProgressIndicator( child: VideoProgressIndicator(
progress == -1 color: theme.colorScheme.primary,
backgroundColor:
theme.colorScheme.secondaryContainer,
progress: progress == -1
? 1 ? 1
: progress / videoItem.duration!, : progress / videoItem.duration!,
), ),

View File

@@ -141,8 +141,16 @@ class VideoCardHMemberVideo extends StatelessWidget {
left: 0, left: 0,
right: 0, right: 0,
bottom: 0, bottom: 0,
child: videoProgressIndicator( child: VideoProgressIndicator(
videoItem.history!.progress! / color:
theme.colorScheme.primary,
backgroundColor: theme
.colorScheme
.secondaryContainer,
progress:
videoItem
.history!
.progress! /
videoItem videoItem
.history! .history!
.duration!, .duration!,