Created
April 3, 2022 21:03
-
-
Save disco0/49ab09cc90355f0ed0a56b09ebf9b635 to your computer and use it in GitHub Desktop.
TwitterLiveFeed
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
///<reference types="typed-query-selector"/> | |
///<reference types="typed-query-selector/parser"/> | |
///<reference path="./xpath-text-search.d.ts"/> | |
///<reference types="mousetrap"/> | |
// ==UserScript== | |
// @name TwitterLiveFeed | |
// @author disk0 | |
// @description Force Twitter feed to display latest tweets | |
// @version 0.2.1 | |
// @run-at document-idle | |
// @namespace disc0/github/TwitterLiveFeed | |
// @icon  | |
// @match https://*.twitter.com/* | |
// @match https://twitter.com/* | |
// @grant GM_info | |
// @grant GM_addStyle | |
// @grant GM_getValue | |
// @grant GM_setValue | |
// @grant GM_registerMenuCommand | |
// @grant GM_unregisterMenuCommand | |
// @require http://ajax.googleapis.com/ajax/libs/jquery/1.7.2/jquery.min.js | |
// @require https://gist.github.com/raw/2625891/waitForKeyElements.js | |
// ==/UserScript== | |
// Simple | |
//(()=>{let d=document,qs=(s,o=d)=>o.querySelector(s);[qs(`[aria-label="Top Tweets on"]`)].forEach(_=>{_.click();qs(`[role="menu"] [role="menuitem"]`).click();})})() | |
// Overkill | |
(async () => { | |
function isHTMLElement(obj) { return (!!obj) && obj instanceof HTMLElement; } | |
//#region Log | |
class StyledLog { | |
headerText; | |
headerArgs = []; | |
get headerSpread() { return [this.headerText, ...this.headerArgs]; } | |
get header() { return { text: this.headerText, args: this.headerArgs }; } | |
constructor(headerText = '', ...headerArgs) { | |
this.headerText = headerText; | |
Object.assign(this, { headerArgs }); | |
} | |
info = (...msg) => console.info(...(msg.length === 0) | |
? this.headerSpread : | |
(typeof msg[0] === 'string') | |
? [`${this.header.text} ${msg[0]}`, ...this.header.args, ...(msg.length > 1) ? msg.slice(1) : []] | |
: [...this.headerSpread, ...msg]); | |
log = (...msg) => console.log(...(msg.length === 0) | |
? this.headerSpread : | |
(typeof msg[0] === 'string') | |
? [`${this.header.text} ${msg[0]}`, ...this.header.args, ...(msg.length > 1) ? msg.slice(1) : []] | |
: [...this.headerSpread, ...msg]); | |
debug = (...msg) => console.debug(...(msg.length === 0) | |
? this.headerSpread : | |
(typeof msg[0] === 'string') | |
? [`${this.header.text} ${msg[0]}`, ...this.header.args, ...(msg.length > 1) ? msg.slice(1) : []] | |
: [...this.headerSpread, ...msg]); | |
warn = (...msg) => console.warn(...(msg.length === 0) | |
? this.headerSpread : | |
(typeof msg[0] === 'string') | |
? [`${this.header.text} ${msg[0]}`, ...this.header.args, ...(msg.length > 1) ? msg.slice(1) : []] | |
: [...this.headerSpread, ...msg]); | |
} | |
const log = new StyledLog(`%c[TwitterLiveFeed]%c`, `background: #44AAFF; color: white; padding: 0.2em 0.25em; margin-bottom: 0.15em; border-radius: 0.15em;`, ''); | |
function consoleGroupScope(...args) { | |
let fn, name; | |
if (typeof args[0] === 'string') { | |
[name, fn] = args; | |
} | |
else { | |
[fn, name = ""] = args; | |
} | |
try { | |
console.group(name); | |
fn(); | |
} | |
catch (error) { | |
console.warn(`Caught in console group wrapper: ${error}`); | |
} | |
finally { | |
console.groupEnd(); | |
} | |
} | |
//#region Userscript Info | |
class Script { | |
static namespace = "TwitterUtil"; | |
static settingsname = "TwitterUtil"; | |
//#region GM Value Controls | |
static AutoFixDisabledValueKey = 'autofix-disabled'; | |
static get AutoFixDisabledState() { | |
return GM_getValue(Script.AutoFixDisabledValueKey, false); | |
} | |
static ToggleDisableLiveFeedUpdate() { | |
log.debug(`Toggling automatic fixes`); | |
GM_setValue(Script.AutoFixDisabledValueKey, !Script.AutoFixDisabledState); | |
log.info(`Redrawing menu`); | |
Script.GM_Menubar_AutoFixState.update(); | |
} | |
static FixTotalValueKey = 'autofix-total'; | |
static get FixTotal() { | |
// No value returned implies no past changes | |
return ((value = 0) => { | |
// Idiot check | |
if (Number.isInteger(value)) | |
return value; | |
// Write warning if invalid | |
log.warn(`Read invalid value from storage (value: ${typeof value} => ${String(value)})`); | |
return 0; | |
})(GM_getValue(Script.FixTotalValueKey, 0)); | |
} | |
static IncrementFixTotal() { | |
const value = Script.FixTotal + 1; | |
GM_setValue(Script.FixTotalValueKey, value); | |
log.info(`Total timeline adjustments: ${value}`); | |
log.info(`Redrawing menu`); | |
Script.GM_Menubar_FixTotalView.update(); | |
} | |
//#endregion GM Value Controls | |
//#region GM Menubar Items | |
static GM_Menubar_AutoFixState = new class { | |
menuItemId; | |
constructor() { | |
this.menuItemId = this.update(); | |
} | |
update() { | |
// Remove existing display on repeat updates | |
if (typeof this.menuItemId === 'number') | |
GM_unregisterMenuCommand(this.menuItemId); | |
this.menuItemId = GM_registerMenuCommand((Script.AutoFixDisabledState | |
? `Disabled` | |
: `Enabled`), () => log.debug(`Info menu item no-op`)); | |
return this.menuItemId; | |
} | |
}; | |
static GM_Menubar_ToggleAutoFixState = new class { | |
menuItemId = GM_registerMenuCommand('Toggle automatic feed fixes', () => Script.ToggleDisableLiveFeedUpdate()); | |
}; | |
//#endregion GM Menubar Items | |
static GM_Menubar_FixTotalView = new class { | |
menuItemId; | |
constructor() { | |
this.menuItemId = this.update(); | |
} | |
update() { | |
// Remove existing display on repeat updates | |
if (typeof this.menuItemId === 'number') | |
GM_unregisterMenuCommand(this.menuItemId); | |
this.menuItemId = GM_registerMenuCommand(`Total Feed Fixes: ${Script.FixTotal}`, () => log.debug(`Info menu item no-op`)); | |
return this.menuItemId; | |
} | |
}; | |
static async run() { | |
// var menuHide = GM_registerMenuCommand( 'Force Twitter Live Feed', | |
// Main.toggleForceliveFeed ); | |
// consoleGroupScope('TwitterLiveFeed', async () => { await checkPage() }) | |
checkPage(); | |
} | |
} | |
//#endregion Userscript Info | |
const checkPage = async () => { | |
// Element selectors used to target/navigate page | |
const headerSel = `[data-testid="primaryColumn"] h2[aria-level="2"][role="heading"]`; | |
const menuSel = `div[aria-label="Top Tweets on"]`; | |
const buttonSel = `#layers [role="menu"] div[role="menuitem"]`; | |
// Store clicked state to avoid phantom second click | |
let clicked = false; | |
const clickLog = (...args) => { | |
const msg = typeof args[0] === 'string' | |
? args[0] | |
: 'Clicking element:'; | |
const el = args.slice(-1)[0]; | |
log.log(`${msg}\n%o`, el); | |
el.click(); | |
}; | |
const clickTimelineButton = async (el) => { | |
if (clicked) { | |
log.info(`Skipping clicking timeline menu button, clicked state has already been set.`); | |
return; | |
} | |
try { | |
clickLog(`Clicking timeline menu button:`, el); | |
clicked = true; | |
Script.IncrementFixTotal(); | |
} | |
catch (error) { | |
log.warn(`Error clicking timeline menu button element:\n%o`, error); | |
} | |
finally { | |
try { | |
const buttonEl = await waitSelector(buttonSel); | |
if (!buttonEl) { | |
log.warn(`Timeline update button not found`); | |
return; | |
} | |
clickLog(`Clicking latest tweets view button:`, buttonEl); | |
} | |
catch (error) { | |
log.warn(`Latest tweets view apply button not found. Error:\n%o`, error); | |
} | |
} | |
}; | |
const waitSelector = function (selector, timeout = 2000) { | |
return new Promise((resolve, reject) => { | |
const startTimeout = () => setTimeout(() => reject(), timeout); | |
if (typeof selector === 'undefined') | |
startTimeout(); | |
const waitForElement = function () { | |
const element = document.querySelector(selector); | |
if (element) | |
resolve(element); | |
else | |
window.requestAnimationFrame(waitForElement); | |
}; | |
startTimeout(); | |
waitForElement(); | |
}); | |
}; | |
const updateTimeline = async (headerEl = document.querySelector(headerSel)) => { | |
log.debug(`Heading Element:\n`, headerEl); | |
if (headerEl.textContent !== `Latest Tweets`) { | |
try { | |
const timelineButtonEl = await waitSelector(menuSel); | |
if (!timelineButtonEl) { | |
log.warn(`No value received from waitSelector for timeline element.`); | |
return; | |
} | |
else if (!isHTMLElement(timelineButtonEl)) { | |
log.warn(`Returned timeline element value is not an HTMLElement instance.`); | |
return; | |
} | |
else | |
clickTimelineButton(timelineButtonEl); | |
} | |
catch (error) { | |
log.warn(`Error thrown while searching for timeline button: ${error}`); | |
return; | |
} | |
} | |
else { | |
log.debug(`Already viewing latest tweets`); | |
return; | |
} | |
}; | |
try { | |
log.debug(`Waiting to check for header selector`); | |
const headerEl = await waitSelector(headerSel); | |
if (!headerEl) | |
return; | |
await updateTimeline(headerEl); | |
} | |
catch (error) { | |
log.warn(`Error thrown, timeline header element not found:\n%o`, error); | |
} | |
}; | |
await Script.run(); | |
})(); | |
//# sourceMappingURL=TwitterLiveFeed.js.map |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment