Some checks failed
Flutter CI/CD — PluriWave / Test + Build (pull_request) Has been cancelled
- Modelo Emisora: campos completos Radio Browser API (fromApi + fromMap) - ServicioRadio: cliente Radio Browser API (populares, tendencias, buscar por nombre/país/idioma/tag) - ServicioAudio: just_audio + audio_service wrapper (play/pause/stop/toggle, fade, background handler) - ServicioTimer: countdown con fade out gradual (15/30/60/90 min) - ServicioFavoritos: actualizado a v2 con campos codec/bitrate/votes/clickcount - EstadoRadio: ChangeNotifier global con Provider - PantallaInicio: grid emisoras populares, chips género, shimmer loading, pull-to-refresh - PantallaBuscar: SearchBar + filtros país/idioma, lista resultados - PantallaFavoritos: ReorderableListView + swipe-to-delete (Dismissible) - TarjetaEmisora: card + modo compacto ListTile, cached_network_image, shimmer fallback - MiniReproductor: barra inferior persistente con stream de estado - app.dart: MaterialApp + Provider + NavigationBar + timer dialog - main.dart: punto de entrada limpio - AndroidManifest.xml: permisos INTERNET + FOREGROUND_SERVICE + audio_service receivers
115 lines
4.0 KiB
Dart
115 lines
4.0 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:provider/provider.dart';
|
|
import '../estado/estado_radio.dart';
|
|
import '../servicios/servicio_audio.dart';
|
|
|
|
/// Barra inferior persistente con controles básicos de reproducción.
|
|
/// Se muestra siempre que haya una emisora cargada.
|
|
class MiniReproductor extends StatelessWidget {
|
|
const MiniReproductor({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final estado = context.watch<EstadoRadio>();
|
|
final emisora = estado.emisoraActual;
|
|
|
|
if (emisora == null) return const SizedBox.shrink();
|
|
|
|
final theme = Theme.of(context);
|
|
|
|
return Container(
|
|
decoration: BoxDecoration(
|
|
color: theme.colorScheme.surfaceContainer,
|
|
border: Border(top: BorderSide(color: theme.colorScheme.outlineVariant, width: 0.5)),
|
|
),
|
|
child: SafeArea(
|
|
top: false,
|
|
child: Padding(
|
|
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
|
|
child: Row(
|
|
children: [
|
|
// Logo emisora
|
|
ClipRRect(
|
|
borderRadius: BorderRadius.circular(6),
|
|
child: Container(
|
|
width: 40,
|
|
height: 40,
|
|
color: theme.colorScheme.primaryContainer,
|
|
child: const Icon(Icons.radio, size: 22),
|
|
),
|
|
),
|
|
const SizedBox(width: 12),
|
|
// Nombre y estado
|
|
Expanded(
|
|
child: Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(
|
|
emisora.nombre,
|
|
style: theme.textTheme.bodyMedium?.copyWith(fontWeight: FontWeight.w600),
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
),
|
|
StreamBuilder<EstadoReproduccion>(
|
|
stream: estado.estadoStream,
|
|
builder: (context, snapshot) {
|
|
final s = snapshot.data ?? EstadoReproduccion.detenido;
|
|
return Text(
|
|
_labelEstado(s),
|
|
style: theme.textTheme.bodySmall?.copyWith(
|
|
color: theme.colorScheme.onSurfaceVariant,
|
|
),
|
|
);
|
|
},
|
|
),
|
|
],
|
|
),
|
|
),
|
|
// Botón play/pause
|
|
StreamBuilder<EstadoReproduccion>(
|
|
stream: estado.estadoStream,
|
|
builder: (context, snapshot) {
|
|
final s = snapshot.data ?? EstadoReproduccion.detenido;
|
|
if (s == EstadoReproduccion.cargando) {
|
|
return const SizedBox(
|
|
width: 40,
|
|
height: 40,
|
|
child: Padding(
|
|
padding: EdgeInsets.all(10),
|
|
child: CircularProgressIndicator(strokeWidth: 2),
|
|
),
|
|
);
|
|
}
|
|
return IconButton(
|
|
icon: Icon(s == EstadoReproduccion.reproduciendo
|
|
? Icons.pause_rounded
|
|
: Icons.play_arrow_rounded),
|
|
onPressed: estado.togglePlay,
|
|
tooltip: s == EstadoReproduccion.reproduciendo ? 'Pausar' : 'Reproducir',
|
|
);
|
|
},
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
String _labelEstado(EstadoReproduccion estado) {
|
|
switch (estado) {
|
|
case EstadoReproduccion.cargando:
|
|
return 'Conectando...';
|
|
case EstadoReproduccion.reproduciendo:
|
|
return 'En directo ●';
|
|
case EstadoReproduccion.pausado:
|
|
return 'Pausado';
|
|
case EstadoReproduccion.error:
|
|
return 'Error de conexión';
|
|
case EstadoReproduccion.detenido:
|
|
return 'Detenido';
|
|
}
|
|
}
|
|
}
|