Another copy of my dotfiles. Because I don't completely trust GitHub.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

350 lines
20 KiB

  1. /**
  2. * @name PlatformIndicators
  3. * @displayName PlatformIndicators
  4. * @authorId 415849376598982656
  5. * @invite gvA2ree
  6. */
  7. /*@cc_on
  8. @if (@_jscript)
  9. // Offer to self-install for clueless users that try to run this directly.
  10. var shell = WScript.CreateObject("WScript.Shell");
  11. var fs = new ActiveXObject("Scripting.FileSystemObject");
  12. var pathPlugins = shell.ExpandEnvironmentStrings("%APPDATA%\BetterDiscord\plugins");
  13. var pathSelf = WScript.ScriptFullName;
  14. // Put the user at ease by addressing them in the first person
  15. shell.Popup("It looks like you've mistakenly tried to run me directly. \n(Don't do that!)", 0, "I'm a plugin for BetterDiscord", 0x30);
  16. if (fs.GetParentFolderName(pathSelf) === fs.GetAbsolutePathName(pathPlugins)) {
  17. shell.Popup("I'm in the correct folder already.", 0, "I'm already installed", 0x40);
  18. } else if (!fs.FolderExists(pathPlugins)) {
  19. shell.Popup("I can't find the BetterDiscord plugins folder.\nAre you sure it's even installed?", 0, "Can't install myself", 0x10);
  20. } else if (shell.Popup("Should I copy myself to BetterDiscord's plugins folder for you?", 0, "Do you need some help?", 0x34) === 6) {
  21. fs.CopyFile(pathSelf, fs.BuildPath(pathPlugins, fs.GetFileName(pathSelf)), true);
  22. // Show the user where to put plugins in the future
  23. shell.Exec("explorer " + pathPlugins);
  24. shell.Popup("I'm installed!", 0, "Successfully installed", 0x40);
  25. }
  26. WScript.Quit();
  27. @else@*/
  28. module.exports = (() => {
  29. const config = {
  30. info: {
  31. name: "PlatformIndicators",
  32. authors: [
  33. {
  34. name: "Strencher",
  35. discord_id: "415849376598982656",
  36. github_username: "Strencher",
  37. twitter_username: "Strencher3"
  38. }
  39. ],
  40. version: "0.0.5",
  41. description: "Adds indicators for every platform that the user is using. Source code availble on the repo in the 'src' folder.",
  42. github: "https://github.com/Strencher/BetterDiscordStuff/blob/master/PlatformIndicators/APlatformIndicators.plugin.js",
  43. github_raw: "https://raw.githubusercontent.com/Strencher/BetterDiscordStuff/master/PlatformIndicators/APlatformIndicators.plugin.js"
  44. },
  45. changelog: [
  46. {
  47. title: "v0.0.5",
  48. type: "fixed",
  49. items: [
  50. "Thanks to @qwert#1441 for fixing the padding issue in chat messages!",
  51. "I still need ideas where to show all of them at one position that is not next to the username... join my Support server => https://discord.gg/gvA2ree to send me ideas!"
  52. ]
  53. },
  54. {
  55. title: "v0.0.4",
  56. type: "added",
  57. items: [
  58. "2 Attempt to fix conflicts with BetterRoleColors.",
  59. "It'll probably require you to update 2 times because the filename has changed.",
  60. "Bug fixes... styling fixes..."
  61. ]
  62. }
  63. ],
  64. defaultConfig: [
  65. {
  66. type: "switch",
  67. name: "Show in MemberList",
  68. note: "Shows the platform indicators in the memberlist",
  69. id: "showInMemberList",
  70. value: true
  71. },
  72. {
  73. type: "switch",
  74. name: "Show next to username",
  75. note: "Shows the platform indicators next the username in messages.",
  76. id: "showOnMessages",
  77. value: true
  78. },
  79. {
  80. type: "switch",
  81. name: "Show in Dmd List",
  82. note: "Shows the platform indicators in the dm list.",
  83. id: "showInDmsList",
  84. value: true
  85. },
  86. {
  87. type: "switch",
  88. name: "Show next to discord tags",
  89. note: "Shows the platform indicators right next to the discord tag.",
  90. id: "showOnTags",
  91. value: true
  92. },
  93. {
  94. type: "switch",
  95. name: "Ignore Bots",
  96. note: "Ignores the status of bots which is always web anyways.",
  97. id: "ignoreBots",
  98. value: true
  99. },
  100. {
  101. type: "category",
  102. name: "icons",
  103. id: "icons",
  104. settings: [
  105. {
  106. type: "switch",
  107. name: "Web Icon",
  108. note: "Show the Web icon.",
  109. id: "web",
  110. value: true
  111. },
  112. {
  113. type: "switch",
  114. name: "Desktop Icon",
  115. note: "Show the Desktop icon.",
  116. id: "desktop",
  117. value: true
  118. },
  119. {
  120. type: "switch",
  121. name: "Mobile Icon",
  122. note: "Show the Mobile icon.",
  123. id: "mobile",
  124. value: true
  125. }
  126. ]
  127. }
  128. ]
  129. };
  130. //@ts-ignore
  131. const BdApi = window.BdApi;
  132. // @ts-ignore
  133. return !global.ZeresPluginLibrary ? class {
  134. constructor() {
  135. this._config = config;
  136. }
  137. getName() { return config.info.name; }
  138. getAuthor() { return config.info.authors.map(a => a.name).join(", "); }
  139. getDescription() { return config.info.description; }
  140. getVersion() { return config.info.version; }
  141. load() {
  142. BdApi.showConfirmationModal("Library plugin is needed", [`The library plugin needed for ${config.info.name} is missing. Please click Download Now to install it.`], {
  143. confirmText: "Download",
  144. cancelText: "Cancel",
  145. onConfirm: () => {
  146. require("request").get("https://rauenzi.github.io/BDPluginLibrary/release/0PluginLibrary.plugin.js", async (error, response, body) => {
  147. if (error)
  148. return require("electron").shell.openExternal("https://betterdiscord.net/ghdl?url=https://raw.githubusercontent.com/rauenzi/BDPluginLibrary/master/release/0PluginLibrary.plugin.js");
  149. await new Promise(r => require("fs").writeFile(require("path").join(BdApi.Plugins.folder, "0PluginLibrary.plugin.js"), body, r));
  150. });
  151. }
  152. });
  153. }
  154. start() { }
  155. stop() { }
  156. } : (([Plugin, Api]) => {
  157. const plugin = (Plugin, Api) => {
  158. const { Utilities, WebpackModules, PluginUtilities, ReactTools, Patcher, Logger, DiscordModules: { React, UserStatusStore, Dispatcher, DiscordConstants: { ActionTypes } } } = Api;
  159. const Utils = Object.assign(Utilities, {
  160. joinClassNames: (...classNames) => classNames.filter(Boolean).join(" "),
  161. capFirst(text) {
  162. return text[0].toUpperCase() + text.slice(1);
  163. }
  164. });
  165. const DesktopIcon = React.memo(props => (React.createElement("svg", Object.assign({ className: "PI-icon_desktop", width: "24", height: "24" }, props, { viewBox: "0 0 24 24" }),
  166. React.createElement("path", { fill: "currentColor", d: "M4 2.5C2.897 2.5 2 3.397 2 4.5V15.5C2 16.604 2.897 17.5 4 17.5H11V19.5H7V21.5H17V19.5H13V17.5H20C21.103 17.5 22 16.604 22 15.5V4.5C22 3.397 21.103 2.5 20 2.5H4ZM20 4.5V13.5H4V4.5H20Z" }))));
  167. const WebIcon = React.memo(props => (React.createElement("svg", Object.assign({ className: "PI-icon_web", width: "24", height: "24" }, props, { viewBox: "0 0 24 24" }),
  168. React.createElement("path", { fill: "currentColor", d: "M12 2C6.48 2 2 6.48 2 12C2 17.52 6.48 22 12 22C17.52 22 22 17.52 22 12C22 6.48 17.52 2 12 2ZM11 19.93C7.05 19.44 4 16.08 4 12C4 11.38 4.08 10.79 4.21 10.21L9 15V16C9 17.1 9.9 18 11 18V19.93ZM17.9 17.39C17.64 16.58 16.9 16 16 16H15V13C15 12.45 14.55 12 14 12H8V10H10C10.55 10 11 9.55 11 9V7H13C14.1 7 15 6.1 15 5V4.59C17.93 5.78 20 8.65 20 12C20 14.08 19.2 15.97 17.9 17.39Z" }))));
  169. const MobileIcon = React.memo(props => (React.createElement("svg", Object.assign({ className: "PI-icon_mobile", width: "24", height: "24" }, props, { viewBox: "0 0 24 24" }),
  170. React.createElement("g", { fill: "none" },
  171. React.createElement("path", { fill: "currentColor", d: "M15.5 1h-8C6.12 1 5 2.12 5 3.5v17C5 21.88 6.12 23 7.5 23h8c1.38 0 2.5-1.12 2.5-2.5v-17C18 2.12 16.88 1 15.5 1zm-4 21c-.83 0-1.5-.67-1.5-1.5s.67-1.5 1.5-1.5 1.5.67 1.5 1.5-.67 1.5-1.5 1.5zm4.5-4H7V4h9v14z" })))));
  172. const Icons = {
  173. mobile: MobileIcon,
  174. web: WebIcon,
  175. desktop: DesktopIcon
  176. };
  177. const getClass = (props = [], items = props, exclude = [], selector = false) => {
  178. const module = WebpackModules.getModule(m => m && props.every(prop => m[prop] !== undefined) && exclude.every(e => m[e] == undefined));
  179. if (!module)
  180. return '';
  181. return (selector ? '.' : '') + items.map(item => module[item]).join(selector ? '.' : ' ');
  182. };
  183. const { TooltipContainer: Tooltip } = WebpackModules.getByProps("TooltipContainer");
  184. const StatusModule = WebpackModules.getByProps("Status", "getStatusMask");
  185. const Flux = WebpackModules.getByProps("connectStores");
  186. const MessageTimestamp = WebpackModules.getByProps("MessageTimestamp");
  187. const { Messages } = WebpackModules.getByProps("Messages", "setLocale");
  188. const AuthStore = WebpackModules.getByProps("getId", "getEmail");
  189. let plugin, currentClientStatus;
  190. const StatusIndicators = function StatusIndicators(props) {
  191. if (!props)
  192. return null;
  193. return (React.createElement("div", { className: Utils.joinClassNames("PI-indicatorContainer", "PI-type_" + props.type) }, Object.keys(props).filter(e => plugin.settings.icons[e]).map(e => {
  194. const color = StatusModule.getStatusColor(props[e]);
  195. const Icon = Icons[e];
  196. return React.createElement(Tooltip, { text: Utils.capFirst(e) + ": " + Messages[`STATUS_${(props[e] == "mobile" ? "mobile_online" : props[e]).toUpperCase()}`], position: "top" },
  197. React.createElement(Icon, { style: { color }, width: "18", height: "18" }));
  198. })));
  199. };
  200. return class PlatformIndicators extends Plugin {
  201. constructor() {
  202. super(...arguments);
  203. this.css = `
  204. .PI-indicatorContainer {
  205. display: inline-flex;
  206. }
  207. .PI-indicatorContainer svg {
  208. margin-left: 2px;
  209. }
  210. .header-23xsNx {
  211. display: flex !important;
  212. flex-direction: row !important;
  213. }
  214. .PI-container {
  215. display: flex;
  216. }
  217. `;
  218. this.getSettingsPanel = () => {
  219. return this.buildSettingsPanel().getElement();
  220. };
  221. this.ON_PRESENCE_UPDATE = ({ user, clientStatus }) => {
  222. if (user.id != AuthStore.getId())
  223. return;
  224. currentClientStatus = clientStatus;
  225. UserStatusStore.emitChange();
  226. };
  227. }
  228. getClients(userId) {
  229. const isSelf = userId == AuthStore.getId();
  230. const status = isSelf ? currentClientStatus : UserStatusStore.getState().clientStatuses[userId];
  231. return status !== null && status !== void 0 ? status : {};
  232. }
  233. onStart() {
  234. plugin = this;
  235. PluginUtilities.addStyle(config.info.name, this.css);
  236. Utils.suppressErrors(this.patchMessageHeader.bind(this))();
  237. Utils.suppressErrors(this.patchMemberListItem.bind(this))();
  238. Utils.suppressErrors(this.patchDmList.bind(this))();
  239. Utils.suppressErrors(this.patchDiscordTag.bind(this))();
  240. Dispatcher.subscribe(ActionTypes.PRESENCE_UPDATE, this.ON_PRESENCE_UPDATE);
  241. }
  242. async patchMemberListItem() {
  243. const MemberListItem = WebpackModules.getByDisplayName("MemberListItem");
  244. Patcher.after(MemberListItem.prototype, "renderDecorators", ({ props }, _, returnValue) => {
  245. var _a;
  246. if (!this.settings.showInMemberList)
  247. return;
  248. try {
  249. const tree = (_a = returnValue === null || returnValue === void 0 ? void 0 : returnValue.props) === null || _a === void 0 ? void 0 : _a.children;
  250. if (!Array.isArray(tree) || (this.settings.ignoreBots && props.user.bot))
  251. return;
  252. const FluxWrapper = Flux.connectStores([UserStatusStore], () => this.getClients(props.user.id))(clients => React.createElement(StatusIndicators, Object.assign({}, clients, { type: "memberList" })));
  253. tree.unshift(React.createElement(FluxWrapper, null));
  254. }
  255. catch (error) {
  256. Logger.error("Error while patching MemberListItem:", error);
  257. }
  258. });
  259. this.forceUpdate(getClass(["member"], ["member"], [], true));
  260. }
  261. patchMessageHeader() {
  262. Patcher.after(MessageTimestamp, "default", (_, [props], returnValue) => {
  263. if (!this.settings.showOnMessages)
  264. return;
  265. try {
  266. const tree = Utils.getNestedProp(returnValue, "props.children.1.props.children");
  267. if (!Array.isArray(tree) || (this.settings.ignoreBots && props.message.author.bot))
  268. return;
  269. const FluxWrapper = Flux.connectStores([UserStatusStore], () => this.getClients(props.message.author.id))(clients => React.createElement(StatusIndicators, Object.assign({}, clients, { type: "chat" })));
  270. tree.splice(2, 0, React.createElement(FluxWrapper, null));
  271. }
  272. catch (error) {
  273. Logger.error("Error while patching MessageTimestammp:", error);
  274. }
  275. });
  276. }
  277. patchDmList() {
  278. var _a;
  279. const { default: PrivateChannel } = (_a = WebpackModules.getModule(m => { var _a; return ((_a = m === null || m === void 0 ? void 0 : m.default) === null || _a === void 0 ? void 0 : _a.displayName) === "PrivateChannel"; })) !== null && _a !== void 0 ? _a : {};
  280. Patcher.after(PrivateChannel.prototype, "render", (_this, _, ret) => {
  281. const unpatch = Patcher.after(ret.type, "render", (_, __, ret) => {
  282. var _a, _b;
  283. unpatch();
  284. if (!this.settings.showInDmsList)
  285. return;
  286. const tree = Utils.findInReactTree(ret, m => { var _a; return ((_a = m === null || m === void 0 ? void 0 : m.className) === null || _a === void 0 ? void 0 : _a.indexOf("nameAndDecorators")) > -1; });
  287. if (!tree)
  288. return;
  289. if (!Array.isArray(tree === null || tree === void 0 ? void 0 : tree.children) || (this.settings.ignoreBots && ((_b = (_a = _this.props) === null || _a === void 0 ? void 0 : _a.user) === null || _b === void 0 ? void 0 : _b.bot)))
  290. return;
  291. const FluxWrapper = Flux.connectStores([UserStatusStore], () => { var _a, _b; return this.getClients((_b = (_a = _this.props) === null || _a === void 0 ? void 0 : _a.user) === null || _b === void 0 ? void 0 : _b.id); })(clients => React.createElement(StatusIndicators, Object.assign({}, clients, { type: "dmList" })));
  292. tree.children = [
  293. tree.children,
  294. React.createElement(FluxWrapper, null)
  295. ];
  296. });
  297. });
  298. this.forceUpdate(getClass(["privateChannels"], ["privateChannels"], [], true));
  299. }
  300. forceUpdate(selector) {
  301. const nodes = document.querySelectorAll(selector);
  302. if (!nodes.length)
  303. return;
  304. for (const node of nodes) {
  305. const instance = ReactTools.getOwnerInstance(node);
  306. if (!instance)
  307. return;
  308. instance.forceUpdate();
  309. }
  310. }
  311. patchDiscordTag() {
  312. const DiscordTag = WebpackModules.getModule(m => { var _a; return ((_a = m === null || m === void 0 ? void 0 : m.default) === null || _a === void 0 ? void 0 : _a.displayName) === "DiscordTag"; });
  313. const NameTag = WebpackModules.getModule(m => { var _a; return ((_a = m === null || m === void 0 ? void 0 : m.default) === null || _a === void 0 ? void 0 : _a.displayName) === "NameTag"; });
  314. Patcher.after(DiscordTag, "default", (_, [{ user }], ret) => {
  315. ret.props.user = user;
  316. });
  317. Patcher.after(NameTag, "default", (_, [args], ret) => {
  318. if (!this.settings.showOnTags)
  319. return;
  320. const tree = ret === null || ret === void 0 ? void 0 : ret.props;
  321. var { user } = args;
  322. if (!Array.isArray(tree === null || tree === void 0 ? void 0 : tree.children) || (this.settings.ignoreBots && (user === null || user === void 0 ? void 0 : user.bot)))
  323. return;
  324. const FluxWrapper = Flux.connectStores([UserStatusStore], () => this.getClients(user === null || user === void 0 ? void 0 : user.id))(clients => React.createElement(StatusIndicators, Object.assign({}, clients, { type: "discordTag" })));
  325. try {
  326. tree.children.push(React.createElement(FluxWrapper, null));
  327. }
  328. catch (error) {
  329. Logger.error("Failed to inject into NameTag:\n", error);
  330. }
  331. return ret;
  332. });
  333. }
  334. onStop() {
  335. Patcher.unpatchAll();
  336. PluginUtilities.removeStyle(config.info.name);
  337. Dispatcher.unsubscribe(ActionTypes.PRESENCE_UPDATE, this.ON_PRESENCE_UPDATE);
  338. }
  339. };
  340. };
  341. return plugin(Plugin, Api);
  342. //@ts-ignore
  343. })(global.ZeresPluginLibrary.buildPlugin(config));
  344. })();
  345. /*@end@*/