Last active
April 12, 2018 16:43
-
-
Save Elemecca/8e5f7746ca19d199ae2158d59afcc941 to your computer and use it in GitHub Desktop.
User script for the Discord web client that makes all embeds collapsible.
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
// ==UserScript== | |
// @name Discord Collapsible Embeds | |
// @namespace maltera.net | |
// @version 1.0 | |
// @downloadURL https://gist.githubusercontent.com/Elemecca/8e5f7746ca19d199ae2158d59afcc941/raw/discord-collapse.user.js | |
// @description Makes all embeds collapsible in the Discord web client. | |
// @author Sam Hanes <[email protected]> | |
// @match https://discordapp.com/channels/* | |
// @run-at document-start | |
// @grant none | |
// ==/UserScript== | |
(function() { | |
"use strict"; | |
const elementKey = Symbol.for("react.element"); | |
const fragmentKey = Symbol.for("react.fragment"); | |
const arrowUp = ""; | |
const arrowDown = ""; | |
let React; | |
/** Walks an element tree calling a visitor function for each element. | |
* The tree will be walked depth-first. If the visitor returns a truthy | |
* value walking will stop and that value will be returned immediately. | |
* | |
* @param element the root of the element tree to walk | |
* @param visitor a function that will be called with each element | |
* @return the first truthy value returned by the visitor | |
*/ | |
function visitElements(element, visitor) { | |
let result; | |
if (_.isArray(element)) { | |
for (let value of element) { | |
if ((result = visitElements(value, visitor))) return result; | |
} | |
} else if (_.isObject(element) && elementKey === element.$$typeof) { | |
if ((result = visitor(element))) { | |
return result; | |
} else if (element.props && element.props.children) { | |
return visitElements(element.props.children, visitor); | |
} | |
} | |
return null; | |
} | |
function patchMessage(Message) { | |
console.info("monkey-patching Message", {Message}); | |
const P = Message.prototype; | |
const render = P.render; | |
P.render = function() { | |
const result = render.apply(this, arguments); | |
const msg = this.props.message; | |
if (msg.embeds.length === 0 && msg.attachments.length === 0) return result; | |
visitElements(result, (elem) => { | |
if ("message-text" === elem.props.className) { | |
const show = this.props.renderEmbeds; | |
elem.props.children.splice(0, 0, ( | |
React.createElement( | |
"div", | |
{ className: "btn-option", | |
style: { | |
color: 'black', | |
backgroundImage: "url(" + (show ? arrowUp : arrowDown) + ")", | |
visibility: (show ? undefined : "visible"), | |
}, | |
onClick: this.props.xxToggleEmbed, | |
}, | |
[] | |
) | |
)); | |
} | |
}); | |
return result; | |
}; | |
} | |
function patchMessageGroup(MessageGroup) { | |
console.info("monkey-patching MessageGroup", {MessageGroup}); | |
const P = MessageGroup.prototype; | |
if (!MessageGroup.displayName) { | |
MessageGroup.displayName = "MessageGroup"; | |
} | |
// updates the show state to include new messages in the props | |
function updateShowStateForProps(props) { | |
const oldShowState = this.state.xxShowEmbeds || {}; | |
const newShowState = {}; | |
for (let msg of props.messages) { | |
newShowState[msg.id] = | |
(_.isNil(oldShowState[msg.id]) ? true : oldShowState[msg.id]); | |
} | |
if (!_.isEqual(oldShowState, newShowState)) { | |
this.setState(() => ({ | |
xxShowEmbeds: newShowState, | |
})); | |
} | |
} | |
const componentWillMount = P.componentWillMount; | |
P.componentWillMount = function() { | |
if (componentWillMount) { | |
componentWillMount.apply(this, arguments); | |
} | |
updateShowStateForProps.call(this, this.props); | |
}; | |
const componentWillReceiveProps = P.componentWillReceiveProps; | |
P.componentWillReceiveProps = function (nextProps) { | |
if (componentWillReceiveProps) { | |
componentWillReceiveProps.apply(this, arguments); | |
} | |
updateShowStateForProps.call(this, nextProps); | |
}; | |
// overrides the messages' `renderEmbed` prop based on state | |
const render = P.render; | |
function renderMain() { | |
const result = render.apply(this, arguments); | |
visitElements(result, (elem) => { | |
if (_.isFunction(elem.type) && "Message" === elem.type.displayName) { | |
const id = elem.props.message.id; | |
if (!this.state.xxShowEmbeds[id]) { | |
elem.props.renderEmbeds = false; | |
elem.props.inlineAttachmentMedia = false; | |
} | |
elem.props.xxToggleEmbed = () => { | |
this.setState((prevState) => { | |
const showState = _.clone(prevState.xxShowEmbeds); | |
showState[id] = !showState[id]; | |
return {xxShowEmbeds: showState}; | |
}); | |
}; | |
} | |
}); | |
return result; | |
} | |
// Message isn't exported, so we catch it the first time | |
// it's returned from a MessageGroup and patch it at that point | |
// this runs once and replaces itself once Message is found | |
P.render = function() { | |
const result = render.apply(this, arguments); | |
const Message = visitElements(result, (cand) => ( | |
_.isFunction(cand.type) && "Message" === cand.type.displayName && cand.type | |
)); | |
if (Message) { | |
patchMessage(Message); | |
P.render = renderMain; | |
} | |
// just re-run `render`, it's idempotent | |
return renderMain.apply(this, arguments); | |
}; | |
} | |
/** Called when Webpack loads a module. | |
* @param module the exports of the module being loaded | |
*/ | |
function onModuleLoaded(module) { | |
if (!React && fragmentKey == module.Fragment) { | |
React = module; | |
return; | |
} | |
if ("function" === typeof module) { | |
if (module.defaultProps) { | |
const defs = module.defaultProps; | |
if ("undefined" !== typeof defs.renderEmbedsSmall) { | |
patchMessageGroup(module); | |
} | |
} | |
} | |
} | |
// this hijacks the Webpack module loading machinery | |
// and calls `onModuleLoaded` for each module as it loads | |
let webpackJsonp; | |
Object.defineProperty(window, "webpackJsonp", { | |
enumerable: true, | |
get: function() { | |
return webpackJsonp; | |
}, | |
set: function(value) { | |
const wrapMod = (func) => function(module, internal, require) { | |
func.apply(this, arguments); | |
if (module.exports) { | |
onModuleLoaded(module.exports); | |
} | |
}; | |
webpackJsonp = function(a, mods, c) { | |
if ("function" === typeof mods.map) { | |
mods = mods.map(wrapMod); | |
} else if ("object" === typeof mods) { | |
for (let [key, value] of Object.entries(mods)) { | |
mods[key] = wrapMod(value); | |
} | |
} | |
return value.call(this, a, mods, c); | |
}; | |
}, | |
}); | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment