import 'dart:io'; import 'package:file_picker/file_picker.dart'; import 'package:spotiflac_android/models/track.dart'; import 'package:spotiflac_android/services/platform_bridge.dart'; import 'package:spotiflac_android/utils/logger.dart'; class CsvImportService { static final _log = AppLogger('CsvImportService'); static final RegExp _lineSplitPattern = RegExp(r'\r\n|\r|\n'); static Future> pickAndParseCsv({ void Function(int current, int total)? onProgress, }) async { try { final FilePickerResult? result = await FilePicker.platform.pickFiles( type: FileType.custom, allowedExtensions: ['csv'], ); if (result != null && result.files.single.path != null) { final file = File(result.files.single.path!); final content = await file.readAsString(); final tracks = _parseCsv(content); if (tracks.isNotEmpty) { return await _enrichTracksMetadata(tracks, onProgress: onProgress); } return tracks; } } catch (e) { _log.e('Error picking/parsing CSV: $e'); } return []; } static Future> _enrichTracksMetadata( List tracks, { void Function(int current, int total)? onProgress, }) async { _log.i('Enriching metadata for ${tracks.length} tracks from Deezer...'); final enrichedTracks = []; for (int i = 0; i < tracks.length; i++) { final track = tracks[i]; onProgress?.call(i + 1, tracks.length); if (track.coverUrl == null || track.duration == 0) { Map? trackData; if (track.isrc != null && track.isrc!.isNotEmpty) { try { trackData = await PlatformBridge.searchDeezerByISRC(track.isrc!); _log.d('ISRC enrichment success for ${track.name}'); } catch (e) { _log.w( 'ISRC search failed for ${track.name}, trying text search...', ); } } if (trackData == null) { try { final query = '${track.artistName} ${track.name}'; final searchResult = await PlatformBridge.customSearchWithExtension( 'deezer', query, options: {'filter': 'track', 'limit': 5}, ); if (searchResult.isNotEmpty) { for (final resultMap in searchResult) { final resultName = (resultMap['name'] as String?)?.toLowerCase() ?? ''; final trackNameLower = track.name.toLowerCase(); if (resultName.contains(trackNameLower) || trackNameLower.contains(resultName)) { trackData = resultMap; _log.d('Text search match for ${track.name}: $resultName'); break; } } if (trackData == null) { trackData = searchResult.first; _log.d('Using first search result for ${track.name}'); } } } catch (e) { _log.w('Text search also failed for ${track.name}: $e'); } } if (trackData != null) { final coverUrl = trackData['images'] as String?; final durationMs = trackData['duration_ms'] as int? ?? 0; final deezerIdRaw = trackData['spotify_id'] as String?; enrichedTracks.add( Track( id: deezerIdRaw ?? track.id, name: trackData['name'] as String? ?? track.name, artistName: trackData['artists'] as String? ?? track.artistName, albumName: trackData['album_name'] as String? ?? track.albumName, albumArtist: trackData['album_artist'] as String?, artistId: trackData['artist_id']?.toString(), albumId: trackData['album_id']?.toString(), coverUrl: coverUrl ?? track.coverUrl, isrc: trackData['isrc'] as String? ?? track.isrc, duration: durationMs > 0 ? durationMs ~/ 1000 : track.duration, trackNumber: trackData['track_number'] as int? ?? track.trackNumber, discNumber: trackData['disc_number'] as int? ?? track.discNumber, releaseDate: trackData['release_date'] as String? ?? track.releaseDate, ), ); _log.d( 'Enriched: ${track.name} - cover: ${coverUrl != null}, duration: ${durationMs ~/ 1000}s', ); if (i < tracks.length - 1) { await Future.delayed(const Duration(milliseconds: 100)); } continue; } } enrichedTracks.add(track); } _log.i('Enrichment complete: ${enrichedTracks.length} tracks'); return enrichedTracks; } static List _parseCsv(String content) { final List tracks = []; final lines = content.split(_lineSplitPattern); if (lines.isEmpty) return tracks; int startIdx = 0; while (startIdx < lines.length && lines[startIdx].trim().isEmpty) { startIdx++; } if (startIdx >= lines.length) return tracks; final headers = _parseLine(lines[startIdx]); final colMap = {}; for (int i = 0; i < headers.length; i++) { String h = _cleanValue(headers[i]).toLowerCase(); colMap[h] = i; } _log.d('CSV Headers: ${colMap.keys.toList()}'); for (int i = startIdx + 1; i < lines.length; i++) { final line = lines[i].trim(); if (line.isEmpty) continue; final values = _parseLine(line); String? getVal(List keys) { return _getValue(values, colMap, keys); } String? trackName = getVal(['track name', 'track', 'name', 'title']); String? artistName = getVal([ 'artist name(s)', 'artist name', 'artist', 'artists', ]); String? albumName = getVal(['album name', 'album']); String? isrc = getVal(['isrc']); String? spotifyId = getVal([ 'track uri', 'spotify - id', 'spotify id', 'spotify_id', 'id', 'uri', ]); if (spotifyId != null && spotifyId.startsWith('spotify:track:')) { spotifyId = spotifyId.replaceAll('spotify:track:', ''); } if ((trackName != null && trackName.isNotEmpty && artistName != null) || (spotifyId != null && spotifyId.isNotEmpty)) { tracks.add( Track( id: spotifyId ?? 'csv_${DateTime.now().millisecondsSinceEpoch}_$i', name: trackName ?? 'Unknown Track', artistName: artistName ?? 'Unknown Artist', albumName: albumName ?? 'Unknown Album', isrc: isrc, duration: 0, coverUrl: null, ), ); } } _log.i('Parsed ${tracks.length} tracks from CSV'); return tracks; } static String? _getValue( List values, Map colMap, List possibleKeys, ) { for (final key in possibleKeys) { if (colMap.containsKey(key)) { final index = colMap[key]!; if (index < values.length) { return _cleanValue(values[index]); } } } return null; } static String _cleanValue(String val) { val = val.trim(); if (val.startsWith('"') && val.endsWith('"') && val.length >= 2) { val = val.substring(1, val.length - 1); } val = val.replaceAll('""', '"'); return val; } static List _parseLine(String line) { final List result = []; bool inQuote = false; var buffer = StringBuffer(); for (int i = 0; i < line.length; i++) { final char = line[i]; if (char == '"') { if (inQuote && i + 1 < line.length && line[i + 1] == '"') { buffer.write('"'); i++; } else { inQuote = !inQuote; } continue; } if (char == ',' && !inQuote) { result.add(buffer.toString()); buffer = StringBuffer(); continue; } buffer.write(char); } result.add(buffer.toString()); return result; } }