fix(player): stabilize equalizer and visualizer
Build & Deploy Pluriwave / Análisis de código (push) Successful in 12s
Build & Deploy Pluriwave / Build APK + AAB release (push) Successful in 1m50s

This commit is contained in:
2026-05-21 21:56:25 +02:00
parent d0ceaac3f3
commit 921e972183
8 changed files with 427 additions and 228 deletions
+16
View File
@@ -69,6 +69,7 @@ class EstadoRadio extends ChangeNotifier {
final Map<String, PresetEcualizador> _presetsEmisoraMap = {}; final Map<String, PresetEcualizador> _presetsEmisoraMap = {};
PresetEcualizador _presetPrincipal = PresetEcualizador.flat; PresetEcualizador _presetPrincipal = PresetEcualizador.flat;
PresetEcualizador _presetActual = PresetEcualizador.flat; PresetEcualizador _presetActual = PresetEcualizador.flat;
bool _ecualizadorActivo = true;
bool _cargandoPopulares = false; bool _cargandoPopulares = false;
bool _cargandoBusqueda = false; bool _cargandoBusqueda = false;
@@ -102,6 +103,7 @@ class EstadoRadio extends ChangeNotifier {
Stream<EstadoReproduccion> get estadoStream => audio.estadoStream; Stream<EstadoReproduccion> get estadoStream => audio.estadoStream;
PresetEcualizador get presetEcualizador => _presetActual; PresetEcualizador get presetEcualizador => _presetActual;
PresetEcualizador get presetPrincipalEcualizador => _presetPrincipal; PresetEcualizador get presetPrincipalEcualizador => _presetPrincipal;
bool get ecualizadorActivo => _ecualizadorActivo;
bool get ecualizadorDisponible => audio.ecualizadorDisponible; bool get ecualizadorDisponible => audio.ecualizadorDisponible;
bool get emisoraActualEsFavorita { bool get emisoraActualEsFavorita {
@@ -172,13 +174,16 @@ class EstadoRadio extends ChangeNotifier {
final config = await servicioEcualizador.cargar(); final config = await servicioEcualizador.cargar();
_presetPrincipal = config.principal; _presetPrincipal = config.principal;
_presetActual = config.principal; _presetActual = config.principal;
_ecualizadorActivo = config.activo;
_presetsEmisoraMap _presetsEmisoraMap
..clear() ..clear()
..addAll(config.porEmisora); ..addAll(config.porEmisora);
await audio.setEcualizadorActivo(_ecualizadorActivo);
await audio.aplicarPreset(_presetPrincipal); await audio.aplicarPreset(_presetPrincipal);
} catch (_) { } catch (_) {
_presetPrincipal = PresetEcualizador.flat; _presetPrincipal = PresetEcualizador.flat;
_presetActual = PresetEcualizador.flat; _presetActual = PresetEcualizador.flat;
_ecualizadorActivo = true;
_presetsEmisoraMap.clear(); _presetsEmisoraMap.clear();
} }
} }
@@ -483,6 +488,16 @@ class EstadoRadio extends ChangeNotifier {
} }
} }
Future<void> cambiarEcualizadorActivo(bool activo) async {
_ecualizadorActivo = activo;
await servicioEcualizador.guardarActivo(activo);
await audio.setEcualizadorActivo(activo);
if (activo) {
await audio.aplicarPreset(_presetActual);
}
notifyListeners();
}
Future<void> cambiarPresetEcualizador( Future<void> cambiarPresetEcualizador(
PresetEcualizador preset, { PresetEcualizador preset, {
bool guardarPorEmisora = true, bool guardarPorEmisora = true,
@@ -632,6 +647,7 @@ class EstadoRadio extends ChangeNotifier {
ConfiguracionEcualizador( ConfiguracionEcualizador(
principal: _presetPrincipal, principal: _presetPrincipal,
porEmisora: _presetsEmisoraMap, porEmisora: _presetsEmisoraMap,
activo: _ecualizadorActivo,
), ),
); );
+17 -4
View File
@@ -181,13 +181,26 @@ class _SeccionEcualizador extends StatelessWidget {
style: Theme.of(ctx).textTheme.titleMedium, style: Theme.of(ctx).textTheme.titleMedium,
), ),
const Spacer(), const Spacer(),
if (!disponible) Chip(
const Chip( label: Text(
label: Text('Se guarda aunque no esté activo'), estado.ecualizadorActivo ? 'Activo' : 'Desactivado',
visualDensity: VisualDensity.compact,
), ),
visualDensity: VisualDensity.compact,
),
], ],
), ),
const SizedBox(height: 8),
SwitchListTile.adaptive(
contentPadding: EdgeInsets.zero,
title: const Text('Activar ecualizador'),
subtitle: Text(
disponible
? 'Los cambios se aplican en tiempo real a la emisora actual.'
: 'Se guardan los cambios y se aplicarán cuando Android habilite el efecto.',
),
value: estado.ecualizadorActivo,
onChanged: estado.cambiarEcualizadorActivo,
),
if (mostrarModoPorEmisora) ...[ if (mostrarModoPorEmisora) ...[
const SizedBox(height: 8), const SizedBox(height: 8),
SwitchListTile.adaptive( SwitchListTile.adaptive(
+15
View File
@@ -90,6 +90,21 @@ class _PantallaReproductorState extends State<PantallaReproductor>
onPressed: () => Navigator.pop(context), onPressed: () => Navigator.pop(context),
), ),
actions: [ actions: [
IconButton(
icon: Icon(
estado.ecualizadorActivo
? Icons.equalizer_rounded
: Icons.equalizer_outlined,
color: estado.ecualizadorActivo ? tokens.warmCoral : null,
),
tooltip:
estado.ecualizadorActivo
? 'Desactivar ecualizador'
: 'Activar ecualizador',
onPressed:
() =>
estado.cambiarEcualizadorActivo(!estado.ecualizadorActivo),
),
IconButton( IconButton(
icon: Icon( icon: Icon(
esFavorito esFavorito
+28 -4
View File
@@ -77,8 +77,11 @@ class ServicioAudio {
Future<void> setVolumen(double vol) => _handler.setVolumen(vol); Future<void> setVolumen(double vol) => _handler.setVolumen(vol);
double get volumen => _handler.volumen; double get volumen => _handler.volumen;
bool get estaSonando => _handler.playbackState.value.playing; bool get estaSonando => _handler.playbackState.value.playing;
Stream<int?> get androidAudioSessionIdStream => Stream<int?> get androidAudioSessionIdStream async* {
_handler.androidAudioSessionIdStream; yield _handler.androidAudioSessionId;
yield* _handler.androidAudioSessionIdStream;
}
Future<void> dispose() async {} Future<void> dispose() async {}
// ── Ecualizador ─────────────────────────────────────────────────────────── // ── Ecualizador ───────────────────────────────────────────────────────────
@@ -88,6 +91,8 @@ class ServicioAudio {
Future<void> aplicarPreset(PresetEcualizador preset) => Future<void> aplicarPreset(PresetEcualizador preset) =>
_handler.aplicarPreset(preset); _handler.aplicarPreset(preset);
Future<void> setEcualizadorActivo(bool activo) =>
_handler.setEcualizadorActivo(activo);
Future<void> setBanda(int index, double db) => _handler.setBanda(index, db); Future<void> setBanda(int index, double db) => _handler.setBanda(index, db);
} }
@@ -106,6 +111,7 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
StreamSubscription<PlaybackEvent>? _eventosSub; StreamSubscription<PlaybackEvent>? _eventosSub;
StreamSubscription<int?>? _androidAudioSessionIdSub; StreamSubscription<int?>? _androidAudioSessionIdSub;
final _androidAudioSessionIdController = StreamController<int?>.broadcast(); final _androidAudioSessionIdController = StreamController<int?>.broadcast();
int? _androidAudioSessionId;
Future<void> _colaCambioFuente = Future<void>.value(); Future<void> _colaCambioFuente = Future<void>.value();
int _revisionFuente = 0; int _revisionFuente = 0;
@@ -116,9 +122,12 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
AndroidEqualizer? get ecualizador => _eq; AndroidEqualizer? get ecualizador => _eq;
bool _eqDisponible = false; bool _eqDisponible = false;
bool get ecualizadorDisponible => _eqDisponible; bool get ecualizadorDisponible => _eqDisponible;
bool _ecualizadorActivo = true;
bool get ecualizadorActivo => _ecualizadorActivo;
PresetEcualizador _presetActual = PresetEcualizador.flat; PresetEcualizador _presetActual = PresetEcualizador.flat;
PresetEcualizador get presetActual => _presetActual; PresetEcualizador get presetActual => _presetActual;
int? get androidAudioSessionId => _androidAudioSessionId;
Stream<int?> get androidAudioSessionIdStream => Stream<int?> get androidAudioSessionIdStream =>
_androidAudioSessionIdController.stream; _androidAudioSessionIdController.stream;
@@ -166,6 +175,7 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
_androidAudioSessionIdSub = _player.androidAudioSessionIdStream.listen(( _androidAudioSessionIdSub = _player.androidAudioSessionIdStream.listen((
sessionId, sessionId,
) { ) {
_androidAudioSessionId = sessionId;
if (!_androidAudioSessionIdController.isClosed) { if (!_androidAudioSessionIdController.isClosed) {
_androidAudioSessionIdController.add(sessionId); _androidAudioSessionIdController.add(sessionId);
} }
@@ -316,6 +326,7 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
_eq = AndroidEqualizer(); _eq = AndroidEqualizer();
_eqDisponible = false; _eqDisponible = false;
_androidAudioSessionId = null;
_player = _crearPlayer(); _player = _crearPlayer();
await _player.setVolume(_volumen); await _player.setVolume(_volumen);
_conectarStreamsPlayer(); _conectarStreamsPlayer();
@@ -342,7 +353,7 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
final params = await _eq.parameters; final params = await _eq.parameters;
_eqDisponible = params.bands.isNotEmpty; _eqDisponible = params.bands.isNotEmpty;
if (_eqDisponible) { if (_eqDisponible) {
await _eq.setEnabled(true); await _eq.setEnabled(_ecualizadorActivo);
await aplicarPreset(_presetActual); await aplicarPreset(_presetActual);
} }
} catch (_) { } catch (_) {
@@ -355,6 +366,8 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
_presetActual = preset; _presetActual = preset;
if (!_eqDisponible) return; if (!_eqDisponible) return;
try { try {
await _eq.setEnabled(_ecualizadorActivo);
if (!_ecualizadorActivo) return;
final params = await _eq.parameters; final params = await _eq.parameters;
for ( for (
int i = 0; int i = 0;
@@ -368,12 +381,12 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
/// Ajusta una banda individual. /// Ajusta una banda individual.
Future<void> setBanda(int index, double db) async { Future<void> setBanda(int index, double db) async {
if (!_eqDisponible) return;
final bandas = List<double>.from(_presetActual.bandas); final bandas = List<double>.from(_presetActual.bandas);
if (index >= 0 && index < bandas.length) { if (index >= 0 && index < bandas.length) {
bandas[index] = db; bandas[index] = db;
_presetActual = _presetActual.copyWithBandas(bandas); _presetActual = _presetActual.copyWithBandas(bandas);
} }
if (!_eqDisponible || !_ecualizadorActivo) return;
try { try {
final params = await _eq.parameters; final params = await _eq.parameters;
if (index < params.bands.length) { if (index < params.bands.length) {
@@ -382,6 +395,17 @@ class PluriWaveAudioHandler extends BaseAudioHandler with SeekHandler {
} catch (_) {} } catch (_) {}
} }
Future<void> setEcualizadorActivo(bool activo) async {
_ecualizadorActivo = activo;
if (!_eqDisponible) return;
try {
await _eq.setEnabled(activo);
if (activo) {
await aplicarPreset(_presetActual);
}
} catch (_) {}
}
Future<void> setVolumen(double vol) async { Future<void> setVolumen(double vol) async {
_volumen = vol.clamp(0.0, 1.0); _volumen = vol.clamp(0.0, 1.0);
await _player.setVolume(_volumen); await _player.setVolume(_volumen);
+11 -3
View File
@@ -8,15 +8,18 @@ class ConfiguracionEcualizador {
const ConfiguracionEcualizador({ const ConfiguracionEcualizador({
required this.principal, required this.principal,
required this.porEmisora, required this.porEmisora,
this.activo = true,
}); });
final PresetEcualizador principal; final PresetEcualizador principal;
final Map<String, PresetEcualizador> porEmisora; final Map<String, PresetEcualizador> porEmisora;
final bool activo;
} }
class ServicioEcualizador { class ServicioEcualizador {
static const _keyPresetPrincipal = 'eq_preset_principal_v1'; static const _keyPresetPrincipal = 'eq_preset_principal_v1';
static const _keyPresetsPorEmisora = 'eq_presets_por_emisora_v1'; static const _keyPresetsPorEmisora = 'eq_presets_por_emisora_v1';
static const _keyActivo = 'eq_activo_v1';
Future<ConfiguracionEcualizador> cargar() async { Future<ConfiguracionEcualizador> cargar() async {
final prefs = await SharedPreferences.getInstance(); final prefs = await SharedPreferences.getInstance();
@@ -25,6 +28,7 @@ class ServicioEcualizador {
return ConfiguracionEcualizador( return ConfiguracionEcualizador(
principal: principal, principal: principal,
porEmisora: porEmisora, porEmisora: porEmisora,
activo: prefs.getBool(_keyActivo) ?? true,
); );
} }
@@ -40,6 +44,11 @@ class ServicioEcualizador {
await _guardarPresetsPorEmisora(prefs, mapa); await _guardarPresetsPorEmisora(prefs, mapa);
} }
Future<void> guardarActivo(bool activo) async {
final prefs = await SharedPreferences.getInstance();
await prefs.setBool(_keyActivo, activo);
}
Future<void> eliminarPorEmisora(String uuid) async { Future<void> eliminarPorEmisora(String uuid) async {
final prefs = await SharedPreferences.getInstance(); final prefs = await SharedPreferences.getInstance();
final mapa = _leerPresetsPorEmisora(prefs); final mapa = _leerPresetsPorEmisora(prefs);
@@ -54,6 +63,7 @@ class ServicioEcualizador {
jsonEncode(config.principal.toJson()), jsonEncode(config.principal.toJson()),
); );
await _guardarPresetsPorEmisora(prefs, config.porEmisora); await _guardarPresetsPorEmisora(prefs, config.porEmisora);
await prefs.setBool(_keyActivo, config.activo);
} }
PresetEcualizador _leerPresetPrincipal(SharedPreferences prefs) { PresetEcualizador _leerPresetPrincipal(SharedPreferences prefs) {
@@ -82,9 +92,7 @@ class ServicioEcualizador {
return data.map( return data.map(
(uuid, preset) => MapEntry( (uuid, preset) => MapEntry(
uuid, uuid,
PresetEcualizador.desdeJson( PresetEcualizador.desdeJson(Map<String, dynamic>.from(preset as Map)),
Map<String, dynamic>.from(preset as Map),
),
), ),
); );
} catch (_) { } catch (_) {
+59 -8
View File
@@ -40,7 +40,9 @@ class _VisualizadorAudioState extends State<VisualizadorAudio>
late final AnimationController _controller; late final AnimationController _controller;
bool _activo = false; bool _activo = false;
int? _sessionId; int? _sessionId;
List<double> _ondaReal = const []; List<double> _ondaObjetivo = const [];
List<double> _ondaVisual = const [];
DateTime? _ultimaOndaReal;
StreamSubscription<EstadoReproduccion>? _estadoSubscription; StreamSubscription<EstadoReproduccion>? _estadoSubscription;
StreamSubscription<int?>? _sessionSubscription; StreamSubscription<int?>? _sessionSubscription;
StreamSubscription<dynamic>? _ondaSubscription; StreamSubscription<dynamic>? _ondaSubscription;
@@ -52,7 +54,10 @@ class _VisualizadorAudioState extends State<VisualizadorAudio>
vsync: this, vsync: this,
duration: const Duration(seconds: 2), duration: const Duration(seconds: 2),
)..addListener(() { )..addListener(() {
if (mounted) setState(() {}); if (mounted) {
_actualizarOndaVisual();
setState(() {});
}
}); });
_estadoSubscription = widget.estadoStream.listen(_onEstado); _estadoSubscription = widget.estadoStream.listen(_onEstado);
_sessionSubscription = widget.androidAudioSessionIdStream?.listen( _sessionSubscription = widget.androidAudioSessionIdStream?.listen(
@@ -87,9 +92,7 @@ class _VisualizadorAudioState extends State<VisualizadorAudio>
if (!puedeCapturar) { if (!puedeCapturar) {
unawaited(_ondaSubscription?.cancel()); unawaited(_ondaSubscription?.cancel());
_ondaSubscription = null; _ondaSubscription = null;
if (_ondaReal.isNotEmpty && mounted) { _ultimaOndaReal = null;
setState(() => _ondaReal = const []);
}
return; return;
} }
@@ -107,18 +110,66 @@ class _VisualizadorAudioState extends State<VisualizadorAudio>
.map((v) => v.toDouble().clamp(0.0, 1.0)) .map((v) => v.toDouble().clamp(0.0, 1.0))
.toList(growable: false); .toList(growable: false);
if (muestras.isNotEmpty) { if (muestras.isNotEmpty) {
setState(() => _ondaReal = muestras); _ultimaOndaReal = DateTime.now();
_ondaObjetivo = _normalizar(muestras);
} }
}, },
onError: (_) { onError: (_) {
unawaited(_ondaSubscription?.cancel()); unawaited(_ondaSubscription?.cancel());
_ondaSubscription = null; _ondaSubscription = null;
if (mounted) setState(() => _ondaReal = const []); _ultimaOndaReal = null;
}, },
cancelOnError: false, cancelOnError: false,
); );
} }
void _actualizarOndaVisual() {
final objetivo = _objetivoActual();
if (_ondaVisual.length != objetivo.length) {
_ondaVisual = List<double>.from(objetivo);
return;
}
_ondaVisual = List<double>.generate(objetivo.length, (i) {
final suavizado = _ondaVisual[i] + (objetivo[i] - _ondaVisual[i]) * 0.16;
return suavizado.clamp(0.0, 1.0);
}, growable: false);
}
List<double> _objetivoActual() {
final ahora = DateTime.now();
final tieneReal =
_ultimaOndaReal != null &&
ahora.difference(_ultimaOndaReal!) <
const Duration(milliseconds: 900) &&
_ondaObjetivo.isNotEmpty;
if (tieneReal) return _ondaObjetivo;
return _ondaOrganica();
}
List<double> _ondaOrganica() {
final count = widget.barras.clamp(8, 96);
final phase = _controller.value * pi * 2;
final intensidad = _activo ? 1.0 : 0.18;
return List<double>.generate(count, (i) {
final p = count <= 1 ? 0.0 : i / (count - 1);
final envelope = sin(pi * p).clamp(0.10, 1.0);
final flow =
sin(phase + p * pi * 2.2) * 0.24 +
sin(phase * 0.63 - p * pi * 5.1) * 0.18 +
sin(phase * 1.37 + p * pi * 9.0) * 0.08;
final value = 0.5 + flow * envelope * intensidad;
return value.clamp(0.12, 0.88);
}, growable: false);
}
List<double> _normalizar(List<double> muestras) {
final maximo = muestras.fold<double>(0, (max, v) => v > max ? v : max);
if (maximo <= 0.001) return muestras;
return muestras
.map((v) => (0.08 + (v / maximo) * 0.84).clamp(0.0, 1.0))
.toList(growable: false);
}
@override @override
void dispose() { void dispose() {
_estadoSubscription?.cancel(); _estadoSubscription?.cancel();
@@ -141,7 +192,7 @@ class _VisualizadorAudioState extends State<VisualizadorAudio>
color: color, color: color,
phase: t, phase: t,
active: _activo, active: _activo,
waveform: _ondaReal, waveform: _ondaVisual,
), ),
child: const SizedBox.expand(), child: const SizedBox.expand(),
), ),
+238 -192
View File
@@ -35,223 +35,269 @@ void main() {
); );
}); });
test('carga EQ principal persistido antes de decidir EQ de reproducción', test(
() async { 'carga EQ principal persistido antes de decidir EQ de reproducción',
final audio = FakeServicioAudio(); () async {
final principal = PresetEcualizador.rock; final audio = FakeServicioAudio();
final emisora = emisoraDemo(uuid: 'api-1', nombre: 'API Uno'); final principal = PresetEcualizador.rock;
final estado = EstadoRadio( final emisora = emisoraDemo(uuid: 'api-1', nombre: 'API Uno');
audio: audio, final estado = EstadoRadio(
favoritos: FakeServicioFavoritos(), audio: audio,
radio: FakeServicioRadio(populares: [emisora]), favoritos: FakeServicioFavoritos(),
servicioEcualizador: FakeServicioEcualizador(principal: principal), radio: FakeServicioRadio(populares: [emisora]),
resolverArchivoCustom: _archivoCustomVacio, servicioEcualizador: FakeServicioEcualizador(principal: principal),
iniciarAutomaticamente: false, resolverArchivoCustom: _archivoCustomVacio,
); iniciarAutomaticamente: false,
);
await estado.inicializar(); await estado.inicializar();
await estado.reproducir(emisora); await estado.reproducir(emisora);
expect(estado.presetEcualizador, principal); expect(estado.presetEcualizador, principal);
expect(audio.presetsAplicados.first, principal); expect(audio.presetsAplicados.first, principal);
expect(audio.presetsAplicados.last, principal); expect(audio.presetsAplicados.last, principal);
}); },
);
test('mantiene EQ persistido aunque el ecualizador nativo no esté disponible',
() async {
final principal = PresetEcualizador.jazz;
final porEmisora = {'fav-1': PresetEcualizador.rock};
final estado = EstadoRadio(
audio: FakeServicioAudio(ecualizadorActivo: false),
favoritos: FakeServicioFavoritos(),
radio: FakeServicioRadio(),
servicioEcualizador: FakeServicioEcualizador(
principal: principal,
porEmisora: porEmisora,
),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar();
expect(estado.ecualizadorDisponible, isFalse);
expect(estado.presetEcualizador, principal);
expect(estado.presetPrincipalEcualizador, principal);
expect(
estado.presetEcualizadorPorEmisora('fav-1'),
PresetEcualizador.rock,
);
});
test( test(
'inicializar deja error tras fallo y cargarPopulares manual recupera estaciones', 'mantiene EQ persistido aunque el ecualizador nativo no esté disponible',
() async { () async {
final radio = FakeServicioRadio( final principal = PresetEcualizador.jazz;
erroresPopularesPorLlamada: [Exception('sin red')], final porEmisora = {'fav-1': PresetEcualizador.rock};
popularesPorLlamada: [ final estado = EstadoRadio(
const [], audio: FakeServicioAudio(ecualizadorActivo: false),
[emisoraDemo(uuid: 'api-ok', nombre: 'API Recuperada')], favoritos: FakeServicioFavoritos(),
], radio: FakeServicioRadio(),
tendenciasPorLlamada: [ servicioEcualizador: FakeServicioEcualizador(
const [], principal: principal,
[emisoraDemo(uuid: 'trend-ok', nombre: 'Trend Recuperada')], porEmisora: porEmisora,
], ),
); resolverArchivoCustom: _archivoCustomVacio,
final estado = EstadoRadio( iniciarAutomaticamente: false,
audio: FakeServicioAudio(), );
favoritos: FakeServicioFavoritos(),
radio: radio,
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar(); await estado.inicializar();
expect(estado.error, 'Sin conexión a la API de radio');
await estado.cargarPopulares(); expect(estado.ecualizadorDisponible, isFalse);
expect(estado.presetEcualizador, principal);
expect(estado.presetPrincipalEcualizador, principal);
expect(
estado.presetEcualizadorPorEmisora('fav-1'),
PresetEcualizador.rock,
);
},
);
expect(estado.error, isNull); test(
expect(estado.populares.map((e) => e.uuid), contains('api-ok')); 'inicializar deja error tras fallo y cargarPopulares manual recupera estaciones',
expect(estado.tendencias.map((e) => e.uuid), contains('trend-ok')); () async {
expect(radio.obtenerPopularesCalls, 2); final radio = FakeServicioRadio(
}); erroresPopularesPorLlamada: [Exception('sin red')],
popularesPorLlamada: [
const [],
[emisoraDemo(uuid: 'api-ok', nombre: 'API Recuperada')],
],
tendenciasPorLlamada: [
const [],
[emisoraDemo(uuid: 'trend-ok', nombre: 'Trend Recuperada')],
],
);
final estado = EstadoRadio(
audio: FakeServicioAudio(),
favoritos: FakeServicioFavoritos(),
radio: radio,
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
test('EQ propio por emisora pisa al principal y puede volver a fallback', await estado.inicializar();
() async { expect(estado.error, 'Sin conexión a la API de radio');
final audio = FakeServicioAudio();
final favoritos = FakeServicioFavoritos();
final emisora = emisoraDemo(uuid: 'fav-1', nombre: 'Favorita');
final principal = PresetEcualizador.pop;
final propio = PresetEcualizador.jazz;
await favoritos.agregar(emisora);
final estado = EstadoRadio( await estado.cargarPopulares();
audio: audio,
favoritos: favoritos,
radio: FakeServicioRadio(populares: [emisora]),
servicioEcualizador: FakeServicioEcualizador(principal: principal),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar(); expect(estado.error, isNull);
await estado.cargarFavoritos(); expect(estado.populares.map((e) => e.uuid), contains('api-ok'));
await estado.guardarPresetEcualizadorPorEmisora(emisora.uuid, propio); expect(estado.tendencias.map((e) => e.uuid), contains('trend-ok'));
expect(radio.obtenerPopularesCalls, 2);
},
);
await estado.reproducir(emisora); test(
expect(estado.presetEcualizador, propio); 'EQ propio por emisora pisa al principal y puede volver a fallback',
expect(audio.presetsAplicados.last, propio); () async {
final audio = FakeServicioAudio();
final favoritos = FakeServicioFavoritos();
final emisora = emisoraDemo(uuid: 'fav-1', nombre: 'Favorita');
final principal = PresetEcualizador.pop;
final propio = PresetEcualizador.jazz;
await favoritos.agregar(emisora);
await estado.deshabilitarPresetEcualizadorPorEmisora(emisora.uuid); final estado = EstadoRadio(
await estado.reproducir(emisora); audio: audio,
expect(estado.presetEcualizador, principal); favoritos: favoritos,
expect(audio.presetsAplicados.last, principal); radio: FakeServicioRadio(populares: [emisora]),
}); servicioEcualizador: FakeServicioEcualizador(principal: principal),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
test('favorita sin EQ propio usa EQ principal desde el primer play', () async { await estado.inicializar();
final audio = FakeServicioAudio(); await estado.cargarFavoritos();
final favoritos = FakeServicioFavoritos(); await estado.guardarPresetEcualizadorPorEmisora(emisora.uuid, propio);
final emisora = emisoraDemo(uuid: 'fav-main', nombre: 'Favorita Main');
final principal = PresetEcualizador.voz;
await favoritos.agregar(emisora);
final estado = EstadoRadio( await estado.reproducir(emisora);
audio: audio, expect(estado.presetEcualizador, propio);
favoritos: favoritos, expect(audio.presetsAplicados.last, propio);
radio: FakeServicioRadio(populares: [emisora]),
servicioEcualizador: FakeServicioEcualizador(principal: principal),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar(); await estado.deshabilitarPresetEcualizadorPorEmisora(emisora.uuid);
await estado.cargarFavoritos(); await estado.reproducir(emisora);
await estado.reproducir(emisora); expect(estado.presetEcualizador, principal);
expect(audio.presetsAplicados.last, principal);
},
);
expect(estado.tienePresetEcualizadorPorEmisora(emisora.uuid), isFalse); test(
expect(estado.presetEcualizador, principal); 'favorita sin EQ propio usa EQ principal desde el primer play',
expect(audio.presetsAplicados.last, principal); () async {
}); final audio = FakeServicioAudio();
final favoritos = FakeServicioFavoritos();
final emisora = emisoraDemo(uuid: 'fav-main', nombre: 'Favorita Main');
final principal = PresetEcualizador.voz;
await favoritos.agregar(emisora);
final estado = EstadoRadio(
audio: audio,
favoritos: favoritos,
radio: FakeServicioRadio(populares: [emisora]),
servicioEcualizador: FakeServicioEcualizador(principal: principal),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar();
await estado.cargarFavoritos();
await estado.reproducir(emisora);
test('notifica cambios de estado de audio para mostrar reproductor al primer play', () async { expect(estado.tienePresetEcualizadorPorEmisora(emisora.uuid), isFalse);
final audio = FakeServicioAudio(); expect(estado.presetEcualizador, principal);
final emisora = emisoraDemo(uuid: 'play-1', nombre: 'Primera'); expect(audio.presetsAplicados.last, principal);
final estado = EstadoRadio( },
audio: audio, );
favoritos: FakeServicioFavoritos(),
radio: FakeServicioRadio(populares: [emisora]),
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
var notificaciones = 0;
estado.addListener(() => notificaciones++);
await estado.inicializar(); test(
final antes = notificaciones; 'permite activar y desactivar el ecualizador de forma persistente',
audio.emitirEstado(EstadoReproduccion.cargando); () async {
await Future<void>.delayed(Duration.zero); final audio = FakeServicioAudio();
final servicioEcualizador = FakeServicioEcualizador();
final estado = EstadoRadio(
audio: audio,
favoritos: FakeServicioFavoritos(),
radio: FakeServicioRadio(),
servicioEcualizador: servicioEcualizador,
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
expect(notificaciones, greaterThan(antes)); await estado.inicializar();
}); expect(estado.ecualizadorActivo, isTrue);
test('reproducir la misma emisora mientras suena fuerza recarga del stream', () async { await estado.cambiarEcualizadorActivo(false);
final audio = FakeServicioAudio(); expect(estado.ecualizadorActivo, isFalse);
final emisora = emisoraDemo(uuid: 'same-1', nombre: 'Misma'); expect(servicioEcualizador.config.activo, isFalse);
final estado = EstadoRadio( expect(audio.cambiosEcualizadorActivo.last, isFalse);
audio: audio,
favoritos: FakeServicioFavoritos(),
radio: FakeServicioRadio(populares: [emisora]),
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar(); await estado.cambiarEcualizadorActivo(true);
await estado.reproducir(emisora); expect(estado.ecualizadorActivo, isTrue);
await estado.reproducir(emisora); expect(servicioEcualizador.config.activo, isTrue);
expect(audio.cambiosEcualizadorActivo.last, isTrue);
},
);
expect(audio.emisorasReproducidas, hasLength(2)); test(
}); 'notifica cambios de estado de audio para mostrar reproductor al primer play',
() async {
final audio = FakeServicioAudio();
final emisora = emisoraDemo(uuid: 'play-1', nombre: 'Primera');
final estado = EstadoRadio(
audio: audio,
favoritos: FakeServicioFavoritos(),
radio: FakeServicioRadio(populares: [emisora]),
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
var notificaciones = 0;
estado.addListener(() => notificaciones++);
test('ignora finalizaciones stale cuando se cambia de emisora rapido', () async { await estado.inicializar();
final audio = _AudioControlado(); final antes = notificaciones;
final radio = FakeServicioRadio(); audio.emitirEstado(EstadoReproduccion.cargando);
final primera = emisoraDemo(uuid: 'slow-1', nombre: 'Lenta'); await Future<void>.delayed(Duration.zero);
final segunda = emisoraDemo(uuid: 'fast-2', nombre: 'Rapida');
final estado = EstadoRadio(
audio: audio,
favoritos: FakeServicioFavoritos(),
radio: radio,
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar(); expect(notificaciones, greaterThan(antes));
await estado.guardarPresetEcualizadorPorEmisora( },
primera.uuid, );
PresetEcualizador.rock,
);
await estado.guardarPresetEcualizadorPorEmisora(
segunda.uuid,
PresetEcualizador.jazz,
);
final primeraFuture = estado.reproducir(primera); test(
final segundaFuture = estado.reproducir(segunda); 'reproducir la misma emisora mientras suena fuerza recarga del stream',
audio.completar(segunda.uuid); () async {
await segundaFuture; final audio = FakeServicioAudio();
audio.completar(primera.uuid); final emisora = emisoraDemo(uuid: 'same-1', nombre: 'Misma');
await primeraFuture; final estado = EstadoRadio(
audio: audio,
favoritos: FakeServicioFavoritos(),
radio: FakeServicioRadio(populares: [emisora]),
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
expect(estado.presetEcualizador, PresetEcualizador.jazz); await estado.inicializar();
expect(radio.ultimoUuidClick, segunda.uuid); await estado.reproducir(emisora);
}); await estado.reproducir(emisora);
expect(audio.emisorasReproducidas, hasLength(2));
},
);
test(
'ignora finalizaciones stale cuando se cambia de emisora rapido',
() async {
final audio = _AudioControlado();
final radio = FakeServicioRadio();
final primera = emisoraDemo(uuid: 'slow-1', nombre: 'Lenta');
final segunda = emisoraDemo(uuid: 'fast-2', nombre: 'Rapida');
final estado = EstadoRadio(
audio: audio,
favoritos: FakeServicioFavoritos(),
radio: radio,
servicioEcualizador: FakeServicioEcualizador(),
resolverArchivoCustom: _archivoCustomVacio,
iniciarAutomaticamente: false,
);
await estado.inicializar();
await estado.guardarPresetEcualizadorPorEmisora(
primera.uuid,
PresetEcualizador.rock,
);
await estado.guardarPresetEcualizadorPorEmisora(
segunda.uuid,
PresetEcualizador.jazz,
);
final primeraFuture = estado.reproducir(primera);
final segundaFuture = estado.reproducir(segunda);
audio.completar(segunda.uuid);
await segundaFuture;
audio.completar(primera.uuid);
await primeraFuture;
expect(estado.presetEcualizador, PresetEcualizador.jazz);
expect(radio.ultimoUuidClick, segunda.uuid);
},
);
test('reordenar favoritos reindexa de forma determinística', () async { test('reordenar favoritos reindexa de forma determinística', () async {
final favoritos = FakeServicioFavoritos(); final favoritos = FakeServicioFavoritos();
@@ -266,8 +312,6 @@ void main() {
expect(lista.map((e) => e.orden).toList(), equals([0, 1, 2])); expect(lista.map((e) => e.orden).toList(), equals([0, 1, 2]));
}); });
test('cargarMasBusqueda pagina resultados y acota memoria', () async { test('cargarMasBusqueda pagina resultados y acota memoria', () async {
final emisoras = List.generate( final emisoras = List.generate(
70, 70,
@@ -358,6 +402,8 @@ Future<File> _archivoCustomVacio() async => _crearArchivoCustom(const []);
Future<File> _crearArchivoCustom(List<dynamic> emisoras) async { Future<File> _crearArchivoCustom(List<dynamic> emisoras) async {
final dir = await Directory.systemTemp.createTemp('pluriwave-test-'); final dir = await Directory.systemTemp.createTemp('pluriwave-test-');
final archivo = File('${dir.path}/emisoras_custom.json'); final archivo = File('${dir.path}/emisoras_custom.json');
await archivo.writeAsString(jsonEncode(emisoras.map((e) => e.toMap()).toList())); await archivo.writeAsString(
jsonEncode(emisoras.map((e) => e.toMap()).toList()),
);
return archivo; return archivo;
} }
+43 -17
View File
@@ -16,6 +16,7 @@ class FakeServicioAudio extends ServicioAudio {
final _estadoController = StreamController<EstadoReproduccion>.broadcast(); final _estadoController = StreamController<EstadoReproduccion>.broadcast();
final List<PresetEcualizador> presetsAplicados = []; final List<PresetEcualizador> presetsAplicados = [];
final List<Emisora> emisorasReproducidas = []; final List<Emisora> emisorasReproducidas = [];
final List<bool> cambiosEcualizadorActivo = [];
Emisora? _emisoraActual; Emisora? _emisoraActual;
EstadoReproduccion _estadoActual = EstadoReproduccion.detenido; EstadoReproduccion _estadoActual = EstadoReproduccion.detenido;
@@ -57,6 +58,11 @@ class FakeServicioAudio extends ServicioAudio {
@override @override
Future<void> setBanda(int index, double db) async {} Future<void> setBanda(int index, double db) async {}
@override
Future<void> setEcualizadorActivo(bool activo) async {
cambiosEcualizadorActivo.add(activo);
}
@override @override
Future<void> dispose() async { Future<void> dispose() async {
await _estadoController.close(); await _estadoController.close();
@@ -122,13 +128,13 @@ class FakeServicioRadio extends ServicioRadio {
List<List<Emisora>>? tendenciasPorLlamada, List<List<Emisora>>? tendenciasPorLlamada,
List<Object>? erroresPopularesPorLlamada, List<Object>? erroresPopularesPorLlamada,
List<Object>? erroresTendenciasPorLlamada, List<Object>? erroresTendenciasPorLlamada,
}) : _populares = populares ?? [], }) : _populares = populares ?? [],
_tendencias = tendencias ?? [], _tendencias = tendencias ?? [],
_busqueda = busqueda ?? [], _busqueda = busqueda ?? [],
_popularesPorLlamada = popularesPorLlamada ?? const [], _popularesPorLlamada = popularesPorLlamada ?? const [],
_tendenciasPorLlamada = tendenciasPorLlamada ?? const [], _tendenciasPorLlamada = tendenciasPorLlamada ?? const [],
_erroresPopularesPorLlamada = erroresPopularesPorLlamada ?? const [], _erroresPopularesPorLlamada = erroresPopularesPorLlamada ?? const [],
_erroresTendenciasPorLlamada = erroresTendenciasPorLlamada ?? const []; _erroresTendenciasPorLlamada = erroresTendenciasPorLlamada ?? const [];
final List<Emisora> _populares; final List<Emisora> _populares;
final List<Emisora> _tendencias; final List<Emisora> _tendencias;
@@ -147,14 +153,18 @@ class FakeServicioRadio extends ServicioRadio {
error is Exception ? error : Exception(error.toString()); error is Exception ? error : Exception(error.toString());
@override @override
Future<List<Emisora>> obtenerPopulares({int limit = 30, int offset = 0}) async { Future<List<Emisora>> obtenerPopulares({
int limit = 30,
int offset = 0,
}) async {
final llamada = obtenerPopularesCalls++; final llamada = obtenerPopularesCalls++;
if (llamada < _erroresPopularesPorLlamada.length) { if (llamada < _erroresPopularesPorLlamada.length) {
throw _normalizarError(_erroresPopularesPorLlamada[llamada]); throw _normalizarError(_erroresPopularesPorLlamada[llamada]);
} }
final data = llamada < _popularesPorLlamada.length final data =
? _popularesPorLlamada[llamada] llamada < _popularesPorLlamada.length
: _populares; ? _popularesPorLlamada[llamada]
: _populares;
return data.take(limit).toList(); return data.take(limit).toList();
} }
@@ -164,9 +174,10 @@ class FakeServicioRadio extends ServicioRadio {
if (llamada < _erroresTendenciasPorLlamada.length) { if (llamada < _erroresTendenciasPorLlamada.length) {
throw _normalizarError(_erroresTendenciasPorLlamada[llamada]); throw _normalizarError(_erroresTendenciasPorLlamada[llamada]);
} }
final data = llamada < _tendenciasPorLlamada.length final data =
? _tendenciasPorLlamada[llamada] llamada < _tendenciasPorLlamada.length
: _tendencias; ? _tendenciasPorLlamada[llamada]
: _tendencias;
return data.take(limit).toList(); return data.take(limit).toList();
} }
@@ -193,12 +204,15 @@ class FakeServicioEcualizador extends ServicioEcualizador {
FakeServicioEcualizador({ FakeServicioEcualizador({
PresetEcualizador? principal, PresetEcualizador? principal,
Map<String, PresetEcualizador>? porEmisora, Map<String, PresetEcualizador>? porEmisora,
bool activo = true,
}) : _config = ConfiguracionEcualizador( }) : _config = ConfiguracionEcualizador(
principal: principal ?? PresetEcualizador.flat, principal: principal ?? PresetEcualizador.flat,
porEmisora: porEmisora ?? {}, porEmisora: porEmisora ?? {},
); activo: activo,
);
ConfiguracionEcualizador _config; ConfiguracionEcualizador _config;
ConfiguracionEcualizador get config => _config;
@override @override
Future<ConfiguracionEcualizador> cargar() async => _config; Future<ConfiguracionEcualizador> cargar() async => _config;
@@ -208,6 +222,16 @@ class FakeServicioEcualizador extends ServicioEcualizador {
_config = ConfiguracionEcualizador( _config = ConfiguracionEcualizador(
principal: preset, principal: preset,
porEmisora: _config.porEmisora, porEmisora: _config.porEmisora,
activo: _config.activo,
);
}
@override
Future<void> guardarActivo(bool activo) async {
_config = ConfiguracionEcualizador(
principal: _config.principal,
porEmisora: _config.porEmisora,
activo: activo,
); );
} }
@@ -218,6 +242,7 @@ class FakeServicioEcualizador extends ServicioEcualizador {
_config = ConfiguracionEcualizador( _config = ConfiguracionEcualizador(
principal: _config.principal, principal: _config.principal,
porEmisora: mapa, porEmisora: mapa,
activo: _config.activo,
); );
} }
@@ -228,6 +253,7 @@ class FakeServicioEcualizador extends ServicioEcualizador {
_config = ConfiguracionEcualizador( _config = ConfiguracionEcualizador(
principal: _config.principal, principal: _config.principal,
porEmisora: mapa, porEmisora: mapa,
activo: _config.activo,
); );
} }
} }