Files
SpotiFLAC-Mobile/lib/widgets/preview_button.dart
T
zarzet 12fb942f16 feat: playback queue, preview exclusivity, player bug fixes
- PlaybackController with queue methods for albums/library/playlists
- Library tab play builds merged queue (downloaded + local together)
- Preview vs main player exclusivity
- Preview stops on bottom-nav tab switch
- Duration 0:00 fix, deleted track cleanup, Up Next sheet
- Animation utilities improvements
2026-06-28 22:22:51 +07:00

125 lines
4.2 KiB
Dart

import 'package:audio_service/audio_service.dart';
import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:spotiflac_android/l10n/l10n.dart';
import 'package:spotiflac_android/models/track.dart';
import 'package:spotiflac_android/providers/music_player_provider.dart';
import 'package:spotiflac_android/providers/preview_player_provider.dart';
class PreviewButton extends ConsumerWidget {
final Track track;
final double size;
const PreviewButton({super.key, required this.track, this.size = 24});
Future<void> _onPressed(BuildContext context, WidgetRef ref) async {
try {
await ref.read(previewPlayerProvider.notifier).toggle(track.previewUrl);
} catch (_) {
if (!context.mounted) return;
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text(context.l10n.previewUnavailable)),
);
}
}
/// Loosely matches the built-in player's current item to this track so the
/// per-track button stays in sync with the mini player instead of showing a
/// conflicting preview state.
bool _isCurrentMainTrack(MediaItem? item) {
if (item == null) return false;
String norm(String? s) => (s ?? '').toLowerCase().trim();
return norm(item.title) == norm(track.name) &&
norm(item.artist) == norm(track.artistName);
}
@override
Widget build(BuildContext context, WidgetRef ref) {
final colorScheme = Theme.of(context).colorScheme;
// When the built-in player is currently on this track, mirror and control
// it (consistent with the mini player) rather than the preview snippet.
final mainItem = ref.watch(currentMediaItemProvider).value;
if (_isCurrentMainTrack(mainItem)) {
final isPlaying =
ref.watch(playbackStateProvider).value?.playing ?? false;
return Transform.translate(
offset: const Offset(18, 0),
child: IconButton(
iconSize: size,
padding: EdgeInsets.zero,
alignment: Alignment.centerRight,
visualDensity: VisualDensity.compact,
constraints: const BoxConstraints(minWidth: 24, minHeight: 36),
icon: Icon(
isPlaying
? Icons.pause_circle_filled_rounded
: Icons.play_circle_fill_rounded,
color: colorScheme.primary,
),
tooltip: isPlaying ? context.l10n.previewStop : context.l10n.previewPlay,
onPressed: () =>
ref.read(musicPlayerControllerProvider).togglePlayPause(isPlaying),
),
);
}
if (!track.hasPreview) return const SizedBox.shrink();
final previewState = ref.watch(previewPlayerProvider);
final isActive = previewState.isActiveUrl(track.previewUrl);
final status = isActive ? previewState.status : PreviewStatus.idle;
final Widget icon;
final String tooltip;
switch (status) {
case PreviewStatus.loading:
icon = SizedBox(
width: size * 0.7,
height: size * 0.7,
child: CircularProgressIndicator(
strokeWidth: 2,
color: colorScheme.primary,
),
);
tooltip = context.l10n.previewStop;
break;
case PreviewStatus.playing:
icon = Icon(
Icons.pause_circle_filled_rounded,
color: colorScheme.primary,
);
tooltip = context.l10n.previewStop;
break;
case PreviewStatus.paused:
icon = Icon(
Icons.play_circle_fill_rounded,
color: colorScheme.primary,
);
tooltip = context.l10n.previewPlay;
break;
case PreviewStatus.idle:
icon = Icon(
Icons.play_circle_outline_rounded,
color: colorScheme.onSurfaceVariant,
);
tooltip = context.l10n.previewPlay;
break;
}
return Transform.translate(
offset: const Offset(18, 0),
child: IconButton(
iconSize: size,
padding: EdgeInsets.zero,
alignment: Alignment.centerRight,
visualDensity: VisualDensity.compact,
constraints: const BoxConstraints(minWidth: 24, minHeight: 36),
icon: icon,
tooltip: tooltip,
onPressed: () => _onPressed(context, ref),
),
);
}
}