First implement of #40
This commit is contained in:
parent
32c9f24961
commit
de6caf2949
|
@ -0,0 +1,20 @@
|
|||
import 'package:equatable/equatable.dart';
|
||||
import 'package:youtube_explode_dart/src/videos/video_id.dart';
|
||||
|
||||
/// Metadata related to a search query result (playlist)
|
||||
class ChannelVideo with EquatableMixin {
|
||||
/// Video ID.
|
||||
final VideoId videoId;
|
||||
|
||||
/// Video title.
|
||||
final String videoTitle;
|
||||
|
||||
/// Initialize an instance of [ChannelVideo]
|
||||
ChannelVideo(this.videoId, this.videoTitle);
|
||||
|
||||
@override
|
||||
String toString() => '(ChannelVideo) $videoId ($videoTitle)';
|
||||
|
||||
@override
|
||||
List<Object> get props => [videoId];
|
||||
}
|
|
@ -3,4 +3,5 @@ library youtube_explode.channels;
|
|||
export 'channel.dart';
|
||||
export 'channel_client.dart';
|
||||
export 'channel_id.dart';
|
||||
export 'channel_video.dart';
|
||||
export 'username.dart';
|
||||
|
|
|
@ -0,0 +1,140 @@
|
|||
import 'dart:convert';
|
||||
|
||||
import 'package:html/dom.dart';
|
||||
import 'package:html/parser.dart' as parser;
|
||||
|
||||
import '../../channels/channel_video.dart';
|
||||
import '../../extensions/helpers_extension.dart';
|
||||
import '../../retry.dart';
|
||||
import '../../videos/videos.dart';
|
||||
import '../youtube_http_client.dart';
|
||||
|
||||
class ChannelWatchPage {
|
||||
final String channelId;
|
||||
final Document _root;
|
||||
|
||||
_InitialData _initialData;
|
||||
|
||||
_InitialData get initialData =>
|
||||
_initialData ??= _InitialData(json.decode(_matchJson(_extractJson(
|
||||
_root
|
||||
.querySelectorAll('script')
|
||||
.map((e) => e.text)
|
||||
.toList()
|
||||
.firstWhere((e) => e.contains('window["ytInitialData"] =')),
|
||||
'window["ytInitialData"] ='))));
|
||||
|
||||
String _extractJson(String html, String separator) {
|
||||
return _matchJson(
|
||||
html.substring(html.indexOf(separator) + separator.length));
|
||||
}
|
||||
|
||||
String _matchJson(String str) {
|
||||
var bracketCount = 0;
|
||||
int lastI;
|
||||
for (var i = 0; i < str.length; i++) {
|
||||
lastI = i;
|
||||
if (str[i] == '{') {
|
||||
bracketCount++;
|
||||
} else if (str[i] == '}') {
|
||||
bracketCount--;
|
||||
} else if (str[i] == ';') {
|
||||
if (bracketCount == 0) {
|
||||
return str.substring(0, i);
|
||||
}
|
||||
}
|
||||
}
|
||||
return str.substring(0, lastI + 1);
|
||||
}
|
||||
|
||||
ChannelWatchPage(this._root, this.channelId);
|
||||
|
||||
Future<ChannelWatchPage> nextPage() {}
|
||||
|
||||
static Future<ChannelWatchPage> get(
|
||||
YoutubeHttpClient httpClient, String channelId) {
|
||||
var url =
|
||||
'https://www.youtube.com/channel/$channelId/videos?view=0&sort=dd&flow=grid';
|
||||
return retry(() async {
|
||||
var raw = await httpClient.getString(url);
|
||||
return ChannelWatchPage.parse(raw, channelId);
|
||||
});
|
||||
}
|
||||
|
||||
ChannelWatchPage.parse(String raw, this.channelId)
|
||||
: _root = parser.parse(raw);
|
||||
}
|
||||
|
||||
class _InitialData {
|
||||
// Json parsed map
|
||||
final Map<String, dynamic> _root;
|
||||
|
||||
_InitialData(this._root);
|
||||
|
||||
/* Cache results */
|
||||
|
||||
List<ChannelVideo> _uploads;
|
||||
String _continuation;
|
||||
String _clickTrackingParams;
|
||||
|
||||
List<Map<String, dynamic>> getContentContext(Map<String, dynamic> root) {
|
||||
if (root['contents'] != null) {
|
||||
return (_root['contents']['twoColumnBrowseResultsRenderer']['tabs']
|
||||
as List<dynamic>)
|
||||
.map((e) => e['tabRenderer'])
|
||||
.firstWhere((e) => e['selected'] == true)['content']
|
||||
['sectionListRenderer']['contents']
|
||||
.first['itemSectionRenderer']['contents']
|
||||
.first['gridRenderer']['items']
|
||||
.cast<Map<String, dynamic>>();
|
||||
;
|
||||
}
|
||||
if (root['response'] != null) {
|
||||
return _root['response']['continuationContents']['gridContinuation']
|
||||
['items']
|
||||
.cast<Map<String, dynamic>>();
|
||||
}
|
||||
throw Exception('Couldn\'t find the content data');
|
||||
}
|
||||
|
||||
Map<String, dynamic> getContinuationContext(Map<String, dynamic> root) {
|
||||
if (_root['contents'] != null) {
|
||||
return (_root['contents']['twoColumnBrowseResultsRenderer']['tabs']
|
||||
as List<dynamic>)
|
||||
?.map((e) => e['tabRenderer'])
|
||||
?.firstWhere((e) => e['selected'] == true)['content']
|
||||
['sectionListRenderer']['contents']
|
||||
?.first['itemSectionRenderer']['contents']
|
||||
?.first['gridRenderer']['continuations']
|
||||
?.first['nextContinuationData']
|
||||
?.cast<String, dynamic>();
|
||||
}
|
||||
if (_root['response'] != null) {
|
||||
return _root['response']['continuationContents']['gridContinuation']
|
||||
['continuations']
|
||||
?.first
|
||||
?.cast<String, dynamic>();
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
List<dynamic> get uploads => _uploads ??= getContentContext(_root)
|
||||
?.map(_parseContent)
|
||||
?.where((e) => e != null)
|
||||
?.toList();
|
||||
|
||||
String get continuation => _continuation ??=
|
||||
getContinuationContext(_root)?.getValue('continuation') ?? '';
|
||||
|
||||
String get clickTrackingParams => _clickTrackingParams ??=
|
||||
getContinuationContext(_root)?.getValue('clickTrackingParams') ?? '';
|
||||
|
||||
dynamic _parseContent(content) {
|
||||
if (content == null || content['gridVideoRenderer'] == null) {
|
||||
return null;
|
||||
}
|
||||
var video = content['gridVideoRenderer'] as Map<String, dynamic>;
|
||||
return ChannelVideo(
|
||||
VideoId(video['videoId']), video['title']['simpleText']);
|
||||
}
|
||||
}
|
|
@ -1,7 +1,9 @@
|
|||
import 'package:equatable/equatable.dart';
|
||||
|
||||
import '../playlists/playlist_id.dart';
|
||||
|
||||
/// Metadata related to a search query result (playlist)
|
||||
class SearchPlaylist {
|
||||
class SearchPlaylist with EquatableMixin {
|
||||
/// PlaylistId.
|
||||
final PlaylistId playlistId;
|
||||
|
||||
|
@ -16,4 +18,7 @@ class SearchPlaylist {
|
|||
|
||||
@override
|
||||
String toString() => '(Playlist) $playlistTitle ($playlistId)';
|
||||
|
||||
@override
|
||||
List<Object> get props => [playlistId];
|
||||
}
|
||||
|
|
|
@ -20,8 +20,8 @@ class SearchVideo {
|
|||
/// Video View Count
|
||||
final int videoViewCount;
|
||||
|
||||
/// Initialize a [RelatedQuery] instance.
|
||||
SearchVideo(this.videoId, this.videoTitle, this.videoAuthor,
|
||||
/// Initialize a [SearchVideo] instance.
|
||||
const SearchVideo(this.videoId, this.videoTitle, this.videoAuthor,
|
||||
this.videoDescriptionSnippet, this.videoDuration, this.videoViewCount);
|
||||
|
||||
@override
|
||||
|
|
Loading…
Reference in New Issue