// ==UserScript==
// @name South Plus 视频链接播放器
// @description 自动播放视频链接
// @version 0.5.0.beta.2
// @author 大碗宽面wtw
// @homepage https://greasyfork.org/zh-CN/scripts/457637
// @supportURL https://github.com/bigbowl-wtw/south-plus-video-player/issues
// @match *://*.blue-plus.net/read.php*
// @match *://*.east-plus.net/read.php*
// @match *://*.imoutolove.me/read.php*
// @match *://*.level-plus.net/read.php*
// @match *://*.north-plus.net/read.php*
// @match *://*.snow-plus.net/read.php*
// @match *://*.soul-plus.net/read.php*
// @match *://*.south-plus.net/read.php*
// @match *://*.south-plus.org/read.php*
// @match *://*.spring-plus.net/read.php*
// @match *://*.summer-plus.net/read.php*
// @match *://*.white-plus.net/read.php*
// @match *://blue-plus.net/read.php*
// @match *://east-plus.net/read.php*
// @match *://imoutolove.me/read.php*
// @match *://level-plus.net/read.php*
// @match *://north-plus.net/read.php*
// @match *://snow-plus.net/read.php*
// @match *://soul-plus.net/read.php*
// @match *://south-plus.net/read.php*
// @match *://south-plus.org/read.php*
// @match *://spring-plus.net/read.php*
// @match *://summer-plus.net/read.php*
// @match *://white-plus.net/read.php*
// @connect bilibili.com
// @connect b23.tv
// @connect youtube.com
// @connect youtu.be
// @connect twitter.com
// @connect twimg.com
// @connect 91porny.com
// @connect jiuse.cloud
// @grant GM.getValue
// @grant GM.setValue
// @grant GM_addStyle
// @grant GM_getValue
// @grant GM_info
// @grant GM_openInTab
// @grant GM_registerMenuCommand
// @grant GM_unregisterMenuCommand
// @grant GM_xmlhttpRequest
// @icon 
// @license MIT
// @namespace com.github.bigbowl-wtw
// @require https://cdn.jsdelivr.net/npm/hls.js@1.4.4/dist/hls.min.js
// @require https://greasyfork.org/scripts/470000/code/GM%20Requests.js
// ==/UserScript==
/******/ (() => { // webpackBootstrap
/******/ var __webpack_modules__ = ({
/***/ 264:
/***/ ((module) => {
module.exports = "1. 更名为 “South Plus 视频链接播放器”\n2. 修复不能播放 Twitter 视频的问题\n3. 添加了对九色网(91prony 及各种马甲)的支持\n点击“确定”查看更多细节"
/***/ }),
/***/ 382:
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
/* harmony export */ sd: () => (/* binding */ defaultEventEmitter),
/* harmony export */ zW: () => (/* binding */ Events)
/* harmony export */ });
/* unused harmony export EventEmitter */
/* eslint-disable @typescript-eslint/no-unused-vars */
var Events;
(function (Events) {
Events["SHOW_UPDATE_INFORMATION"] = "show-update-infromation";
Events["TWITTER_VERIFY"] = "twitter-vertify";
Events["TWITTWR_INVALIDATE"] = "twitter-invalidate";
Events["TWITTER_REAUTHORIZE"] = "twitter-reauthorize";
Events["MENU_REGISTER_AUTHORIZATION"] = "menu-register-authorization";
Events["MENU_REGISTER_CLEARCACHE"] = "menu-register-clearcache";
Events["VIEW_SHOW_TWITTER_CT0"] = "view-show-twitter-ct0";
Events["VIEW_SHOW_TWITTER_LOGGEDIN"] = "view-show-twitter-loggedin";
Events["VIEW_SHOW_TWITTER_GUEST"] = "view-show-twitter-guest";
Events["VIEW_CLOSE_DIALOG"] = "view-close-dialog";
})(Events || (Events = {}));
/**
* EventEmitter based on jQuery event
*/
class EventEmitter {
constructor(element = document) {
this.emitter = jQuery(element);
}
on(event, listener, context = this) {
this.emitter.on(event, listener.bind(context));
}
once(event, listener, context = this) {
this.emitter.one(event, listener.bind(context));
}
/**
* 如果传入 `listener`,将移除所有已绑定的事件,然后绑定 `listener`
*/
off(event, listener, context = this) {
if (!listener)
this.emitter.off(event);
else
this.emitter.off(event, '**', listener.bind(context));
}
trigger(event, ...datas) {
this.emitter.trigger(event, ...datas);
}
}
let emitter;
function defaultEventEmitter() {
if (!emitter)
emitter = new EventEmitter();
return emitter;
}
/***/ }),
/***/ 537:
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
/* harmony export */ Z: () => (__WEBPACK_DEFAULT_EXPORT__)
/* harmony export */ });
class Handler {
/**
* 注册 handler 类,使代码生效
*
* @param handler 待注册的 handler 类
*/
static registerHandler(handler) {
if (!Handler.handlers)
Handler.handlers = [];
Handler.handlers.push(handler);
}
/**
* TODO: 由于某些 handler 实现的 ``.apply`` 可能为异步方法,在页面上有多个匹配的
* URL 时,将产生状态冲突的问题(比如上次产生的 id 与本次冲突,而上次调用还未
* 完成,不能简单的进行清理,进而影响到本次调用),且比较难以解决,因此采用每
* 调用生成新实例的方法来避免。更好的方法待探索。
*/
static process() {
jQuery('.tpc_content a')
.filter((_, a) => /^http/.test(a.href))
.each((_, a) => {
for (const H of Handler.handlers) {
const handler = new H();
if (handler.test(a)) {
handler.apply(a);
break;
}
}
});
}
}
/* 实际生效的 handler */
Handler.handlers = [];
/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (Handler);
/***/ }),
/***/ 504:
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
// EXPORTS
__webpack_require__.d(__webpack_exports__, {
I: () => (/* binding */ TwitterAPI)
});
// NAMESPACE OBJECT: ./src/libs/twitter/service/fetch.ts
var fetch_namespaceObject = {};
__webpack_require__.r(fetch_namespaceObject);
__webpack_require__.d(fetch_namespaceObject, {
FetchService: () => (FetchService),
requests: () => ((external_requests_default()))
});
// EXTERNAL MODULE: ./src/managers/config.ts
var config = __webpack_require__(137);
;// CONCATENATED MODULE: ./src/libs/twitter/models/data/user.ts
/**
* The details of a single user.
*
* @internal
*/
class User {
/**
* @param user The raw user data.
*/
constructor(user) {
this.id = user.rest_id;
this.userName = user.legacy.screen_name;
this.fullName = user.legacy.name;
this.createdAt = user.legacy.created_at;
this.description = user.legacy.description;
this.isVerified = user.legacy.verified;
this.favouritesCount = user.legacy.favourites_count;
this.followersCount = user.legacy.followers_count;
this.followingsCount = user.legacy.friends_count;
this.statusesCount = user.legacy.statuses_count;
this.location = user.legacy.location;
this.pinnedTweet = user.legacy.pinned_tweet_ids_str[0];
this.profileBanner = user.legacy.profile_banner_url;
this.profileImage = user.legacy.profile_image_url_https;
}
}
class UserFromInitialState {
constructor(initialState) {
const entities = initialState.entities.users.entities;
this.id = Object.keys(entities)[0];
const user = entities[this.id];
this.userName = user.screen_name;
this.fullName = user.name;
this.createdAt = user.created_at;
this.description = user.description;
this.isVerified = user.verified;
this.favouritesCount = user.favourites_count;
this.followersCount = user.followers_count;
this.followingsCount = user.friends_count;
this.statusesCount = user.statuses_count;
this.location = user.location;
this.pinnedTweet = user.pinned_tweet_ids_str[0];
this.profileBanner = user.profile_banner_url;
this.profileImage = user.profile_image_url_https;
}
}
// EXTERNAL MODULE: external "requests"
var external_requests_ = __webpack_require__(173);
var external_requests_default = /*#__PURE__*/__webpack_require__.n(external_requests_);
;// CONCATENATED MODULE: ./src/libs/twitter/service/fetch.ts
var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
// eslint-disable-next-line import/no-extraneous-dependencies
class FetchService {
constructor(auth) {
this.requests = (external_requests_default());
if (auth)
this.auth = auth;
}
get(url, options) {
return this.request('GET', url, options);
}
post(url, options) {
return this.request('POST', url, options);
}
// eslint-disable-next-line class-methods-use-this
refreshAuth() {
this.auth.refresh();
}
// eslint-disable-next-line class-methods-use-this
request(method, url, options = {}) {
return __awaiter(this, void 0, void 0, function* () {
const opt = {};
if (options.json !== false)
opt.responseType = 'json';
opt.data = options.data;
opt.json = options.dataJSON;
if (options.authenticate !== undefined)
this.auth.setAuthenticate(options.authenticate);
opt.auth = this.auth;
opt.headers = { 'Content-Type': 'application/json' };
return external_requests_default().session()
.request(method, url.toString(), opt);
});
}
}
;// CONCATENATED MODULE: ./src/libs/twitter/service/account.ts
var account_awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
class AccountService {
constructor(auth) {
this.http = new FetchService(auth);
}
verifyLogin() {
return account_awaiter(this, void 0, void 0, function* () {
const result = yield this.getSelf();
return typeof result !== 'string';
});
}
getSelf() {
return account_awaiter(this, void 0, void 0, function* () {
if (config/* default */.Z.user_info)
return config/* default */.Z.user_info;
const url = 'https://twitter.com/home';
const html = yield this.http.requests
.get(url)
.then(resp => (resp.finalUrl === url ? resp.responseText : null));
if (!html) {
console.error('TwitterAPI.account.getSelf', 'can not get response text');
return { error: true, reason: 'can not get response text' };
}
const result = /window.__INITIAL_STATE__=(.*);window.__META_DATA__/.exec(html);
if (!result) {
console.error('TwitterAPI.account.getSelf', 'no __INITIAL_STATE__ find');
return { error: true, reason: 'no __INITIAL_STATE__ find' };
}
const initialState = JSON.parse(result[1]);
const user = new UserFromInitialState(initialState);
config/* default */.Z.user_info = user;
return user;
});
}
refreshAuth() {
this.http.refreshAuth();
}
// eslint-disable-next-line class-methods-use-this
setCSRFToken(ct0) {
this.http.auth.setCSRFToken(ct0);
}
}
// EXTERNAL MODULE: ./src/managers/cache.ts
var cache = __webpack_require__(944);
;// CONCATENATED MODULE: ./src/libs/twitter/config.ts
const config_config = {
twitter_auth_token: 'Bearer AAAAAAAAAAAAAAAAAAAAAPYXBAAAAAAACLXUNDekMxqa8h%2F40K4moUkGsoc%3DTYfbDKbT3jJPCEVnMYqilB28NHfOPqkca3qaAxGfsyKCs0wRbw',
guest_token_url: 'https://api.twitter.com/1.1/guest/activate.json',
};
;// CONCATENATED MODULE: ./src/libs/twitter/service/auth.ts
var auth_awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
// eslint-disable-next-line @typescript-eslint/no-unused-vars
class AuthService {
// eslint-disable-next-line class-methods-use-this
get guestTokne() {
return cache/* default */.Z.get("GUEST_TOKEN" /* Key.GUEST_TOKEN */);
}
// eslint-disable-next-line class-methods-use-this
set guestTokne(token) {
cache/* default */.Z.set("GUEST_TOKEN" /* Key.GUEST_TOKEN */, token, 10800);
}
constructor(ct0) {
this.authenticate = true;
this.authToken = config_config.twitter_auth_token;
this.setCSRFToken(ct0);
}
setAuthenticate(authenticate) {
this.authenticate = authenticate;
return this;
}
setCSRFToken(ct0) {
this.csrfToken = ct0;
this.isAuthenticated = !!ct0;
}
refresh() {
return auth_awaiter(this, void 0, void 0, function* () {
const { guest_token } = yield external_requests_default().post(config_config.guest_token_url, {
headers: {
Authorization: this.authToken,
},
responseType: 'json',
});
this.guestTokne = guest_token;
});
}
build(header) {
return auth_awaiter(this, void 0, void 0, function* () {
// 已经登录,cookie 由浏览器管理
if (this.authenticate && this.isAuthenticated) {
header.update({
Authorization: this.authToken,
'x-csrf-token': this.csrfToken,
'x-twitter-auth-type': 'OAuth2Session',
'x-twitter-active-user': 'yes',
'x-twitter-client-language': 'zh-cn',
});
}
else {
if (!this.guestTokne) {
yield this.refresh();
}
header.update({
Authorization: this.authToken,
'x-guest-token': this.guestTokne,
});
}
});
}
}
;// CONCATENATED MODULE: ./src/libs/twitter/endpoints.ts
const Endpoint = {
TweetDetail: {
queryId: '3XDB26fBve-MmjHaWTUZxA',
operationName: 'TweetDetail',
operationType: 'query',
metadata: {
featureSwitches: [
'rweb_lists_timeline_redesign_enabled',
'responsive_web_graphql_exclude_directive_enabled',
'verified_phone_label_enabled',
'creator_subscriptions_tweet_preview_api_enabled',
'responsive_web_graphql_timeline_navigation_enabled',
'responsive_web_graphql_skip_user_profile_image_extensions_enabled',
'tweetypie_unmention_optimization_enabled',
'responsive_web_edit_tweet_api_enabled',
'graphql_is_translatable_rweb_tweet_is_translatable_enabled',
'view_counts_everywhere_api_enabled',
'longform_notetweets_consumption_enabled',
'responsive_web_twitter_article_tweet_consumption_enabled',
'tweet_awards_web_tipping_enabled',
'freedom_of_speech_not_reach_fetch_enabled',
'standardized_nudges_misinfo',
'tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled',
'longform_notetweets_rich_text_read_enabled',
'longform_notetweets_inline_media_enabled',
'responsive_web_media_download_video_enabled',
'responsive_web_enhance_cards_enabled',
],
fieldToggles: ['withArticleRichContentState'],
},
features: {
rweb_lists_timeline_redesign_enabled: true,
responsive_web_graphql_exclude_directive_enabled: true,
verified_phone_label_enabled: false,
creator_subscriptions_tweet_preview_api_enabled: true,
responsive_web_graphql_timeline_navigation_enabled: true,
responsive_web_graphql_skip_user_profile_image_extensions_enabled: false,
tweetypie_unmention_optimization_enabled: true,
responsive_web_edit_tweet_api_enabled: true,
graphql_is_translatable_rweb_tweet_is_translatable_enabled: true,
view_counts_everywhere_api_enabled: true,
longform_notetweets_consumption_enabled: true,
responsive_web_twitter_article_tweet_consumption_enabled: false,
tweet_awards_web_tipping_enabled: false,
freedom_of_speech_not_reach_fetch_enabled: true,
standardized_nudges_misinfo: true,
tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled: true,
longform_notetweets_rich_text_read_enabled: true,
longform_notetweets_inline_media_enabled: true,
responsive_web_media_download_video_enabled: false,
responsive_web_enhance_cards_enabled: false,
},
},
};
/* harmony default export */ const endpoints = (Endpoint);
;// CONCATENATED MODULE: ./src/libs/twitter/enums/errors.ts
/**
* Copied from https://github.com/Rishikant181/Rettiwt-API/blob/e2df415c131582d6c534bbbc2976afabd0bbc43e/src/enums/Errors.ts
*/
/**
* Different type of error messages related to data that are returned by services.
*
* @public
*/
var DataErrors;
(function (DataErrors) {
DataErrors["UserNotFound"] = "An account with given username/id was not found";
DataErrors["TweetNotFound"] = "A tweet with the given id was not found";
})(DataErrors || (DataErrors = {}));
;// CONCATENATED MODULE: ./src/libs/twitter/helper/parser.ts
/**
* Copied from https://github.com/Rishikant181/Rettiwt-API/blob/e2df415c131582d6c534bbbc2976afabd0bbc43e/src/services/helper/Parser.ts
*/
/**
* @returns Whether the given json object is empty or not
* @param data The input JSON object which needs to be checked
*/
function isJSONEmpty(data) {
// If the JSON has any keys, it's not empty
if (Object.keys(data).length === 0) {
return true;
}
// Else, it's empty
return false;
}
/**
* @param text The text to be normalized
* @returns The text after being formatted to remove unnecessary characters
*/
function normalizeText(text) {
let normalizedText = ''; // To store the normalized text
// Removing unnecessary full stops, and other characters
normalizedText = text.replace(/\n/g, '.').replace(/[.]+[\s+.\s+]+/g, '. ');
// Adding full-stop to the end if does not exist already
normalizedText = normalizedText.endsWith('.')
? normalizedText
: `${normalizedText}.`;
return normalizedText;
}
;// CONCATENATED MODULE: ./src/libs/twitter/helper/extractors/tweets.ts
// TYPES
// PARSERS
/**
* @returns The raw tweet data formatted and sorted into required and additional data
* @param res The raw response received from TwitterAPI
* @param tweetId The rest id of the tweet to fetch
*/
function extractTweet(res, tweetId) {
var _a;
const required = []; // To store the reqruied raw data
const cursor = ''; // To store the cursor to next batch
const users = []; // To store additional user data
const tweets = []; // To store additional tweet data
// If tweet does not exist
if (isJSONEmpty(res.data)) {
throw new Error(DataErrors.TweetNotFound);
}
// Destructuring the received raw data
(_a = res.data.threaded_conversation_with_injections_v2.instructions
.filter(item => item.type === 'TimelineAddEntries')[0]
.entries) === null || _a === void 0 ? void 0 : _a.forEach(entry => {
var _a, _b, _c;
// If entry is of type tweet and tweet exists
if (entry.entryId.indexOf('tweet') !== -1 &&
((_b = (_a = entry.content.itemContent) === null || _a === void 0 ? void 0 : _a.tweet_results) === null || _b === void 0 ? void 0 : _b.result.__typename) ===
'Tweet') {
// If this is the required tweet
if (entry.entryId.indexOf(tweetId) !== -1) {
required.push(entry.content.itemContent.tweet_results.result);
}
tweets.push(entry.content.itemContent.tweet_results.result);
users.push(entry.content.itemContent.tweet_results.result.core
.user_results.result);
}
// If entry if of type conversation
else if (entry.entryId.indexOf('conversationthread') !== -1) {
// Iterating over the conversation
(_c = entry.content.items) === null || _c === void 0 ? void 0 : _c.forEach(item => {
var _a, _b;
// If item is of type tweet and tweet exists
if (item.entryId.indexOf('tweet') !== -1 &&
((_b = (_a = item.item.itemContent.tweet_results) === null || _a === void 0 ? void 0 : _a.result) === null || _b === void 0 ? void 0 : _b.__typename) === 'Tweet') {
required.push(item.item.itemContent.tweet_results.result);
tweets.push(item.item.itemContent.tweet_results.result);
users.push(item.item.itemContent.tweet_results.result.core
.user_results.result);
}
});
}
});
// Returning the data
return {
required,
cursor,
users,
tweets,
};
}
;// CONCATENATED MODULE: ./src/libs/twitter/models/data/tweet.ts
/**
* Copies from https://github.com/Rishikant181/Rettiwt-API/blob/e2df415c131582d6c534bbbc2976afabd0bbc43e/src/models/data/Tweet.ts
*/
/**
* The different types parsed entities like urls, media, mentions, hashtags, etc.
*
* @internal
*/
class TweetEntities {
// MEMBER METHODS
constructor(entities) {
// MEMBER DATA
/** The list of hashtags mentioned in the tweet. */
this.hashtags = [];
/** The list of urls mentioned in the tweet. */
this.urls = [];
/** The list of IDs of users mentioned in the tweet. */
this.mentionedUsers = [];
/** The list of urls to various media mentioned in the tweet. */
this.media = [];
// Extracting user mentions
if (entities.user_mentions) {
for (const user of entities.user_mentions) {
this.mentionedUsers.push(user.id_str);
}
}
// Extracting urls
if (entities.urls) {
for (const url of entities.urls) {
this.urls.push(url.expanded_url);
}
}
// Extracting hashtags
if (entities.hashtags) {
for (const hashtag of entities.hashtags) {
this.hashtags.push(hashtag.text);
}
}
// Extracting media urls (if any)
if (entities.media) {
for (const media of entities.media) {
this.media.push(media.media_url_https);
}
}
}
}
class TweetMeida {
constructor(media) {
this.poster = media.media_url_https;
this.durationMs = media.video_info.duration_millis;
const variants = media.video_info.variants
.map(x => {
if (x.bitrate === undefined)
x.bitrate = Infinity;
return x;
})
.sort((a, b) => b.bitrate - a.bitrate);
this.stream = variants[0].url;
this.fallback = variants[1].url;
}
}
class TweetExtendedEntities {
constructor(extendedEntities) {
this.media = [];
extendedEntities.media.forEach(media => {
this.media.push(new TweetMeida(media));
});
}
}
/**
* The details of a single Tweet.
*
* @internal
*/
class Tweet {
/**
* @param tweet The raw tweet data.
*/
constructor(tweet) {
this.id = tweet.rest_id;
this.createdAt = tweet.legacy.created_at;
this.tweetBy = tweet.legacy.user_id_str;
this.entities = new TweetEntities(tweet.legacy.entities);
this.extendedEntities = new TweetExtendedEntities(tweet.legacy.extended_entities);
this.quoted = tweet.legacy.quoted_status_id_str;
this.fullText = normalizeText(tweet.legacy.full_text);
this.replyTo = tweet.legacy.in_reply_to_status_id_str;
this.lang = tweet.legacy.lang;
this.quoteCount = tweet.legacy.quote_count;
this.replyCount = tweet.legacy.reply_count;
this.retweetCount = tweet.legacy.retweet_count;
this.likeCount = tweet.legacy.favorite_count;
}
}
;// CONCATENATED MODULE: ./src/libs/twitter/variables.ts
// eslint-disable-next-line no-underscore-dangle
const _Variables = {
TweetDetail: {
with_rux_injections: false,
includePromotedContent: true,
withCommunity: true,
withQuickPromoteEligibilityTweetFields: true,
withBirdwatchNotes: true,
withVoice: true,
withV2Timeline: true,
},
};
class Variables {
constructor(endpoint, query) {
this.variables = Object.assign(Object.assign({}, (query !== null && query !== void 0 ? query : {})), _Variables[endpoint.operationName]);
}
toString() {
return JSON.stringify(this.variables);
}
}
/* harmony default export */ const variables = (Variables);
;// CONCATENATED MODULE: ./src/libs/twitter/url.ts
class Url {
constructor(endpoint, args) {
var _a, _b;
this.baseUrl = 'https://twitter.com/i/api/graphql';
const url = new URL(`${this.baseUrl}/${endpoint.queryId}/${endpoint.operationName}`);
url.searchParams.set('variables', new variables(endpoint, { focalTweetId: args.id }).toString());
url.searchParams.set('features', JSON.stringify(endpoint.features));
if (((_a = endpoint.metadata) === null || _a === void 0 ? void 0 : _a.fieldToggles.length) > 0)
url.searchParams.set('fieldToggles', JSON.stringify(Object.fromEntries((_b = endpoint.metadata) === null || _b === void 0 ? void 0 : _b.fieldToggles.map(key => [key, false]))));
this.url = url;
}
toString() {
return this.url.toString();
}
}
;// CONCATENATED MODULE: ./src/libs/twitter/service/tweet.ts
var tweet_awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
class TweetService {
constructor(auth) {
this.http = new FetchService(auth);
}
getTweetDetail(id) {
return tweet_awaiter(this, void 0, void 0, function* () {
const cachedData = cache/* default */.Z.get(id);
if (cachedData) {
return cachedData;
}
const url = new Url(endpoints.TweetDetail, { id });
const ret = yield this.http.get(url);
// Fetching the raw data
const data = extractTweet(ret, id);
// Parsing data
const tweet = new Tweet(data.required[0]);
// Caching data
cache/* default */.Z.set(id, tweet);
return tweet;
});
}
}
;// CONCATENATED MODULE: ./src/libs/twitter/service/user.ts
var user_awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
class UserService {
constructor(auth) {
this.http = fetch_namespaceObject;
this.auth = auth;
}
getSelf() {
return user_awaiter(this, void 0, void 0, function* () {
if (config/* default */.Z.user_info)
return config/* default */.Z.user_info;
const url = 'https://twitter.com/home';
const html = yield this.http.requests
.get(url)
.then(resp => (resp.finalUrl === url ? resp.responseText : null));
if (!html)
return null;
const result = /window.__INITIAL_STATE__=(.*);window.__META_DATA__/.exec(html);
if (!result)
return null;
const initialState = JSON.parse(result[1]);
const user = new UserFromInitialState(initialState);
config/* default */.Z.user_info = user;
return user;
});
}
}
;// CONCATENATED MODULE: ./src/libs/twitter/index.ts
let api;
function TwitterAPI(ct0) {
if (!api) {
const auth = new AuthService(ct0);
api = {
account: new AccountService(auth),
user: new UserService(auth),
tweet: new TweetService(auth),
};
}
else if (ct0 && ct0 !== '') {
api.account.setCSRFToken(ct0);
}
return api;
}
/***/ }),
/***/ 944:
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
/* harmony export */ Z: () => (__WEBPACK_DEFAULT_EXPORT__)
/* harmony export */ });
/* harmony import */ var _manager__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(450);
const MAXTTL = 7 * 24 * 3600;
class Data {
constructor(value, ttl, timestamp = false) {
this.value = value;
const now = Math.floor(Date.now() / 1000);
this.lastActive = now;
this.ttl = timestamp ? ttl - now : ttl;
this.refresh = !timestamp;
}
isDead() {
const now = Math.floor(Date.now() / 1000);
if (this.refresh)
this.lastActive = now;
return now - this.lastActive >= this.ttl;
}
}
class Cache extends _manager__WEBPACK_IMPORTED_MODULE_0__/* ["default"] */ .Z {
constructor() {
super("cache" /* Key.CACHE */);
this.dataSet = oldViersonHandler(this.dataSet);
}
get(key) {
const data = this.dataSet[key];
if (!data)
return undefined;
if (data.isDead()) {
delete this.dataSet[key];
data.value = undefined;
}
this.dump();
return data.value;
}
set(key, value, expire = MAXTTL, timestamp = false) {
this.dataSet[key] = new Data(value, expire, timestamp);
this.dump();
}
clean() {
const dataSet = this.dataSet;
for (const key of Object.keys(dataSet)) {
const data = dataSet[key];
if (data.isDead())
delete dataSet[key];
}
this.dump();
}
clear() {
this.dataSet = {};
this.dump();
}
ttl(key, ttl) {
if (ttl)
this.dataSet[key].ttl = ttl;
return this.dataSet[key].ttl;
}
}
function oldViersonHandler(dataSet) {
return Object.fromEntries(Object.entries(dataSet).map(([key, val]) => {
const { value, timestamp, lastActive } = val;
const data = new Data(value, MAXTTL);
data.lastActive = timestamp !== null && timestamp !== void 0 ? timestamp : lastActive;
return [key, data];
}));
}
const cache = new Cache();
/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (cache);
/***/ }),
/***/ 137:
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
/* harmony export */ Z: () => (__WEBPACK_DEFAULT_EXPORT__)
/* harmony export */ });
/* harmony import */ var _manager__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(450);
class Config extends _manager__WEBPACK_IMPORTED_MODULE_0__/* ["default"] */ .Z {
constructor() {
super("config" /* Key.CONFIG */);
}
get(key) {
return this.dataSet[key];
}
set(key, value) {
this.dataSet[key] = value;
this.dump();
}
delete(key) {
delete this.dataSet[key];
this.dump();
}
}
const config = new Proxy(new Config(), {
get(target, prop) {
return target.get(prop);
},
set(target, prop, value) {
target.set(prop, value);
return true;
},
deleteProperty(target, property) {
target.delete(property);
return true;
},
});
/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (config);
/***/ }),
/***/ 450:
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
/* harmony export */ Z: () => (/* binding */ Manager)
/* harmony export */ });
/* harmony import */ var _utils_utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(803);
class Manager {
constructor(name) {
this.name = name;
this.dump = _utils_utils__WEBPACK_IMPORTED_MODULE_0__/* .dump */ .$w;
this.dataSet = GM_getValue(name, {});
}
}
/***/ }),
/***/ 803:
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
// EXPORTS
__webpack_require__.d(__webpack_exports__, {
$w: () => (/* binding */ dump),
E8: () => (/* binding */ insertHlsPlayer),
f4: () => (/* binding */ insertIFrameElement),
aS: () => (/* binding */ insertVideoElement),
_C: () => (/* binding */ isUpdated)
});
// UNUSED EXPORTS: debounce
;// CONCATENATED MODULE: external "Hls"
const external_Hls_namespaceObject = Hls;
var external_Hls_default = /*#__PURE__*/__webpack_require__.n(external_Hls_namespaceObject);
;// CONCATENATED MODULE: ./src/view/iframe.html
// Module
var code = "<div style=\"position:relative;padding:30% 45%\"> <iframe class=\"embed lazy\" style=\"position:absolute;width:100%;height:100%;left:0;top:0\" referrerpolicy=\"no-referrer\"></iframe> </div> ";
// Exports
/* harmony default export */ const view_iframe = (code);
;// CONCATENATED MODULE: ./src/view/video.html
// Module
var video_code = "<div style=\"position:relative;padding:30% 45%\"> <video class=\"volume-sync\" style=\"position:absolute;width:100%;height:100%;left:0;top:0\" controls></video> </div> ";
// Exports
/* harmony default export */ const view_video = (video_code);
;// CONCATENATED MODULE: ./src/utils/utils.ts
var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __rest = (undefined && undefined.__rest) || function (s, e) {
var t = {};
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
t[p] = s[p];
if (s != null && typeof Object.getOwnPropertySymbols === "function")
for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
t[p[i]] = s[p[i]];
}
return t;
};
/* eslint-disable func-names */
// eslint-disable-next-line import/no-extraneous-dependencies
/**
* 防抖
* @param fn 回调
* @param timeout 间隔
*/
function debounce(fn, timeout) {
let timer;
return function (...rest) {
window.clearTimeout(timer);
timer = window.setTimeout(fn.bind(this, ...rest), timeout);
};
}
const dump = debounce(function () {
GM.setValue(this.name, this.dataSet);
}, 200);
function insertVideoElement(a, attrs) {
if (typeof attrs === 'string' /* 直接传入 URL */) {
const src = attrs;
attrs = { src };
}
const $htmlText = jQuery(view_video);
const $video = jQuery('video', $htmlText);
$video.attr(attrs);
$htmlText.insertBefore(a);
const video = $video[0];
return { video, attrs };
}
function insertHlsPlayer(a, attributes) {
const { video, attrs } = insertVideoElement(a, attributes);
// apple 平台原生支持 hls,无需处理
if (!/Mac|iPod|iPhone|iPad/.test(navigator.userAgent)) {
if (external_Hls_default().isSupported()) {
const hls = new (external_Hls_default())();
hls.loadSource(attrs.src);
hls.attachMedia(video);
hls.on((external_Hls_default()).Events.MANIFEST_PARSED, function (event, data) {
data.levels.sort((x, y) => y.bitrate - x.bitrate);
hls.currentLevel = 0;
});
}
else {
console.log('Hls不支持当前浏览器');
if (attrs.fallback)
video.src = attrs.fallback;
}
}
}
function bindVolumeSync() {
return __awaiter(this, void 0, void 0, function* () {
let volume = yield GM.getValue("lastVolume" /* Key.VOLUME */, 1);
jQuery('video.volume-sync')
.on('volume-sync', function (_, volume_) {
this.volume = volume_;
})
.on('mouseover', function () {
jQuery(this).on('volumechange', debounce(function () {
volume = this.volume;
GM.setValue("lastVolume" /* Key.VOLUME */, volume);
jQuery('video.volume-sync')
.not(this)
.trigger('volume-sync', volume);
}, 200));
})
.on('mouseout', function () {
jQuery(this).off('volumechange');
});
});
}
// $.on 对未来的元素有效,因此直接执行即可
bindVolumeSync();
const observer = new IntersectionObserver(entries => {
entries
.filter(entry => entry.isIntersecting)
.forEach(entry => {
console.log('lazyload');
const e = entry.target;
e.src = e.dataset.src;
e.classList.replace('lazy', 'loaded');
observer.unobserve(e);
});
}, {
rootMargin: '0px 0px 640px',
});
function insertIFrameElement(a, attributes) {
const { src } = attributes, attrs = __rest(attributes, ["src"]);
attrs['data-src'] = src;
const $div = jQuery(view_iframe);
const iframe = jQuery('iframe', $div).attr(attrs)[0];
observer.observe(iframe);
$div.insertBefore(a);
}
function toNumber(version) {
const v = version.replace('alpha', '-2').replace('beta', '-1');
return v.split('.').map(x => parseInt(x, 10));
}
function shiftDefault(array, default_) {
const val = array.shift();
if (val === undefined)
return default_;
return val;
}
function laterThan(newer, local) {
const v1 = toNumber(newer);
const v2 = toNumber(local);
let a;
let b;
while (v1.length || v2.length) {
a = shiftDefault(v1, 0);
b = shiftDefault(v2, 0);
if (a > b)
return true;
}
return false;
}
function isUpdated(callback) {
GM.getValue("version" /* Key.VERSION */, '0.0.0').then(localVersion => {
if (laterThan(GM_info.script.version, localVersion)) {
callback();
GM.setValue("version" /* Key.VERSION */, GM_info.script.version);
}
});
}
/***/ }),
/***/ 173:
/***/ ((module) => {
"use strict";
module.exports = requests;
/***/ })
/******/ });
/************************************************************************/
/******/ // The module cache
/******/ var __webpack_module_cache__ = {};
/******/
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/ // Check if module is in cache
/******/ var cachedModule = __webpack_module_cache__[moduleId];
/******/ if (cachedModule !== undefined) {
/******/ return cachedModule.exports;
/******/ }
/******/ // Create a new module (and put it into the cache)
/******/ var module = __webpack_module_cache__[moduleId] = {
/******/ // no module.id needed
/******/ // no module.loaded needed
/******/ exports: {}
/******/ };
/******/
/******/ // Execute the module function
/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
/******/
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/
/************************************************************************/
/******/ /* webpack/runtime/compat get default export */
/******/ (() => {
/******/ // getDefaultExport function for compatibility with non-harmony modules
/******/ __webpack_require__.n = (module) => {
/******/ var getter = module && module.__esModule ?
/******/ () => (module['default']) :
/******/ () => (module);
/******/ __webpack_require__.d(getter, { a: getter });
/******/ return getter;
/******/ };
/******/ })();
/******/
/******/ /* webpack/runtime/define property getters */
/******/ (() => {
/******/ // define getter functions for harmony exports
/******/ __webpack_require__.d = (exports, definition) => {
/******/ for(var key in definition) {
/******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
/******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
/******/ }
/******/ }
/******/ };
/******/ })();
/******/
/******/ /* webpack/runtime/hasOwnProperty shorthand */
/******/ (() => {
/******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
/******/ })();
/******/
/******/ /* webpack/runtime/make namespace object */
/******/ (() => {
/******/ // define __esModule on exports
/******/ __webpack_require__.r = (exports) => {
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
/******/ }
/******/ Object.defineProperty(exports, '__esModule', { value: true });
/******/ };
/******/ })();
/******/
/************************************************************************/
var __webpack_exports__ = {};
// This entry need to be wrapped in an IIFE because it need to be in strict mode.
(() => {
"use strict";
var __webpack_exports__ = {};
/* unused harmony export default */
/* harmony import */ var gm_requests__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(173);
/* harmony import */ var gm_requests__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(gm_requests__WEBPACK_IMPORTED_MODULE_0__);
/* harmony import */ var _managers_cache__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(944);
/* harmony import */ var _utils_utils__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(803);
/* harmony import */ var _handler__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(537);
var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
// eslint-disable-next-line import/no-extraneous-dependencies
class BilibiliHandler {
constructor() {
this.regex = /b23.tv|(?<bvid>BV[a-lm-zA-HJ-NP-Z1-9]{10})/;
}
test(a) {
const result = this.regex.exec(a.href);
if (result)
this.bvid = result[1];
return !!result;
}
apply(a) {
return __awaiter(this, void 0, void 0, function* () {
let url = a.href;
if (!this.bvid) {
// b23.tv
const cached = _managers_cache__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.get(url);
if (!cached) {
yield this.parseShortURL(url);
this.apply(a);
return;
}
console.log('cached:', url);
url = cached;
this.bvid = this.regex.exec(url)[1];
}
const result = /p=(?<pid>\d+)/.exec(url);
const pid = result ? parseInt(result.groups.pid, 10) : 1;
const cachedInfo = _managers_cache__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.get(this.bvid);
let aid;
let cid;
if (cachedInfo) {
console.log('cached', a.href);
({ aid, cid } = cachedInfo);
this.insertIframe(a, aid, cid, pid);
return;
}
const ret = yield gm_requests__WEBPACK_IMPORTED_MODULE_0___default().get('https://api.bilibili.com/x/web-interface/view', { bvid: this.bvid }, { responseType: 'json' });
aid = ret.data.aid;
cid = ret.data.pages[pid - 1].cid;
_managers_cache__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.set(ret.data.bvid, { aid, cid });
this.insertIframe(a, aid, cid, pid);
});
}
parseShortURL(url) {
return __awaiter(this, void 0, void 0, function* () {
const resp = yield gm_requests__WEBPACK_IMPORTED_MODULE_0___default().get(url, null, { anonymous: true });
_managers_cache__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.set(url, resp.finalUrl);
const result = this.regex.exec(resp.finalUrl);
if (result)
this.bvid = result[1];
});
}
insertIframe(a, aid, cid, pid) {
(0,_utils_utils__WEBPACK_IMPORTED_MODULE_2__/* .insertIFrameElement */ .f4)(a, {
src: `https://player.bilibili.com/player.html?aid=${aid}&bvid=${this.bvid}&cid=${cid}&page=${pid}&as_wide=1&high_quality=1&danmaku=0&autoplay=0`,
frameborder: 'no',
scrolling: 'no',
allowfullscreen: true,
});
}
}
_handler__WEBPACK_IMPORTED_MODULE_3__/* ["default"] */ .Z.registerHandler(BilibiliHandler);
})();
// This entry need to be wrapped in an IIFE because it need to be in strict mode.
(() => {
"use strict";
var __webpack_exports__ = {};
/* unused harmony export default */
/* harmony import */ var _utils_utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(803);
/* harmony import */ var _handler__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(537);
class BrowserSupportedHandler {
constructor() {
this.regex = /(mp4|mov)$/i;
}
test(a) {
return this.regex.test(a.href);
}
// eslint-disable-next-line class-methods-use-this
apply(a) {
(0,_utils_utils__WEBPACK_IMPORTED_MODULE_0__/* .insertVideoElement */ .aS)(a, { src: a.href });
}
}
_handler__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.registerHandler(BrowserSupportedHandler);
})();
// This entry need to be wrapped in an IIFE because it need to be in strict mode.
(() => {
"use strict";
var __webpack_exports__ = {};
/* unused harmony export default */
/* harmony import */ var gm_requests__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(173);
/* harmony import */ var gm_requests__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(gm_requests__WEBPACK_IMPORTED_MODULE_0__);
/* harmony import */ var _managers_cache__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(944);
/* harmony import */ var _utils_utils__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(803);
/* harmony import */ var _handler__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(537);
var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
// eslint-disable-next-line import/no-extraneous-dependencies
const MAX_EXPIRE_SECOND = 5 * 24 * 3600;
class HlsParser {
parse(url) {
return __awaiter(this, void 0, void 0, function* () {
this.baseUrl = url.substring(0, url.lastIndexOf('/'));
const ret = yield gm_requests__WEBPACK_IMPORTED_MODULE_0___default().get(url, null, { anonymous: true });
this.hlsString = ret.response;
this.addBaseUrl();
return this.hlsString;
});
}
addBaseUrl() {
this.hlsString = this.hlsString.replace(/^(?!#)(\S+)/gm, `${this.baseUrl}/$1`);
}
}
class JiuseHandler {
constructor() {
this.regex = /(?:91porny|jiuse|9s\d{3}.xyz|js(?:tv\d?|\d{3}).cc).+\/(\w+)/;
this.parser = new HlsParser();
}
test(a) {
const result = this.regex.exec(a.href);
if (result)
this.id = result[1];
return !!result;
}
apply(a) {
return __awaiter(this, void 0, void 0, function* () {
let attrs = this.getCacheData();
if (!attrs) {
attrs = yield this.getAttrs();
}
(0,_utils_utils__WEBPACK_IMPORTED_MODULE_2__/* .insertHlsPlayer */ .E8)(a, attrs);
});
}
getCacheData() {
this.key = `jiuse-${this.id}`;
const cachedInfo = _managers_cache__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.get(this.key);
if (cachedInfo) {
console.log('cached:', this.key);
return {
src: JiuseHandler.createLevelFileURL(window.atob(cachedInfo.hlsString)),
poster: cachedInfo.poster,
};
}
return undefined;
}
getAttrs() {
return __awaiter(this, void 0, void 0, function* () {
const resp = yield gm_requests__WEBPACK_IMPORTED_MODULE_0___default().get(`https://91porny.com/video/embed/${this.id}`);
const $video = jQuery(resp.responseText, document.implementation.createHTMLDocument('virtual')).filter('#video-play');
const url = $video.data('src');
const poster = $video.data('poster');
const expireAt = Math.floor(Date.now() / 1000 + MAX_EXPIRE_SECOND);
const hlsString = yield this.parser.parse(url);
_managers_cache__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.set(this.key, { hlsString: window.btoa(hlsString), poster }, expireAt);
return {
src: JiuseHandler.createLevelFileURL(hlsString),
poster,
};
});
}
static createLevelFileURL(hlsString) {
return URL.createObjectURL(new File([hlsString], 'index.m3u8', {
type: 'text/plain',
}));
}
}
_handler__WEBPACK_IMPORTED_MODULE_3__/* ["default"] */ .Z.registerHandler(JiuseHandler);
})();
// This entry need to be wrapped in an IIFE because it need to be in strict mode.
(() => {
"use strict";
var __webpack_exports__ = {};
/* unused harmony export default */
/* harmony import */ var _events__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(382);
/* harmony import */ var _libs_twitter__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(504);
/* harmony import */ var _managers_config__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(137);
/* harmony import */ var _utils_utils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(803);
/* harmony import */ var _handler__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(537);
var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
const PREFER_STREAM_MIN_DURATION = 5 * 60 * 1000; // ms
class TweetVideo {
constructor(media) {
this.durationMs = 0;
this.poster = media.poster;
this.durationMs = media.durationMs;
this.isStream = this.durationMs > PREFER_STREAM_MIN_DURATION;
this.src = this.isStream ? media.stream : media.fallback;
this.fallback = media.fallback;
}
dump() {
const data = { src: this.src };
if (this.poster)
data.poster = this.poster;
if (this.isStream)
data.fallback = this.fallback;
return data;
}
}
class TwitterHandler {
constructor() {
this.regex = /video.twimg.com|twitter.com\/.*\/status\/(?<id>\d+)/;
this.twitter = (0,_libs_twitter__WEBPACK_IMPORTED_MODULE_1__/* .TwitterAPI */ .I)(_managers_config__WEBPACK_IMPORTED_MODULE_2__/* ["default"] */ .Z.ct0);
}
test(a) {
const result = this.regex.exec(a.href);
if (result)
this.id = result.groups.id;
return !!result;
}
apply(a) {
return __awaiter(this, void 0, void 0, function* () {
if (this.id === undefined) {
// twimg.com
(0,_utils_utils__WEBPACK_IMPORTED_MODULE_3__/* .insertVideoElement */ .aS)(a, a.href);
return;
}
if (!this.checkCT0()) {
(0,_events__WEBPACK_IMPORTED_MODULE_0__/* .defaultEventEmitter */ .sd)().trigger(_events__WEBPACK_IMPORTED_MODULE_0__/* .Events */ .zW.VIEW_SHOW_TWITTER_GUEST);
return;
}
const tweet = yield this.twitter.tweet
.getTweetDetail(this.id)
.catch(e => console.error(e));
if (!tweet)
return;
tweet.extendedEntities.media
.map(media => new TweetVideo(media))
.forEach(video => {
const attrs = video.dump();
if (video.isStream)
(0,_utils_utils__WEBPACK_IMPORTED_MODULE_3__/* .insertHlsPlayer */ .E8)(a, attrs);
else
(0,_utils_utils__WEBPACK_IMPORTED_MODULE_3__/* .insertVideoElement */ .aS)(a, attrs);
});
});
}
// eslint-disable-next-line class-methods-use-this
checkCT0() {
return !!(_managers_config__WEBPACK_IMPORTED_MODULE_2__/* ["default"] */ .Z.as_guest || _managers_config__WEBPACK_IMPORTED_MODULE_2__/* ["default"] */ .Z.ct0);
}
}
_handler__WEBPACK_IMPORTED_MODULE_4__/* ["default"] */ .Z.registerHandler(TwitterHandler);
})();
// This entry need to be wrapped in an IIFE because it need to be in strict mode.
(() => {
"use strict";
var __webpack_exports__ = {};
/* unused harmony export default */
/* harmony import */ var _utils_utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(803);
/* harmony import */ var _handler__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(537);
class YouTubeHandler {
constructor() {
this.regex = /(youtu.be\/|youtube.com\/(?:(?:embed|v|shorts)\/|watch\?v=))(?<vid>[\w-]+)/;
}
test(a) {
var _a;
this.vid = (_a = this.regex.exec(a.href)) === null || _a === void 0 ? void 0 : _a.groups.vid;
return !!this.vid;
}
apply(a) {
const result = /t|start=(?<t>\d+)s?/.exec(a.href);
let start;
if (result)
start = result[1];
const url = new URL(this.vid, 'https://www.youtube.com/embed/');
if (start)
url.searchParams.set('start', start);
(0,_utils_utils__WEBPACK_IMPORTED_MODULE_0__/* .insertIFrameElement */ .f4)(a, {
src: url.toString(),
allowfullscreen: true,
allow: 'accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share',
});
}
}
_handler__WEBPACK_IMPORTED_MODULE_1__/* ["default"] */ .Z.registerHandler(YouTubeHandler);
})();
// This entry need to be wrapped in an IIFE because it need to be in strict mode.
(() => {
"use strict";
// EXTERNAL MODULE: ./src/events.ts
var events = __webpack_require__(382);
// EXTERNAL MODULE: ./src/libs/twitter/index.ts + 14 modules
var twitter = __webpack_require__(504);
// EXTERNAL MODULE: ./src/managers/config.ts
var config = __webpack_require__(137);
;// CONCATENATED MODULE: ./src/controllers/twitter.ts
/* eslint-disable import/prefer-default-export */
const emitter = (0,events/* defaultEventEmitter */.sd)();
const TWITTER_LOGIN_URL = 'https://twitter.com/login';
class TwitterDialog {
constructor() {
this.registerListeners();
}
// eslint-disable-next-line class-methods-use-this
trigger(event, ...datas) {
return () => emitter.trigger(event, ...datas);
}
}
class TwitterCT0Dialog extends TwitterDialog {
constructor() {
super(...arguments);
this.twitter = (0,twitter/* TwitterAPI */.I)();
}
registerListeners() {
emitter.on(events/* Events */.zW.VIEW_SHOW_TWITTER_CT0, this.initView, this);
emitter.on(events/* Events */.zW.TWITTER_VERIFY, this.verify, this);
}
verify() {
jQuery('#loading-overlay', this.$dialog).fadeIn(200);
const ct0 = jQuery('#pin').val();
if (!ct0)
return;
config/* default */.Z.as_guest = false;
config/* default */.Z.ct0 = ct0;
this.twitter.account.setCSRFToken(ct0);
this.twitter.account
.getSelf()
.then(ret => {
if (!isError(ret)) {
emitter.trigger(events/* Events */.zW.VIEW_SHOW_TWITTER_LOGGEDIN);
emitter.trigger(events/* Events */.zW.MENU_REGISTER_AUTHORIZATION);
jQuery('.gm-btn-close', this.$dialog).trigger('click');
return;
}
jQuery('#error-msg', this.$dialog).text(ret.reason);
jQuery('.gm-toggle', this.$dialog).fadeOut(() => jQuery('.error', this.$dialog).fadeIn());
this.twitter.account.setCSRFToken('');
config/* default */.Z.ct0 = undefined;
})
.finally(() => jQuery('#loading-overlay', this.$dialog).fadeOut());
}
initView() {
this.$dialog = jQuery('#twitter-ct0').fadeIn(200);
jQuery('#gm-verify-btn', this.$dialog).on('click', this.trigger(events/* Events */.zW.TWITTER_VERIFY));
}
}
class TwitterLoggedInDialog extends TwitterDialog {
registerListeners() {
emitter.on(events/* Events */.zW.VIEW_SHOW_TWITTER_LOGGEDIN, this.initView, this);
}
initView() {
this.$dialog = jQuery('#twitter-logged-in').fadeIn(200);
if (!config/* default */.Z.user_info) {
jQuery('#loading-overlay', this.$dialog).fadeIn(200);
(0,twitter/* TwitterAPI */.I)()
.account.getSelf()
.then(ret => {
if (isError(ret))
return;
this.setUserInfo(ret);
})
.finally(() => jQuery('#loading-overlay', this.$dialog).fadeOut());
}
this.setUserInfo(config/* default */.Z.user_info);
}
setUserInfo(user) {
jQuery('#user-avatar', this.$dialog).attr('src', user.profileImage);
jQuery('#user-name', this.$dialog).text(user.fullName);
jQuery('#user-id', this.$dialog).text(`@${user.userName}`);
}
}
class TwitterGuestDialog extends TwitterDialog {
registerListeners() {
emitter.on(events/* Events */.zW.VIEW_SHOW_TWITTER_GUEST, this.initView, this);
}
initView() {
this.$dialog = jQuery('#twitter-guest').fadeIn(200);
config/* default */.Z.as_guest = true;
jQuery('.gm-btn-login', this.$dialog).on('click', () => {
GM_openInTab(TWITTER_LOGIN_URL, false);
this.showCT0View();
});
jQuery('.gm-btn-logged', this.$dialog).on('click', () => {
this.showCT0View();
});
}
showCT0View() {
config/* default */.Z.as_guest = false;
this.$dialog.fadeOut(this.trigger(events/* Events */.zW.VIEW_SHOW_TWITTER_CT0));
}
}
emitter.on(events/* Events */.zW.TWITTER_REAUTHORIZE, () => {
// eslint-disable-next-line no-alert
alert('Twitter 授权错误或过期,请重新授权!');
delete config/* default */.Z.access_token;
emitter.trigger(events/* Events */.zW.VIEW_SHOW_TWITTER_GUEST);
emitter.trigger(events/* Events */.zW.MENU_REGISTER_AUTHORIZATION);
});
function isError(obj) {
return obj.error !== undefined;
}
;// CONCATENATED MODULE: ./src/controllers/inedx.ts
const controllers = [
TwitterCT0Dialog,
TwitterGuestDialog,
TwitterLoggedInDialog,
];
function registerControllers() {
controllers.forEach(Controller => new Controller());
}
// EXTERNAL MODULE: ./src/handlers/handler.ts
var handler = __webpack_require__(537);
;// CONCATENATED MODULE: ./src/handlers/index.ts
/* harmony default export */ const handlers = (handler/* default */.Z);
// EXTERNAL MODULE: ./src/managers/cache.ts
var cache = __webpack_require__(944);
;// CONCATENATED MODULE: ./src/menu/menu.ts
const menu_emitter = (0,events/* defaultEventEmitter */.sd)();
let GM_authorization_menu_id;
const menu = {
authorization() {
function isAuthorized() {
return config/* default */.Z && config/* default */.Z.user_info !== undefined && config/* default */.Z.ct0;
}
if (GM_authorization_menu_id !== undefined) {
GM_unregisterMenuCommand(GM_authorization_menu_id);
}
if (isAuthorized()) {
GM_authorization_menu_id = GM_registerMenuCommand(`✔️登录用户:${config/* default */.Z.user_info.fullName} (@${config/* default */.Z.user_info.userName})`, () => menu_emitter.trigger(events/* Events */.zW.VIEW_SHOW_TWITTER_LOGGEDIN));
}
else if (!(config/* default */.Z === null || config/* default */.Z === void 0 ? void 0 : config/* default */.Z.ct0)) {
GM_authorization_menu_id = GM_registerMenuCommand(`⭕点击输入 ct0`, () => menu_emitter.trigger(events/* Events */.zW.VIEW_SHOW_TWITTER_CT0));
}
else {
GM_authorization_menu_id = GM_registerMenuCommand(`❌未登录,点击登录`, () => menu_emitter.trigger(events/* Events */.zW.VIEW_SHOW_TWITTER_GUEST));
}
},
clearCache() {
GM_registerMenuCommand('清除脚本缓存(解决莫名其妙的加载问题)', () => {
cache/* default */.Z.clear();
const $success = jQuery(`<div class="gm-dialog"><div class="success-box gm-dialog"><p>操作成功!</p></div></div>`);
$success.hide().appendTo(document.body).fadeIn(200);
setTimeout(() => $success.fadeOut(() => $success.remove()), 1400);
});
},
reportIssue() {
GM_registerMenuCommand('报告问题', () => {
GM_openInTab('https://github.com/bigbowl-wtw/SouthPlusVideoPlayer/issues', true);
});
},
};
menu_emitter.on(events/* Events */.zW.MENU_REGISTER_AUTHORIZATION, menu.authorization);
function registerMenuCommand() {
for (const register of Object.values(menu))
register();
}
// EXTERNAL MODULE: ./src/utils/utils.ts + 3 modules
var utils = __webpack_require__(803);
// EXTERNAL MODULE: ./src/update-information.txt
var update_information = __webpack_require__(264);
var update_information_default = /*#__PURE__*/__webpack_require__.n(update_information);
;// CONCATENATED MODULE: ./src/view/ct0.html
// Module
var code = "<div class=\"gm-dialog\" id=\"twitter-ct0\"> <h1>Twitter 账户</h1> <div id=\"pin-verifier\" class=\"gm-toggle\"> <p>你已经在 Twitter 登录,请填入你的 ct0 以在本站使用 Twitter API。</p> <p>获取 ct0 的方法请参见 <a href=\"/read.php?tid=1899700#26504410\" target=\"_blank\">这里</a></p> <label for=\"pin\" style=\"font-size:.8em\">ct0:</label> <input class=\"pin\" type=\"text\" placeholder=\"输入 ct0\" id=\"pin\"> <button id=\"gm-verify-btn\" class=\"gm-btn gm-btn-primary\">验证</button> </div> <div class=\"error\" style=\"display:none\"> <p>发生了下面的错误:</p> <p id=\"error-msg\"></p> </div> <p>风险提示:由于使用未经 Twitter 公布的 API,可能会导致 Twitter 账户被封禁,请谨慎考虑。</p> <p>所有数据都保存在本地,脚本不会向任何第三方发送数据。</p> <div class=\"gm-dialog-footer\"> <button class=\"gm-btn gm-btn-danger error\">重新打开</button> <button class=\"gm-btn gm-btn-secondary gm-btn-close\">关闭</button> </div> <div id=\"loading-overlay\" class=\"loading-overlay\" style=\"display:none\"> <div class=\"loading-spinner\"></div> </div> </div>";
// Exports
/* harmony default export */ const ct0 = (code);
;// CONCATENATED MODULE: ./src/view/guest.html
// Module
var guest_code = "<div class=\"gm-dialog\" id=\"twitter-guest\"> <h1>Twitter 账户</h1> <p class=\"gm-toggle\">没有登录 Twitter 账户</p> <p>Twitter API 将使用访客授权,可能会受到 NSFW 内容限制</p> <p>如果你之前已经在 Twitter 登录,点击“已经登录”</p> <button class=\"gm-btn gm-btn-primary gm-btn-login\">去登录</button> <button class=\"gm-btn gm-btn-primary gm-btn-logged\">已经登录</button> <div class=\"gm-dialog-footer\"> <button class=\"gm-btn gm-btn-secondary gm-btn-close\">关闭</button> </div> </div> ";
// Exports
/* harmony default export */ const guest = (guest_code);
;// CONCATENATED MODULE: ./src/view/logged-in.html
// Module
var logged_in_code = "<div class=\"gm-dialog\" id=\"twitter-logged-in\"> <h1>Twitter 账户</h1> <div id=\"user-profile\" class=\"user-profile\"> <label for=\"user-profile\" style=\"margin-right:5px\">登录账户:</label> <div class=\"avatar\"> <img id=\"user-avatar\" src=\"\" alt=\"User Avatar\"/> </div> <div class=\"user-details\"> <div id=\"user-name\"></div> <div id=\"user-id\"></div> </div> </div> <p>风险提示:由于使用为未经 Twitter 公布的 API,可能会导致 Twitter 账户被封禁,请谨慎考虑。</p> <p>要使用访客身份使用本脚本,请在 Twitter 退出登录。</p> <p>所有数据都保存在本地,脚本不会向任何第三方发送数据。</p> <div class=\"gm-dialog-footer\"> <button class=\"gm-btn gm-btn-secondary gm-btn-close\">关闭</button> </div> <div id=\"loading-overlay\" class=\"loading-overlay\" style=\"display:none\"> <div class=\"loading-spinner\"></div> </div> </div>";
// Exports
/* harmony default export */ const logged_in = (logged_in_code);
;// CONCATENATED MODULE: ./src/view/style.css
// Module
var style_code = ".gm-dialog { position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); background-color: #fff; padding: 20px; border-radius: 5px; box-shadow: 0 2px 8px rgba(0, 0, 0, 0.3); width: 400px; } .gm-btn { border: 1px solid; padding: 5px 16px; border-radius: 6px; transition: background-color 0.3s ease; line-height: 20px; } .gm-btn:hover { cursor: pointer; } .gm-btn-primary { background-color: #1f883d; color: #fff; margin-left: 10px; } .gm-btn-primary:hover { background-color: #0f672b; } .gm-btn-secondary { background-color: #6c757d; color: #fff; } .gm-btn-secondary:hover { background-color: #5a6268; } .gm-btn-danger { background-color: #dc3545; color: #fff; margin-left: auto; margin-right: 10px; } .gm-btn-danger:hover { background-color: #bd2130; } .gm-dialog-footer { text-align: right; margin-top: 20px; } .pin { font-size: 14px; line-height: 20px; padding: 5px 12px; border: 1px solid #d0d7de; border-radius: 6px; box-shadow: inset 0 1px 0 rgba(208, 215, 222, 0.2); transition: background-color 0.3s ease; transition-property: color, background-color, box-shadow, border-color; } .pin:hover { background-color: #f6f7fa; } .loading-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.5); z-index: 9999; display: flex; justify-content: center; align-items: center; } .loading-spinner { border: 5px solid rgba(255, 255, 255, 0.3); border-top: 5px solid #fff; border-radius: 50%; width: 30px; height: 30px; animation: spin 1s linear infinite; } @keyframes spin { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } } .user-profile { display: flex; align-items: center; margin-bottom: 20px; } .avatar { width: 50px; height: 50px; border-radius: 50%; margin-left: 15px; overflow: hidden; } .avatar img { width: 100%; height: 100%; object-fit: cover; } .user-details { height: 50px; display: flex; flex-direction: column; justify-content: center; margin-left: 10px; } .user-name { font-weight: bold; font-size: 16px; } .user-id { font-size: 14px; color: #666; } .success-box { display: flex; justify-content: center; align-items: center; width: 200px; height: 50px; background-color: #4caf50; color: white; font-size: 25px; font-weight: bold; border-radius: 20px; } ";
// Exports
/* harmony default export */ const style = (style_code);
;// CONCATENATED MODULE: ./src/view/update-information.html
// Module
var update_information_code = "<div class=\"gm-dialog\" id=\"update-information\"> <h1>更新信息</h1> {{ content }} <div class=\"gm-dialog-footer\"> <button class=\"gm-btn gm-btn-secondary gm-btn-close\">关闭</button> <button class=\"gm-btn gm-btn-primary gm-btn-open\">打开</button> </div> </div>; ";
// Exports
/* harmony default export */ const view_update_information = (update_information_code);
;// CONCATENATED MODULE: ./src/view/index.ts
// eslint-disable-next-line prettier/prettier
const views = [
ct0,
logged_in,
guest,
view_update_information.replace('{{ content }}', update_information_default().replace('\r', '')
.split('\n')
.map(c => `<p>${c}</p>`)
.join('')),
];
const INTRO_POST = `${document.location.origin}/read.php?tid=1899700`;
const view_emitter = (0,events/* defaultEventEmitter */.sd)();
function initViews() {
GM_addStyle(style);
views.forEach(view => {
jQuery(view).hide().appendTo(document.body);
});
jQuery('#update-information')
.filter('.gm-btn-open')
.on('click', () => GM_openInTab(INTRO_POST, false));
jQuery('.gm-btn-close').on('click', ({ target }) => jQuery(target).parents('.gm-dialog').fadeOut());
view_emitter.on(events/* Events */.zW.SHOW_UPDATE_INFORMATION, () => {
jQuery('#update-information').fadeIn();
});
}
;// CONCATENATED MODULE: ./src/index.ts
// import cache from './managers/cache';
// import config from './managers/config';
const NOTIFY = false;
(0,utils/* isUpdated */._C)(() => {
if (NOTIFY)
(0,events/* defaultEventEmitter */.sd)().trigger(events/* Events */.zW.SHOW_UPDATE_INFORMATION);
// v5.0.0: 更新了 twitter api,采用新的授权方式,清除旧数据
// delete config.user_info;
// delete config.access_token;
// cache.clear();
// v5.0.0: 检测是否登录
// defaultEventEmitter().trigger(Events.VIEW_SHOW_TWITTER_GUEST);
});
registerMenuCommand();
initViews();
registerControllers();
handlers.process();
})();
/******/ })()
;