Skip to content

Instantly share code, notes, and snippets.

@Elemecca
Last active April 12, 2018 16:43
Show Gist options
  • Save Elemecca/8e5f7746ca19d199ae2158d59afcc941 to your computer and use it in GitHub Desktop.
Save Elemecca/8e5f7746ca19d199ae2158d59afcc941 to your computer and use it in GitHub Desktop.
User script for the Discord web client that makes all embeds collapsible.

testing adding a readme

// ==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 <sam@maltera.com>
// @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