spotube/lib/components/track_presentation/track_presentation.dart

101 lines
3.5 KiB
Dart

import 'package:flutter/material.dart' show ListTile;
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:shadcn_flutter/shadcn_flutter_extension.dart';
import 'package:spotube/components/titlebar/titlebar.dart';
import 'package:spotube/components/track_presentation/presentation_list.dart';
import 'package:spotube/components/track_presentation/presentation_props.dart';
import 'package:spotube/components/track_presentation/presentation_top.dart';
import 'package:spotube/components/track_presentation/presentation_modifiers.dart';
import 'package:spotube/extensions/constrains.dart';
import 'package:spotube/extensions/context.dart';
import 'package:spotube/utils/platform.dart';
class TrackPresentation extends HookConsumerWidget {
final TrackPresentationOptions options;
const TrackPresentation({
super.key,
required this.options,
});
@override
Widget build(BuildContext context, ref) {
final headerTextStyle = context.theme.typography.small.copyWith(
color: context.theme.colorScheme.mutedForeground,
);
final scrollController = useScrollController();
final focusNode = useFocusNode();
final scale = context.theme.scaling;
useEffect(() {
if (!kIsMobile) return null;
void listener() {
if (!scrollController.hasClients) return;
if (focusNode.hasFocus) {
scrollController.animateTo(
300 * scale,
duration: const Duration(milliseconds: 300),
curve: Curves.easeInOut,
);
}
}
focusNode.addListener(listener);
return () {
focusNode.removeListener(listener);
};
}, [focusNode, scrollController, scale]);
return Data<TrackPresentationOptions>.inherit(
data: options,
child: SafeArea(
child: Scaffold(
headers: const [TitleBar()],
child: CustomScrollView(
controller: scrollController,
slivers: [
const TrackPresentationTopSection(),
const SliverGap(16),
SliverLayoutBuilder(
builder: (context, constrains) {
return SliverList.list(
children: [
TrackPresentationModifiersSection(
focusNode: focusNode,
),
ListTile(
titleTextStyle: headerTextStyle,
subtitleTextStyle: headerTextStyle,
leadingAndTrailingTextStyle: headerTextStyle,
leading: constrains.mdAndUp ? const Text(" #") : null,
title: Row(
children: [
Expanded(
flex: constrains.lgAndUp ? 5 : 6,
child: Text(context.l10n.title),
),
if (constrains.mdAndUp)
Expanded(
flex: 3,
child: Text(context.l10n.album),
),
Text(context.l10n.duration),
],
),
),
],
);
},
),
const PresentationListSection(),
const SliverSafeArea(sliver: SliverGap(10)),
],
),
),
),
);
}
}