import 'dart:core'; import 'dart:developer' as developer; import 'dart:io'; import 'dart:isolate'; import 'dart:math' as math; import 'package:color_thief_flutter/color_thief_flutter.dart'; import 'package:dio/dio.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_isolate/flutter_isolate.dart'; import 'package:image/image.dart' as img; import 'package:path_provider/path_provider.dart'; import 'package:webfeed/webfeed.dart'; import 'package:uuid/uuid.dart'; import 'package:equatable/equatable.dart'; import '../local_storage/key_value_storage.dart'; import '../local_storage/sqflite_localpodcast.dart'; import '../type/fireside_data.dart'; import '../type/podcastlocal.dart'; class GroupEntity { final String name; final String id; final String color; final List podcastList; GroupEntity(this.name, this.id, this.color, this.podcastList); Map toJson() { return {'name': name, 'id': id, 'color': color, 'podcastList': podcastList}; } static GroupEntity fromJson(Map json) { var list = List.from(json['podcastList']); return GroupEntity(json['name'] as String, json['id'] as String, json['color'] as String, list); } } class PodcastGroup extends Equatable { /// Group name. final String name; final String id; /// Group theme color, not used. final String color; /// Id lists of podcasts in group. List _podcastList; List get podcastList => _podcastList; set podcastList(list) { _podcastList = list; } PodcastGroup(this.name, {this.color = '#000000', String id, List podcastList}) : id = id ?? Uuid().v4(), _podcastList = podcastList ?? []; Future getPodcasts() async { var dbHelper = DBHelper(); if (_podcastList != []) { try { _podcasts = await dbHelper.getPodcastLocal(_podcastList); } catch (e) { await Future.delayed(Duration(milliseconds: 200)); try { _podcasts = await dbHelper.getPodcastLocal(_podcastList); } catch (e) { developer.log(e.toString()); } } } } Color getColor() { if (color != '#000000') { var colorInt = int.parse('FF${color.toUpperCase()}', radix: 16); return Color(colorInt).withOpacity(1.0); } else { return Colors.blue[400]; } } ///Podcast in group. List _podcasts; List get podcasts => _podcasts; ///Ordered podcast list. List _orderedPodcasts; List get orderedPodcasts => _orderedPodcasts; set orderedPodcasts(list) => _orderedPodcasts = list; GroupEntity toEntity() { return GroupEntity(name, id, color, podcastList); } static PodcastGroup fromEntity(GroupEntity entity) { return PodcastGroup( entity.name, id: entity.id, color: entity.color, podcastList: entity.podcastList, ); } @override List get props => [id, name]; } enum SubscribeState { none, start, subscribe, fetch, stop, exist, error } class SubscribeItem { ///Rss url. String url; ///Rss title. String title; /// Subscribe status. SubscribeState subscribeState; /// Podcast id. String id; ///Avatar image link. String imgUrl; ///Podcast group, default Home. String group; SubscribeItem(this.url, this.title, {this.subscribeState = SubscribeState.none, this.id = '', this.imgUrl = '', this.group = ''}); } class GroupList extends ChangeNotifier { /// List of all gourps. final List _groups = []; List get groups => _groups; final DBHelper _dbHelper = DBHelper(); /// Groups save in shared_prefrences. final KeyValueStorage _groupStorage = KeyValueStorage(groupsKey); //GroupList({List groups}) : _groups = groups ?? []; /// Default false, true during loading groups from storage. bool _isLoading = false; bool get isLoading => _isLoading; /// Svae ordered gourps info before saved. final List _orderChanged = []; List get orderChanged => _orderChanged; /// Subscribe worker isolate FlutterIsolate subIsolate; ReceivePort receivePort; SendPort subSendPort; /// Current subsribe item from isolate. SubscribeItem _currentSubscribeItem = SubscribeItem('', ''); SubscribeItem get currentSubscribeItem => _currentSubscribeItem; bool _created = false; /// Default false, true if subscribe isolate is created. bool get created => _created; /// Add subsribe item SubscribeItem _subscribeItem; setSubscribeItem(SubscribeItem item) async { _subscribeItem = item; await _start(); } _setCurrentSubscribeItem(SubscribeItem item) { _currentSubscribeItem = item; notifyListeners(); } Future _start() async { if (_created == false) { await _createIsolate(); _created = true; listen(); } else { subSendPort.send([ _subscribeItem.url, _subscribeItem.title, _subscribeItem.imgUrl, _subscribeItem.group ]); } } Future _createIsolate() async { receivePort = ReceivePort(); subIsolate = await FlutterIsolate.spawn(subIsolateEntryPoint, receivePort.sendPort); } /// Isolate listener to get subscrribe status. void listen() { receivePort.distinct().listen((message) { if (message is SendPort) { subSendPort = message; subSendPort.send([ _subscribeItem.url, _subscribeItem.title, _subscribeItem.imgUrl, _subscribeItem.group ]); } else if (message is List) { _setCurrentSubscribeItem(SubscribeItem( message[1], message[0], subscribeState: SubscribeState.values[message[2]], )); if (message.length == 5) { _subscribeNewPodcast(id: message[3], groupName: message[4]); } } else if (message is String && message == "done") { subIsolate.kill(); subIsolate = null; _currentSubscribeItem = SubscribeItem('', ''); _created = false; notifyListeners(); } }); } void addToOrderChanged(PodcastGroup group) { _orderChanged.add(group); notifyListeners(); } void drlFromOrderChanged(String name) { _orderChanged.removeWhere((group) => group.name == name); notifyListeners(); } Future clearOrderChanged() async { if (_orderChanged.length > 0) { for (var group in _orderChanged) { await group.getPodcasts(); } _orderChanged.clear(); // notifyListeners(); } } @override void addListener(VoidCallback listener) { loadGroups().then((value) => super.addListener(listener)); } @override void dispose() { subIsolate?.kill(); subIsolate = null; super.dispose(); } /// Load groups from storage at start. Future loadGroups() async { _isLoading = true; notifyListeners(); _groupStorage.getGroups().then((loadgroups) async { _groups.addAll(loadgroups.map(PodcastGroup.fromEntity)); for (var group in _groups) { await group.getPodcasts(); } _isLoading = false; notifyListeners(); }); } /// Update podcasts of each group Future updateGroups() async { for (var group in _groups) { await group.getPodcasts(); } notifyListeners(); } /// Add new group. Future addGroup(PodcastGroup podcastGroup) async { _isLoading = true; _groups.add(podcastGroup); await _saveGroup(); _isLoading = false; notifyListeners(); } /// Remove group. Future delGroup(PodcastGroup podcastGroup) async { _isLoading = true; for (var podcast in podcastGroup.podcastList) { if (!_groups.first.podcastList.contains(podcast)) { _groups[0].podcastList.insert(0, podcast); } } await _saveGroup(); _groups.remove(podcastGroup); await _groups[0].getPodcasts(); _isLoading = false; notifyListeners(); } Future updateGroup(PodcastGroup podcastGroup) async { var oldGroup = _groups.firstWhere((it) => it.id == podcastGroup.id); var index = _groups.indexOf(oldGroup); _groups.replaceRange(index, index + 1, [podcastGroup]); await podcastGroup.getPodcasts(); notifyListeners(); _saveGroup(); } Future _saveGroup() async { await _groupStorage.saveGroup(_groups.map((it) => it.toEntity()).toList()); } /// Subscribe podcast from search result. Future subscribe(PodcastLocal podcastLocal) async { _groups[0].podcastList.insert(0, podcastLocal.id); await _saveGroup(); await _dbHelper.savePodcastLocal(podcastLocal); await _groups[0].getPodcasts(); notifyListeners(); } Future updatePodcast(String id) async { var counts = await _dbHelper.getPodcastCounts(id); for (var group in _groups) { if (group.podcastList.contains(id)) { group.podcasts.firstWhere((podcast) => podcast.id == id) ..episodeCount = counts; notifyListeners(); } } } /// Subscribe podcast from OMPL. Future _subscribeNewPodcast( {String id, String groupName = 'Home'}) async { //List groupNames = _groups.map((e) => e.name).toList(); for (var group in _groups) { if (group.name == groupName) { if (group.podcastList.contains(id)) { return true; } else { _isLoading = true; notifyListeners(); group.podcastList.insert(0, id); await _saveGroup(); await group.getPodcasts(); _isLoading = false; notifyListeners(); return true; } } } _isLoading = true; notifyListeners(); _groups.add(PodcastGroup(groupName, podcastList: [id])); //_groups.last.podcastList.insert(0, id); await _saveGroup(); await _groups.last.getPodcasts(); _isLoading = false; notifyListeners(); return true; } List getPodcastGroup(String id) { var result = []; for (var group in _groups) { if (group.podcastList.contains(id)) { result.add(group); } } return result; } //Change podcast groups Future changeGroup(String id, List list) async { _isLoading = true; notifyListeners(); for (var group in getPodcastGroup(id)) { if (list.contains(group)) { list.remove(group); } else { group.podcastList.remove(id); } } for (var s in list) { s.podcastList.insert(0, id); } await _saveGroup(); for (var group in _groups) { await group.getPodcasts(); } _isLoading = false; notifyListeners(); } /// Unsubscribe podcast Future removePodcast(String id) async { _isLoading = true; notifyListeners(); for (var group in _groups) { group.podcastList.remove(id); } await _saveGroup(); await _dbHelper.delPodcastLocal(id); for (var group in _groups) { await group.getPodcasts(); } _isLoading = false; notifyListeners(); } saveOrder(PodcastGroup group) async { group.podcastList = group.orderedPodcasts.map((e) => e.id).toList(); await _saveGroup(); await group.getPodcasts(); notifyListeners(); } } Future subIsolateEntryPoint(SendPort sendPort) async { var items = []; var _running = false; final listColor = [ '388E3C', '1976D2', 'D32F2F', '00796B', ]; var subReceivePort = ReceivePort(); sendPort.send(subReceivePort.sendPort); Future _getColor(File file) async { final imageProvider = FileImage(file); var colorImage = await getImageFromProvider(imageProvider); var color = await getColorFromImage(colorImage); var primaryColor = color.toString(); return primaryColor; } Future _subscribe(SubscribeItem item) async { var dbHelper = DBHelper(); var rss = item.url; sendPort.send([item.title, item.url, 1]); var options = BaseOptions( connectTimeout: 30000, receiveTimeout: 90000, ); try { var response = await Dio(options).get(rss); RssFeed p; try { p = RssFeed.parse(response.data); } catch (e) { sendPort.send([item.title, item.url, 6]); await Future.delayed(Duration(seconds: 2)); sendPort.send([item.title, item.url, 4]); items.removeWhere((element) => element.url == item.url); if (items.isNotEmpty) { await _subscribe(items.first); } else { sendPort.send("done"); } } var dir = await getApplicationDocumentsDirectory(); var realUrl = response.redirects.isEmpty ? rss : response.realUri.toString(); var checkUrl = await dbHelper.checkPodcast(realUrl); /// If url not existe in database. if (checkUrl == '') { img.Image thumbnail; String imageUrl; try { var imageResponse = await Dio().get>(p.itunes.image.href, options: Options( responseType: ResponseType.bytes, receiveTimeout: 90000, )); imageUrl = p.itunes.image.href; var image = img.decodeImage(imageResponse.data); thumbnail = img.copyResize(image, width: 300); } catch (e) { try { var imageResponse = await Dio().get>(item.imgUrl, options: Options( responseType: ResponseType.bytes, receiveTimeout: 90000, )); imageUrl = item.imgUrl; var image = img.decodeImage(imageResponse.data); thumbnail = img.copyResize(image, width: 300); } catch (e) { developer.log(e.toString(), name: 'Download image error'); try { var index = math.Random().nextInt(3); var imageResponse = await Dio().get>( "https://ui-avatars.com/api/?size=300&background=" "${listColor[index]}&color=fff&name=${item.title}&length=2&bold=true", options: Options(responseType: ResponseType.bytes)); imageUrl = "https://ui-avatars.com/api/?size=300&background=" "${listColor[index]}&color=fff&name=${item.title}&length=2&bold=true"; thumbnail = img.decodeImage(imageResponse.data); } catch (e) { developer.log(e.toString(), name: 'Donwload image error'); sendPort.send([item.title, item.url, 6]); await Future.delayed(Duration(seconds: 2)); sendPort.send([item.title, item.url, 4]); items.removeWhere((element) => element.url == item.url); if (items.length > 0) { await _subscribe(items.first); } else { sendPort.send("done"); } } } } var uuid = Uuid().v4(); File("${dir.path}/$uuid.png") ..writeAsBytesSync(img.encodePng(thumbnail)); var imagePath = "${dir.path}/$uuid.png"; var primaryColor = await _getColor(File("${dir.path}/$uuid.png")); var author = p.itunes.author ?? p.author ?? ''; var provider = p.generator ?? ''; var link = p.link ?? ''; var podcastLocal = PodcastLocal(p.title, imageUrl, realUrl, primaryColor, author, uuid, imagePath, provider, link, description: p.description); await dbHelper.savePodcastLocal(podcastLocal); sendPort.send([item.title, item.url, 2, uuid, item.group]); if (provider.contains('fireside')) { var data = FiresideData(uuid, link); try { await data.fatchData(); } catch (e) { developer.log(e.toString(), name: 'Fatch fireside data error'); } } await dbHelper.savePodcastRss(p, uuid); sendPort.send([item.title, item.url, 3, uuid]); await Future.delayed(Duration(seconds: 2)); sendPort.send([item.title, item.url, 4]); items.removeAt(0); if (items.length > 0) { await _subscribe(items.first); } else { sendPort.send("done"); } } else { sendPort.send([item.title, realUrl, 5, checkUrl, item.group]); await Future.delayed(Duration(seconds: 2)); sendPort.send([item.title, item.url, 4]); items.removeAt(0); if (items.length > 0) { await _subscribe(items.first); } else { sendPort.send("done"); } } } catch (e) { developer.log('$e confirm'); sendPort.send([item.title, item.url, 6]); await Future.delayed(Duration(seconds: 2)); sendPort.send([item.title, item.url, 4]); items.removeWhere((element) => element.url == item.url); if (items.length > 0) { await _subscribe(items.first); } else { sendPort.send("done"); } } } subReceivePort.distinct().listen((message) { if (message is List) { items.add(SubscribeItem(message[0], message[1], imgUrl: message[2], group: message[3])); if (!_running) { _subscribe(items.first); _running = true; } } }); }