├── .gitignore
├── installer
├── EnhancedDiscordUI
│ ├── ed.ico
│ ├── Resources
│ │ ├── ed_og.png
│ │ ├── discord_dev_16.png
│ │ ├── discord_dev_32.png
│ │ ├── discord_dev_64.png
│ │ ├── discord_canary_16.png
│ │ ├── discord_canary_32.png
│ │ ├── discord_canary_64.png
│ │ ├── discord_stable_16.png
│ │ ├── discord_stable_32.png
│ │ └── discord_stable_64.png
│ ├── App.config
│ ├── Properties
│ │ ├── Settings.settings
│ │ ├── Settings.Designer.cs
│ │ ├── AssemblyInfo.cs
│ │ ├── Resources.Designer.cs
│ │ └── Resources.resx
│ ├── Program.cs
│ ├── LogWriter.cs
│ ├── EnhancedDiscordUI.csproj
│ ├── Form1.resx
│ ├── Form1.Designer.cs
│ └── Form1.cs
└── EnhancedDiscordUI.sln
├── plugins
├── silent_typing.js
├── style.css
├── anti_track.js
├── silence.js
├── quick_save.js
├── double_click_mention.js
├── double_click_edit.js
├── guild_count.js
├── friend_count.js
├── css_loader.js
├── avatar_links.js
├── direct_download.js
├── css_settings.js
├── hidden_channels.js
└── ed_settings.js
├── .github
└── workflows
│ └── main.yml
├── .eslintrc.json
├── LICENSE
├── main_process_shit.js
├── README.md
├── injection.js
├── plugin.js
├── bd.css
├── bd_shit.js
└── dom_shit.js
/.gitignore:
--------------------------------------------------------------------------------
1 | **/.vs
2 | **/bin
3 | **/obj
4 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/ed.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/ed.ico
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/ed_og.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/ed_og.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_dev_16.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_dev_16.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_dev_32.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_dev_32.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_dev_64.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_dev_64.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_canary_16.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_canary_16.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_canary_32.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_canary_32.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_canary_64.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_canary_64.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_stable_16.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_stable_16.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_stable_32.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_stable_32.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Resources/discord_stable_64.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/joe27g/EnhancedDiscord/HEAD/installer/EnhancedDiscordUI/Resources/discord_stable_64.png
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/App.config:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Properties/Settings.settings:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
--------------------------------------------------------------------------------
/plugins/silent_typing.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | module.exports = new Plugin({
4 | name: 'Silent Typing',
5 | author: 'Joe 🎸#7070',
6 | description: `Never appear as typing in any channel.`,
7 | color: 'grey',
8 | disabledByDefault: true,
9 |
10 | load: async function() {
11 | EDApi.monkeyPatch(EDApi.findModule('startTyping'), 'startTyping', () => {});
12 | },
13 | unload: function() {
14 | EDApi.findModule('startTyping').startTyping.unpatch();
15 | }
16 | });
17 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Program.cs:
--------------------------------------------------------------------------------
1 | using System;
2 | using System.Collections.Generic;
3 | using System.Linq;
4 | using System.Threading.Tasks;
5 | using System.Windows.Forms;
6 |
7 | namespace EnhancedDiscordUI
8 | {
9 | static class Program
10 | {
11 | ///
12 | /// The main entry point for the application.
13 | ///
14 | [STAThread]
15 | static void Main()
16 | {
17 | Application.EnableVisualStyles();
18 | Application.SetCompatibleTextRenderingDefault(false);
19 | Application.Run(new EDInstaller());
20 | }
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/plugins/style.css:
--------------------------------------------------------------------------------
1 | @import url(https://enhanceddiscord.com/theme.css);
2 |
3 | .theme-dark {
4 | --bg: url(https://i.imgur.com/2WHzMbs.jpg);
5 | --bg-overlay: rgba(0, 0, 0, 0.8);
6 | --accent: #900;
7 | --accent-bright: #f00;
8 | --accent-back: rgba(255, 0, 0, 0.15);
9 | --accent-back-bright: rgba(255, 0, 0, 0.4);
10 | --icon-color: rgba(250, 166, 26, 0.5);
11 | --link-color: #faa61a;
12 | --link-color-hover: #fad61a;
13 | --popup-background: #222;
14 | --popup-highlight: #333;
15 | --unread-color: var(--accent-bright);
16 | --typing-height: 25px;
17 | --gift-button: none;
18 | --gif-picker: none;
19 | }
--------------------------------------------------------------------------------
/.github/workflows/main.yml:
--------------------------------------------------------------------------------
1 | name: Discord Source detection
2 | on: [issues]
3 | jobs:
4 | autoclose:
5 | runs-on: ubuntu-latest
6 | steps:
7 | - name: Autoclose Discord Source issues
8 | uses: IndyV/IssueChecker@v1.0
9 | with:
10 | repo-token: ${{ secrets.GITHUB_TOKEN }}
11 | issue-close-message: "@${issue.user.login} This issue was automatically closed because we don't accept issue reports from Discord Source.\nThe reason for this is because usually these issues aren't well thought out and are often duplicates.\n\nPlease take a few more minutes to create a well-made, proper issue report."
12 | issue-pattern: "Discord Source - .Issue Report]"
13 |
--------------------------------------------------------------------------------
/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": "eslint:recommended",
3 | "env": {
4 | "browser": true,
5 | "node": true,
6 | "es6": true
7 | },
8 | "parserOptions": {
9 | "ecmaVersion": 8
10 | },
11 | "rules": {
12 | "quotes": ["warn", "single", {"allowTemplateLiterals": true}],
13 | "no-console": "off",
14 | "no-unused-vars": "warn",
15 | "no-useless-escape": "warn",
16 | "no-empty": "warn",
17 | "no-var": "error",
18 | "no-mixed-spaces-and-tabs": "warn",
19 | "prefer-const": "warn"
20 | },
21 | "globals": {
22 | "ED": "readonly",
23 | "EDApi": "readonly"
24 | }
25 | }
26 |
--------------------------------------------------------------------------------
/plugins/anti_track.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | module.exports = new Plugin({
4 | name: 'Anti-Track',
5 | author: 'Joe 🎸#7070',
6 | description: `Prevent Discord from sending "tracking" data that they may be selling to advertisers or otherwise sharing.`,
7 | color: 'white',
8 |
9 | load: async function() {
10 | EDApi.monkeyPatch(EDApi.findModule('track'), 'track', () => {});
11 | EDApi.monkeyPatch(EDApi.findModule('submitLiveCrashReport'), 'submitLiveCrashReport', () => {});
12 | },
13 | unload: async function() {
14 | EDApi.findModule('track').track.unpatch();
15 | EDApi.findModule('submitLiveCrashReport').submitLiveCrashReport.unpatch();
16 | }
17 | });
18 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI.sln:
--------------------------------------------------------------------------------
1 |
2 | Microsoft Visual Studio Solution File, Format Version 12.00
3 | # Visual Studio 14
4 | VisualStudioVersion = 14.0.25420.1
5 | MinimumVisualStudioVersion = 10.0.40219.1
6 | Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EnhancedDiscordUI", "EnhancedDiscordUI\EnhancedDiscordUI.csproj", "{3639AE05-14E6-43B2-9DDB-1A3F4F52657C}"
7 | EndProject
8 | Global
9 | GlobalSection(SolutionConfigurationPlatforms) = preSolution
10 | Debug|Any CPU = Debug|Any CPU
11 | Release|Any CPU = Release|Any CPU
12 | EndGlobalSection
13 | GlobalSection(ProjectConfigurationPlatforms) = postSolution
14 | {3639AE05-14E6-43B2-9DDB-1A3F4F52657C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
15 | {3639AE05-14E6-43B2-9DDB-1A3F4F52657C}.Debug|Any CPU.Build.0 = Debug|Any CPU
16 | {3639AE05-14E6-43B2-9DDB-1A3F4F52657C}.Release|Any CPU.ActiveCfg = Release|Any CPU
17 | {3639AE05-14E6-43B2-9DDB-1A3F4F52657C}.Release|Any CPU.Build.0 = Release|Any CPU
18 | EndGlobalSection
19 | GlobalSection(SolutionProperties) = preSolution
20 | HideSolutionNode = FALSE
21 | EndGlobalSection
22 | EndGlobal
23 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2019 joe27g
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/plugins/silence.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | module.exports = new Plugin({
4 | name: 'Shut up, Clyde',
5 | author: 'Joe 🎸#7070',
6 | description: "Silences Clyde saying stupid shit about Nitro, for users that don't have it.",
7 | color: '#7289da',
8 |
9 | load: async function() {
10 | const gg = EDApi.findModule(m => m.getChannelId && m.getGuildId && !m.getPings), bs = EDApi.findModule('Messages').Messages;
11 |
12 | EDApi.monkeyPatch(EDApi.findModule('sendBotMessage'), 'sendBotMessage', function (b) {
13 | if (gg.getGuildId() !== null) return; // don't send Clyde messages when looking at a server
14 | const message = b.methodArguments[1];
15 | if (message == bs.INVALID_ANIMATED_EMOJI_BODY_UPGRADE || message == bs.INVALID_ANIMATED_EMOJI_BODY || message == bs.INVALID_EXTERNAL_EMOJI_BODY_UPGRADE || message == bs.INVALID_EXTERNAL_EMOJI_BODY) return;
16 | return b.callOriginalMethod(b.methodArguments);
17 | });
18 | },
19 | unload: function() {
20 | EDApi.findModule('sendBotMessage').sendBotMessage.unpatch();
21 | }
22 | });
23 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Properties/Settings.Designer.cs:
--------------------------------------------------------------------------------
1 | //------------------------------------------------------------------------------
2 | //
3 | // This code was generated by a tool.
4 | // Runtime Version:4.0.30319.42000
5 | //
6 | // Changes to this file may cause incorrect behavior and will be lost if
7 | // the code is regenerated.
8 | //
9 | //------------------------------------------------------------------------------
10 |
11 | namespace EnhancedDiscordUI.Properties {
12 |
13 |
14 | [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
15 | [global::System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "14.0.0.0")]
16 | internal sealed partial class Settings : global::System.Configuration.ApplicationSettingsBase {
17 |
18 | private static Settings defaultInstance = ((Settings)(global::System.Configuration.ApplicationSettingsBase.Synchronized(new Settings())));
19 |
20 | public static Settings Default {
21 | get {
22 | return defaultInstance;
23 | }
24 | }
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Properties/AssemblyInfo.cs:
--------------------------------------------------------------------------------
1 | using System.Reflection;
2 | using System.Runtime.CompilerServices;
3 | using System.Runtime.InteropServices;
4 |
5 | // General Information about an assembly is controlled through the following
6 | // set of attributes. Change these attribute values to modify the information
7 | // associated with an assembly.
8 | [assembly: AssemblyTitle("EnhancedDiscordUI")]
9 | [assembly: AssemblyDescription("")]
10 | [assembly: AssemblyConfiguration("")]
11 | [assembly: AssemblyCompany("")]
12 | [assembly: AssemblyProduct("EnhancedDiscordUI")]
13 | [assembly: AssemblyCopyright("Copyright © 2018")]
14 | [assembly: AssemblyTrademark("")]
15 | [assembly: AssemblyCulture("")]
16 |
17 | // Setting ComVisible to false makes the types in this assembly not visible
18 | // to COM components. If you need to access a type in this assembly from
19 | // COM, set the ComVisible attribute to true on that type.
20 | [assembly: ComVisible(false)]
21 |
22 | // The following GUID is for the ID of the typelib if this project is exposed to COM
23 | [assembly: Guid("3639ae05-14e6-43b2-9ddb-1a3f4f52657c")]
24 |
25 | // Version information for an assembly consists of the following four values:
26 | //
27 | // Major Version
28 | // Minor Version
29 | // Build Number
30 | // Revision
31 | //
32 | // You can specify all the values or you can default the Build and Revision Numbers
33 | // by using the '*' as shown below:
34 | // [assembly: AssemblyVersion("1.0.*")]
35 | [assembly: AssemblyVersion("1.0.0.0")]
36 | [assembly: AssemblyFileVersion("1.0.0.0")]
37 |
--------------------------------------------------------------------------------
/plugins/quick_save.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | module.exports = new Plugin({
4 | name: 'Quick Save',
5 | author: 'Joe 🎸#7070',
6 | description: 'Use Ctrl+S or Cmd+S to save server, channel, or account settings.',
7 | color: 'salmon',
8 |
9 | load: async function() {
10 | const hcModules = EDApi.findAllModules('hasChanges');
11 | this._listener = function(e) {
12 | if ((window.navigator.platform.match("Mac") ? e.metaKey : e.ctrlKey) && e.keyCode == 83) {
13 | e.preventDefault();
14 | const types = ['GUILD', 'CHANNEL', 'ACCOUNT', 'GUILD ROLES', 'CHANNEL OVERWRITES'];
15 | let hasChanges = false;
16 | for (const i in types) {
17 | if (hcModules[i] && hcModules[i].hasChanges()) {
18 | hasChanges = true;
19 | //module.exports.log(`saving ${types[i]} settings`);
20 | break;
21 | }
22 | }
23 | if (!hasChanges) {
24 | //module.exports.log('No setting changes detected. Aborting.');
25 | return;
26 | }
27 | const saveButton = document.querySelector('[class*="lookFilled-"][class*="colorGreen-"]');
28 | if (saveButton)
29 | try { saveButton.click(); } catch(err) { module.exports.error(err); }
30 | return;
31 | }
32 | }
33 | document.addEventListener("keydown", this._listener, false);
34 | },
35 | unload: function() {
36 | document.removeEventListener("keydown", this._listener);
37 | delete this._listener;
38 | }
39 | });
40 |
--------------------------------------------------------------------------------
/plugins/double_click_mention.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 | let userM = {}, taM = {}, avM = {}, wM = {}, ree;
3 |
4 | module.exports = new Plugin({
5 | name: 'Double-Click Mention',
6 | author: 'Joe 🎸#7070',
7 | description: 'Allows you to double-click a user\'s name to mention them.',
8 | color: '#00bbff',
9 |
10 | _userTag: '',
11 | load: async function() {
12 | taM = EDApi.findModule('textArea');
13 | userM = EDApi.findModule('username');
14 | avM = EDApi.findModule('avatar');
15 | wM = EDApi.findModule(m => m.wrapper && m.avatar);
16 | ree = this;
17 |
18 | document.addEventListener("dblclick", this.doubleListener);
19 | },
20 | unload: async function() {
21 | document.removeEventListener("dblclick", this.doubleListener);
22 | },
23 |
24 | doubleListener: function(e) {
25 | if (!e || !e.target || !e.target.parentElement) return;
26 | let tag;
27 | try {
28 | if (e.target.className === userM.username)
29 | tag = e.target.parentElement.__reactInternalInstance$.return.return.memoizedProps.message.author.tag;
30 | else if (e.target.className === wM.wrapper && e.target.parentElement.className === avM.avatar)
31 | tag = e.target.parentElement.__reactInternalInstance$.return.return.memoizedProps.user.tag;
32 | } catch(err) {
33 | ree.error(err);
34 | tag = null;
35 | }
36 | if (!tag) return;
37 |
38 | const ta = document.querySelector('.'+taM.textArea);
39 | if (!ta) return;
40 | ta.value = `${ta.value ? ta.value.endsWith(' ') ? ta.value : ta.value+' ' : ''}@${tag} `;
41 | }
42 | });
43 |
--------------------------------------------------------------------------------
/main_process_shit.js:
--------------------------------------------------------------------------------
1 | const electron = require('electron');
2 | const ipcMain = require('electron').ipcMain;
3 | const path = require('path');
4 |
5 | ipcMain.on('main-process-info', (event, arg) => {
6 | switch(arg) {
7 | case "original-node-modules-path":
8 | event.returnValue = path.resolve(electron.app.getAppPath(), 'node_modules');
9 | case "original-preload-script":
10 | event.returnValue = event.sender.__preload;
11 | }
12 | });
13 |
14 | ipcMain.on('main-process-utils', (event, arg) => {
15 | switch(arg) {
16 | case "dialog":
17 | event.returnValue = `${electron.dialog}`;
18 | }
19 | });
20 |
21 | ipcMain.handle('custom-devtools-warning', (event, arg) => {
22 | let wc = event.sender.getOwnerBrowserWindow().webContents;
23 | wc.removeAllListeners('devtools-opened');
24 | wc.on('devtools-opened', () => {
25 | wc.executeJavaScript(`
26 | console.log('%cHold Up!', 'color: #FF5200; -webkit-text-stroke: 2px black; font-size: 72px; font-weight: bold;');
27 | console.log("%cIf you're reading this, you're probably smarter than most Discord developers.", 'font-size: 16px;');
28 | console.log('%cPasting anything in here could actually improve the Discord client.', 'font-size: 18px; font-weight: bold; color: red;');
29 | console.log("%cUnless you understand exactly what you're doing, keep this window open to browse our bad code.", 'font-size: 16px;');
30 | console.log("%cIf you don't understand exactly what you\'re doing, you should come work with us: https://discordapp.com/jobs", 'font-size: 16px;');
31 | `)
32 | });
33 | });
34 |
35 | ipcMain.handle('bd-navigate-page-listener', (event, arg) => {
36 | event.sender.getOwnerBrowserWindow().webContents.on('did-navigate-in-page', arg);
37 | })
38 |
39 | ipcMain.handle('remove-bd-navigate-page-listener', (event, arg) => {
40 | event.sender.getOwnerBrowserWindow().webContents.removeEventListener('did-navigate-in-page', arg);
41 | })
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## END OF SUPPORT
2 | __EnhancedDiscord is being shut down.__ It's been nice while it lasted, but we can't keep up with the maintenance that Discord's constant changes require. There won't be any more updates, and on **April 12th**, all support will be dropped and the site will go offline. We recommend you find another client mod that is actively maintained.
3 |
4 |
5 | # EnhancedDiscord
6 | A lightweight client mod designed to enhance your Discord experience without slowing down your PC.
7 |
8 | #### DISCLAIMER!
9 | > **Using EnhancedDiscord, or any other client mod, is against [Discord's Terms of Service](https://discordapp.com/terms). Use it at your own risk.**
10 | > *It's very unlikely any action will be taken against you, but we take no responsibility if anything happens.*
11 |
12 | ### Installing
13 |
14 | Go to https://enhanceddiscord.com and hit the 'Download' button, or if you're lazy, click [here](https://enhanceddiscord.com/EnhancedDiscord.exe).
15 |
16 | For more info, including installing on Linux or MacOS, see the [installation wiki page](https://github.com/joe27g/EnhancedDiscord/wiki/Installation).
17 |
18 | ### Themes
19 |
20 | By default, the official [EnhancedDiscord theme](https://github.com/joe27g/Discord-Theme) is loaded along with a plugin that allows you to change settings of it in **User Settings** > EnhancedDiscord > **Settings**. For more info on how to change/edit your theme, see the [FAQ](https://github.com/joe27g/EnhancedDiscord/wiki/FAQ).
21 |
22 | ### Plugins
23 |
24 | A list of included plugins and their purpose is included on the [plugins wiki page](https://github.com/joe27g/EnhancedDiscord/wiki/Plugins). It also includes some sources for finding new ED plugins.
25 |
26 | ### Custom plugins
27 |
28 | For info about how to create your own plugins, check out the [custom plugins wiki page](https://github.com/joe27g/EnhancedDiscord/wiki/Custom-plugins).
29 |
30 | ### Having issues?
31 |
32 | First, check out the [FAQ](https://github.com/joe27g/EnhancedDiscord/wiki/FAQ) to see if your issue is listed there. ~~If not, ask in #support in the support server (link below.)~~
--------------------------------------------------------------------------------
/plugins/double_click_edit.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | module.exports = new Plugin({
4 | name: 'Double-Click Edit',
5 | author: 'Joe 🎸#7070',
6 | description: 'Allows you to double-click a message to edit or hold delete + click to delete.',
7 | color: '#ff5900',
8 |
9 | deletePressed: false,
10 | load: function() {
11 | this._mm = findRawModule(m => m.displayName == "Message");
12 | this._edm = EDApi.findModule('startEditMessage');
13 | monkeyPatch(this._mm.exports, 'default', {
14 | silent: true,
15 | before: e => e.methodArguments[0].onClick = () => this.handleClick(e.methodArguments[0])
16 | });
17 |
18 | document.addEventListener("keydown", this.keyDownListener);
19 | document.addEventListener("keyup", this.keyUpListener);
20 | },
21 | unload: function() {
22 | if (this._mm.exports.default)
23 | this._mm.exports.default.unpatch();
24 | document.removeEventListener("keydown", this.keyDownListener);
25 | document.removeEventListener("keyup", this.keyUpListener);
26 | },
27 |
28 | handleDoubleClick: function(message) {
29 | const msgObj = message.childrenMessageContent.props.message;
30 | return this._edm.startEditMessage(msgObj.channel_id, msgObj.id, msgObj.content || '');
31 | },
32 |
33 | handleClick: function(message) {
34 | if (Date.now() - message._lastClick < 250)
35 | return this.handleDoubleClick(message);
36 | message._lastClick = Date.now();
37 | console.log(this.deletePressed);
38 | if (!this.deletePressed) return;
39 | const msgObj = message.childrenMessageContent.props.message;
40 | return this._edm.deleteMessage(msgObj.channel_id, msgObj.id);
41 | },
42 |
43 | keyUpListener: e => {
44 | if (e.keyCode == 46)
45 | module.exports.deletePressed = false;
46 | },
47 | keyDownListener: e => {
48 | if (e.keyCode == 46)
49 | module.exports.deletePressed = true;
50 | }
51 | });
52 |
--------------------------------------------------------------------------------
/injection.js:
--------------------------------------------------------------------------------
1 | require('./main_process_shit');
2 | const electron = require('electron');
3 | const path = require('path');
4 | electron.app.commandLine.appendSwitch("no-force-async-hooks-checks");
5 |
6 | electron.session.defaultSession.webRequest.onHeadersReceived(function(details, callback) {
7 | if (!details.responseHeaders['content-security-policy-report-only'] && !details.responseHeaders['content-security-policy']) return callback({cancel: false});
8 | delete details.responseHeaders['content-security-policy-report-only'];
9 | delete details.responseHeaders['content-security-policy'];
10 | callback({cancel: false, responseHeaders: details.responseHeaders});
11 | });
12 |
13 | class BrowserWindow extends electron.BrowserWindow {
14 | constructor(originalOptions) {
15 | let win = new electron.BrowserWindow(originalOptions);
16 | if (!originalOptions || !originalOptions.webPreferences || !originalOptions.title) return win; // eslint-disable-line constructor-super
17 | const originalPreloadScript = originalOptions.webPreferences.preload;
18 |
19 | originalOptions.webPreferences.preload = path.join(process.env.injDir, 'dom_shit.js');
20 | originalOptions.webPreferences.transparency = true;
21 |
22 | win = new electron.BrowserWindow(originalOptions);
23 | win.webContents.__preload = originalPreloadScript;
24 | return win;
25 | }
26 | }
27 |
28 | BrowserWindow.webContents;
29 |
30 | const electron_path = require.resolve('electron');
31 | Object.assign(BrowserWindow, electron.BrowserWindow); // Assigns the new chrome-specific ones
32 |
33 | if (electron.deprecate && electron.deprecate.promisify) {
34 | const originalDeprecate = electron.deprecate.promisify; // Grab original deprecate promisify
35 | electron.deprecate.promisify = (originalFunction) => originalFunction ? originalDeprecate(originalFunction) : () => void 0; // Override with falsey check
36 | }
37 |
38 | const newElectron = Object.assign({}, electron, {BrowserWindow});
39 | // Tempfix for Injection breakage due to new version of Electron on Canary (Electron 7.x)
40 | // Found by Zerebos (Zack Rauen)
41 | delete require.cache[electron_path].exports;
42 | // /TempFix
43 | require.cache[electron_path].exports = newElectron;
44 | //const browser_window_path = require.resolve(path.resolve(electron_path, '..', '..', 'browser-window.js'));
45 | //require.cache[browser_window_path].exports = BrowserWindow;
46 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/LogWriter.cs:
--------------------------------------------------------------------------------
1 | using System;
2 | using System.IO;
3 | using System.Reflection;
4 |
5 | namespace EnhancedDiscordUI
6 | {
7 | static public class Logger
8 | {
9 | static public void Log(string logMessage)
10 | {
11 | string m_exePath = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
12 | try
13 | {
14 | using (StreamWriter w = File.AppendText(m_exePath + "\\" + "log.txt"))
15 | {
16 | _Log("INFO", logMessage, w);
17 | }
18 | }
19 | catch (Exception ex)
20 | {
21 | }
22 | }
23 |
24 | static public void Warn(string logMessage)
25 | {
26 | string m_exePath = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
27 | try
28 | {
29 | using (StreamWriter w = File.AppendText(m_exePath + "\\" + "log.txt"))
30 | {
31 | _Log("WARN", logMessage, w);
32 | }
33 | }
34 | catch (Exception ex)
35 | {
36 | }
37 | }
38 |
39 | static public void Error(string logMessage)
40 | {
41 | string m_exePath = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
42 | try
43 | {
44 | using (StreamWriter w = File.AppendText(m_exePath + "\\" + "log.txt"))
45 | {
46 | _Log("ERROR", logMessage, w);
47 | }
48 | }
49 | catch (Exception ex)
50 | {
51 | }
52 | }
53 |
54 | static public void _Log(string type, string logMessage, TextWriter txtWriter)
55 | {
56 | try
57 | {
58 | txtWriter.WriteLine("[{0}][{1} {2}]: {3}", type, DateTime.Now.ToLongTimeString(), DateTime.Now.ToLongDateString(), logMessage);
59 | }
60 | catch (Exception ex)
61 | {
62 | }
63 | }
64 |
65 | static public void MakeDivider()
66 | {
67 | string m_exePath = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
68 | try
69 | {
70 | using (StreamWriter w = File.AppendText(m_exePath + "\\" + "log.txt"))
71 | {
72 | w.WriteLine("---------------------------------------------------------------------");
73 | }
74 |
75 | }
76 | catch (Exception ex)
77 | {
78 | }
79 | }
80 | }
81 | }
82 |
--------------------------------------------------------------------------------
/plugins/guild_count.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | let sep = {}, ms = {}, kb = {}, gg, sub;
4 |
5 | module.exports = new Plugin({
6 | name: 'Server Count',
7 | author: 'Joe 🎸#7070',
8 | description: "Adds the number of servers you're currently in right above the list.",
9 | color: 'indigo',
10 |
11 | load: async function() {
12 | sep = EDApi.findModule('guildSeparator');
13 | ms = EDApi.findModule('modeSelectable');
14 | kb = EDApi.findModule('keybind');
15 | gg = EDApi.findModule('getGuilds');
16 | sub = EDApi.findModule('subscribe');
17 |
18 | sub.subscribe('CONNECTION_OPEN', this.refreshCount);
19 | sub.subscribe('CONNECTION_RESUMED', this.refreshCount);
20 | sub.subscribe('DISPATCH_APPLICATION_STATE_UPDATE', this.refreshCount);
21 | sub.subscribe('CHANNEL_PRELOAD', this.refreshCount);
22 | sub.subscribe('GUILD_CREATE', this.refreshCount);
23 | sub.subscribe('GUILD_DELETE', this.refreshCount);
24 | sub.subscribe('GUILD_JOIN', this.refreshCount);
25 | this.refreshCount();
26 | },
27 | refreshCount: function() {
28 | if (!sep) return;
29 | const num = Object.keys(gg.getGuilds()).length;
30 |
31 | let guildCount = document.getElementById('ed_guild_count');
32 | if (guildCount) {
33 | if (num === this._num) return; // don't update if # is the same as before
34 | guildCount.innerHTML = num + ' Servers';
35 | this._num = num;
36 | return;
37 | }
38 | const separator = document.querySelector(`.${sep.guildSeparator}`);
39 | if (separator) {
40 | guildCount = document.createElement('div');
41 | guildCount.className = `${ms ? ms.description+' ' : ''}${sep.listItem} ${kb.keybind}`;
42 | guildCount.innerHTML = num + ' Servers';
43 | guildCount.id = 'ed_guild_count';
44 | try {
45 | separator.parentElement.parentElement.insertBefore(guildCount, separator.parentElement)
46 | this._num = num;
47 | } catch(err) {
48 | this.error(err);
49 | }
50 | }
51 | return;
52 | },
53 | unload: function() {
54 | const guildCount = document.getElementById('ed_guild_count');
55 | if (guildCount) guildCount.remove();
56 |
57 | sub.unsubscribe('CONNECTION_OPEN', this.refreshCount);
58 | sub.unsubscribe('GUILD_CREATE', this.refreshCount);
59 | sub.unsubscribe('GUILD_DELETE', this.refreshCount);
60 | sub.unsubscribe('GUILD_JOIN', this.refreshCount);
61 | }
62 | });
63 |
--------------------------------------------------------------------------------
/plugins/friend_count.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 | let sep = {}, ms = {}, kb = {}, sub;
3 |
4 | module.exports = new Plugin({
5 | name: 'Friend Count',
6 | author: 'Joe 🎸#7070',
7 | description: "Adds the number of friends/online friends under the \"Home\" button in the top left.",
8 | color: 'cornflowerblue',
9 |
10 | defaultSettings: {onlineOnly: false},
11 | onSettingsUpdate: function() { return this.reload(); },
12 |
13 | addFriendCount: function() {
14 | if (!sep) return;
15 | const o = (this.settings || {}).onlineOnly;
16 | const num = o ? EDApi.findModule("getOnlineFriendCount").getOnlineFriendCount() : EDApi.findModule("getFriendIDs").getFriendIDs().length;
17 |
18 | let friendCount = document.getElementById('ed_friend_count');
19 | if (friendCount) {
20 | if (num === this._num) return; // don't update if # is the same as before
21 | friendCount.innerHTML = num + (o ? ' Online' : ' Friends');
22 | this._num = num;
23 | return;
24 | }
25 | const separator = document.querySelector(`.${sep.guildSeparator}`);
26 | if (separator) {
27 | friendCount = document.createElement('div');
28 | friendCount.className = `${ms ? ms.description+' ' : ''}${sep.listItem} ${kb.keybind}`;
29 | friendCount.innerHTML = num + (o ? ' Online' : ' Friends');
30 | friendCount.id = 'ed_friend_count';
31 | try {
32 | separator.parentElement.parentElement.insertBefore(friendCount, separator.parentElement)
33 | this._num = num;
34 | } catch(err) {
35 | this.error(err);
36 | }
37 | }
38 | },
39 |
40 | load: async function() {
41 | sep = EDApi.findModule('guildSeparator');
42 | ms = EDApi.findModule('modeSelectable');
43 | kb = EDApi.findModule('keybind');
44 | sub = EDApi.findModule('subscribe');
45 |
46 | sub.subscribe('CONNECTION_OPEN', this.addFriendCount);
47 | sub.subscribe('CONNECTION_RESUMED', this.addFriendCount);
48 | sub.subscribe('DISPATCH_APPLICATION_STATE_UPDATE', this.addFriendCount);
49 | sub.subscribe('PRESENCE_UPDATE', this.addFriendCount);
50 | sub.subscribe('RELATIONSHIP_ADD', this.addFriendCount);
51 | sub.subscribe('RELATIONSHIP_REMOVE', this.addFriendCount);
52 |
53 | this.addFriendCount();
54 | },
55 | unload: function() {
56 | const friendCount = document.getElementById('ed_friend_count');
57 | if (friendCount) friendCount.remove();
58 |
59 | sub.unsubscribe('CONNECTION_OPEN', this.addFriendCount);
60 | sub.unsubscribe('CONNECTION_RESUMED', this.addFriendCount);
61 | sub.unsubscribe('PRESENCE_UPDATE', this.addFriendCount);
62 | sub.unsubscribe('RELATIONSHIP_ADD', this.addFriendCount);
63 | sub.unsubscribe('RELATIONSHIP_REMOVE', this.addFriendCount);
64 | },
65 | generateSettings: () => ([{
66 | type: "input:boolean",
67 | configName: "onlineOnly",
68 | title: "Online Only",
69 | note: "Only show the number of friends online rather than all friends.",
70 | }])
71 | });
72 |
--------------------------------------------------------------------------------
/plugins/css_loader.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 | const path = require('path');
3 | const fs = require('fs');
4 | const readFile = require('util').promisify(fs.readFile);
5 |
6 | module.exports = new Plugin({
7 | name: 'CSS Loader',
8 | author: 'Joe 🎸#7070',
9 | description: 'Loads and hot-reloads CSS.',
10 | preload: true, //load this before Discord has finished starting up
11 | color: 'blue',
12 |
13 | defaultSettings: {path: './plugins/style.css'},
14 | resetSettings: function(toastText) {
15 | EDApi.showToast(toastText);
16 | this.settings = this.defaultSettings;
17 | },
18 | onSettingsUpdate: function() {
19 | const filePath = (this.settings || {}).path;
20 | if (!filePath) {
21 | return this.resetSettings('Empty path. Settings have been reset.');
22 | }
23 | if (filePath.startsWith('http://') || filePath.startsWith('https://')) {
24 | return this.resetSettings('Invalid file path. Must be a local CSS file (stored on your computer, not a URL.)');
25 | }
26 | if (!filePath.endsWith('.css')) {
27 | return this.resetSettings('Invalid file path. Must be a CSS file.');
28 | }
29 | if (path.isAbsolute(filePath)) {
30 | if (!fs.existsSync(filePath)) {
31 | return this.resetSettings('Invalid file path. File does not exist.');
32 | }
33 | } else {
34 | const p = path.join(process.env.injDir, filePath);
35 | if (!fs.existsSync(p)) {
36 | return this.resetSettings('Invalid file path. File does not exist.');
37 | }
38 | }
39 | return this.reload();
40 | },
41 |
42 | load: async function() {
43 | const filePath = (this.settings || {}).path;
44 | if (!filePath) return;
45 | const cssPath = path.isAbsolute(filePath) ? filePath : path.join(process.env.injDir, filePath);
46 |
47 | readFile(cssPath).then(css => {
48 | if (!ED.customCss) {
49 | ED.customCss = document.createElement('style');
50 | document.head.appendChild(ED.customCss);
51 | }
52 | ED.customCss.innerHTML = css;
53 | this.info('Custom CSS loaded!', ED.customCss);
54 |
55 | if (ED.cssWatcher == null) {
56 | ED.cssWatcher = fs.watch(cssPath, { encoding: 'utf-8' },
57 | eventType => {
58 | if (eventType == 'change') {
59 | readFile(cssPath).then(newCss => ED.customCss.innerHTML = newCss);
60 | }
61 | });
62 | }
63 | }).catch(() => console.info('Custom CSS not found. Skipping...'));
64 | },
65 | unload: function() {
66 | if (ED.customCss) {
67 | document.head.removeChild(ED.customCss);
68 | ED.customCss = null;
69 | }
70 | if (ED.cssWatcher) {
71 | ED.cssWatcher.close();
72 | ED.cssWatcher = null;
73 | }
74 | },
75 | generateSettings: function() { return [{
76 | type: "input:text",
77 | configName: "path",
78 | title: "Custom CSS Path",
79 | desc: "This can be relative to the EnhancedDiscord directory (e.g. `./big_gay.css`) or absolute (e.g. `C:/theme.css`.)",
80 | placeholder: this.defaultSettings.path
81 | }]}
82 | });
83 |
--------------------------------------------------------------------------------
/plugin.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Plugin Class
3 | */
4 | class Plugin {
5 | /**
6 | * Create your plugin, must have a name and load() function
7 | * @constructor
8 | * @param {object} options - Plugin options
9 | */
10 | constructor (opts = {}) {
11 | if (!opts.name || typeof opts.load !== 'function')
12 | return 'Invalid plugin. Needs a name and a load() function.';
13 |
14 | Object.assign(this, opts);
15 | if (!this.color)
16 | this.color = 'orange';
17 | if (!this.author)
18 | this.author = '';
19 | }
20 |
21 | /**
22 | * Load this plugin.
23 | */
24 | load () {}
25 |
26 | /**
27 | * Unload this plugin.
28 | */
29 | unload () {}
30 |
31 | /**
32 | * Reload this plugin.
33 | */
34 | reload () {
35 | this.log('Reloading...');
36 | this.unload();
37 | delete require.cache[require.resolve(`./plugins/${this.id}`)];
38 | const newPlugin = require(`./plugins/${this.id}`);
39 | ED.plugins[this.id] = newPlugin;
40 | newPlugin.id = this.id;
41 | return newPlugin.load();
42 | }
43 |
44 | /**
45 | * Send a decorated console.log prefixed with ED and your plugin name
46 | * @param {...string} msg - Message to be logged
47 | */
48 | log (...msg) {
49 | console.log(`%c[EnhancedDiscord] %c[${this.name}]`, 'color: red;', `color: ${this.color}`, ...msg);
50 | }
51 |
52 | /**
53 | * Send a decorated console.info prefixed with ED and your plugin name
54 | * @param {...string} msg - Message to be logged
55 | */
56 | info (...msg) {
57 | console.info(`%c[EnhancedDiscord] %c[${this.name}]`, 'color: red;', `color: ${this.color}`, ...msg);
58 | }
59 |
60 | /**
61 | * Send a decorated console.warn prefixed with ED and your plugin name
62 | * @param {...string} msg - Message to be logged
63 | */
64 | warn (...msg) {
65 | console.warn(`%c[EnhancedDiscord] %c[${this.name}]`, 'color: red;', `color: ${this.color}`, ...msg);
66 | }
67 |
68 | /**
69 | * Send a decorated console.error prefixed with ED and your plugin name
70 | * @param {...string} msg - Message to be logged
71 | */
72 | error (...msg) {
73 | console.error(`%c[EnhancedDiscord] %c[${this.name}]`, 'color: red;', `color: ${this.color}`, ...msg);
74 | }
75 |
76 | /**
77 | * Returns a Promise that resolves after ms milliseconds.
78 | * @param {number} ms - How long to wait before resolving the promise
79 | */
80 | sleep (ms) {
81 | return new Promise(resolve => {
82 | setTimeout(resolve, ms);
83 | });
84 | }
85 |
86 | /**
87 | * Get plugin settings.
88 | * @returns {Object} Plugin settings object
89 | */
90 | get settings() {
91 | return EDApi.loadPluginSettings(this.id);
92 | }
93 |
94 | /**
95 | * Get particular plugin setting.
96 | * @param {string} key
97 | * @returns Plugin setting value
98 | */
99 | getSetting(key) {
100 | return EDApi.loadData(this.id, key);
101 | }
102 |
103 | /**
104 | * Save plugin settings.
105 | * @param {Object} newSets - New plugin settings object
106 | */
107 | set settings(newSets = {}) {
108 | return EDApi.savePluginSettings(this.id, newSets);
109 | }
110 |
111 | /**
112 | * Set particular plugin setting.
113 | * @param {string} key
114 | * @param data
115 | */
116 | setSetting(key, data) {
117 | return EDApi.saveData(this.id, key, data);
118 | }
119 | }
120 |
121 | module.exports = Plugin;
122 |
--------------------------------------------------------------------------------
/plugins/avatar_links.js:
--------------------------------------------------------------------------------
1 | const Plugin = require("../plugin");
2 | const Clipboard = require("electron").clipboard;
3 |
4 | let cm = {}, Dispatcher, ImageResolver, ContextMenuActions, ree;
5 |
6 | module.exports = new Plugin({
7 | name: "Avatar Links",
8 | author: "Joe 🎸#7070",
9 | description: "Lets you copy a user or guild's avatar URL by right-clicking on it.",
10 | color: "#ffe000",
11 |
12 | load: async function() {
13 | ree = this;
14 | cm = EDApi.findModule('menu');
15 | Dispatcher = EDApi.findModule("dispatch");
16 | ImageResolver = EDApi.findModule("getUserAvatarURL");
17 | ContextMenuActions = EDApi.findModule("closeContextMenu");
18 |
19 | Dispatcher.subscribe("CONTEXT_MENU_OPEN", this.checkMenu);
20 | },
21 |
22 | unload: function() {
23 | Dispatcher.unsubscribe("CONTEXT_MENU_OPEN", this.checkMenu);
24 | },
25 |
26 | checkMenu: async function() {
27 | // Make sure it's already in the DOM
28 | await new Promise(r => {setTimeout(r, 5)});
29 | const theMenu = document.querySelector('.'+cm.menu);
30 |
31 | const reactData = theMenu[Object.keys(theMenu).find(key => key.startsWith("__reactInternalInstance") || key.startsWith("__reactFiber"))];
32 |
33 | let label = "";
34 | let url = "";
35 | let props = {onHeightUpdate: () => {}};
36 |
37 | // For users
38 | if (
39 | reactData.return &&
40 | reactData.return.return &&
41 | reactData.return.return.return &&
42 | reactData.return.return.return.return &&
43 | reactData.return.return.return.return.return &&
44 | reactData.return.return.return.return.return.memoizedProps &&
45 | reactData.return.return.return.return.return.memoizedProps.user
46 | ) {
47 | props = reactData.return.return.return.return.return.memoizedProps;
48 | label = "Copy Avatar URL";
49 | const user = props.user;
50 | const imageType = ImageResolver.hasAnimatedAvatar(user) ? "gif" : "png";
51 |
52 | // Internal module maxes at 1024 hardcoded, so do that and change to 4096.
53 | url = ImageResolver.getUserAvatarURL(user, imageType, 1024).replace("size=1024", "size=4096");
54 | // For default avatars
55 | if (!url.startsWith("http") && url.startsWith("/assets"))
56 | url = `https://discordapp.com${url}`;
57 | }
58 |
59 | // For guilds
60 | if (
61 | reactData.return &&
62 | reactData.return.return &&
63 | reactData.return.return.memoizedProps &&
64 | reactData.return.return.memoizedProps.guild &&
65 | !reactData.return.return.memoizedProps.channel
66 |
67 | ) {
68 | props = reactData.return.return.memoizedProps;
69 | label = "Copy Icon URL";
70 | const guild = props.guild;
71 |
72 | // Internal module maxes at 1024 hardcoded, so do that and change to 4096.
73 | url = ImageResolver.getGuildIconURL({id: guild.id, icon: guild.icon, size: 1024}).replace("size=1024", "size=4096");
74 |
75 | // No way to make it return the animated version, do it manually
76 | if (ImageResolver.hasAnimatedGuildIcon(guild))
77 | url = url.replace(".webp?", ".gif?");
78 | else
79 | url = url.replace(".webp?", ".png?");
80 | }
81 |
82 | // Assume it is already in the DOM and add item ASAP
83 | if (label && url)
84 | ree.addMenuItem(url, label, props);
85 | },
86 |
87 | addMenuItem: function(imageURL, text, menu) {
88 | const cmGroups = document.getElementsByClassName(cm.scroller);
89 | if (!cmGroups || cmGroups.length == 0) return;
90 |
91 | const newCmItem = document.createElement("div");
92 | newCmItem.className = cm.item+' '+cm.labelContainer+' '+cm.colorDefault;
93 | const newCmItemContent = document.createElement("div");
94 | newCmItemContent.className = cm.label;
95 | newCmItemContent.innerHTML = text;
96 | newCmItem.appendChild(newCmItemContent);
97 | const lastGroup = cmGroups[cmGroups.length-1];
98 | lastGroup.appendChild(newCmItem);
99 |
100 | menu.onHeightUpdate();
101 |
102 | newCmItem.onclick = () => {
103 | Clipboard.write({text: imageURL});
104 | ContextMenuActions.closeContextMenu();
105 | }
106 | }
107 | });
108 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/EnhancedDiscordUI.csproj:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Debug
6 | AnyCPU
7 | {3639AE05-14E6-43B2-9DDB-1A3F4F52657C}
8 | WinExe
9 | Properties
10 | EnhancedDiscordUI
11 | EnhancedDiscordUI
12 | v4.7.1
13 | 512
14 | true
15 |
16 |
17 |
18 |
19 |
20 | AnyCPU
21 | true
22 | full
23 | false
24 | bin\Debug\
25 | DEBUG;TRACE
26 | prompt
27 | 4
28 |
29 |
30 | AnyCPU
31 | pdbonly
32 | true
33 | bin\Release\
34 | TRACE
35 | prompt
36 | 4
37 |
38 |
39 | ed.ico
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 | Form
59 |
60 |
61 | Form1.cs
62 |
63 |
64 |
65 |
66 |
67 | Form1.cs
68 |
69 |
70 | ResXFileCodeGenerator
71 | Designer
72 | Resources.Designer.cs
73 |
74 |
75 | SettingsSingleFileGenerator
76 | Settings.Designer.cs
77 |
78 |
79 | True
80 | True
81 | Resources.resx
82 |
83 |
84 | True
85 | Settings.settings
86 | True
87 |
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
96 |
97 |
98 |
99 |
100 |
101 |
102 |
103 |
104 |
105 |
106 |
107 |
108 |
109 |
110 |
111 |
112 |
113 |
114 |
115 |
116 |
117 |
118 |
119 |
120 |
121 |
122 |
123 |
124 |
125 |
126 |
127 |
128 | This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.
129 |
130 |
131 |
138 |
--------------------------------------------------------------------------------
/plugins/direct_download.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | // contains modified code from https://stackoverflow.com/a/47820271
4 | const ipcRenderer = require('electron').ipcRenderer;
5 | const dialog = ipcRenderer.sendSync('main-process-utils', 'dialog');
6 | const http = require('https');
7 | const fs = require('fs');
8 | let ttM = {}, iteM = {};
9 |
10 | function saveAs(url, filename, fileExtension) {
11 | dialog.showSaveDialog({ defaultPath: filename, title: 'Where would you like to store the stolen memes?', buttonLabel: 'Steal this meme', filters: [{ name: 'Stolen meme', extensions: [fileExtension] }] })
12 | .then(e => {
13 | if (!e.canceled) {
14 | download(url, e.filePath, () => {
15 | const wrap = document.createElement('div');
16 | wrap.className = 'theme-dark';
17 | const gay = document.createElement('div');
18 | gay.style = 'position: fixed; bottom: 10%; left: calc(50% - 88px);'
19 | gay.className = `${ttM.tooltip} ${ttM.tooltipTop} ${ttM.tooltipBlack}`;
20 | gay.innerHTML = 'Successfully downloaded | ' + e.filePath;
21 | document.body.appendChild(wrap);
22 | wrap.appendChild(gay);
23 | setTimeout(() => wrap.remove(), 2000);
24 | });
25 | }
26 | });
27 | }
28 | function download (url, dest, cb) {
29 | const file = fs.createWriteStream(dest);
30 | http.get(url, function(response) {
31 | response.pipe(file);
32 | file.on('finish', function() {
33 | file.close(cb);
34 | });
35 | }).on('error', function(err) {
36 | fs.unlink(dest);
37 | if (cb) cb(err.message);
38 | });
39 | }
40 |
41 | function addMenuItem(url, text, filename = true, fileExtension) {
42 | const cmGroups = document.querySelectorAll(`.${iteM.menu} [role='group']`);
43 | if (!cmGroups || cmGroups.length == 0) return;
44 |
45 | const newCmItem = document.createElement('div');
46 | newCmItem.className = `${iteM.item} ${iteM.labelContainer} ${iteM.colorDefault}`;
47 |
48 | // Discord uses JS to add classes, not sure how to recreate
49 | newCmItem.onmouseenter = function() {
50 | this.classList.add(iteM.focused);
51 | };
52 | newCmItem.onmouseleave = function() {
53 | this.classList.remove(iteM.focused);
54 | };
55 |
56 | const newCmItemLabel = document.createElement('div');
57 | newCmItemLabel.className = iteM.label;
58 | newCmItemLabel.innerText = text;
59 |
60 | newCmItem.appendChild(newCmItemLabel);
61 |
62 | const lastGroup = cmGroups[cmGroups.length-1];
63 | lastGroup.appendChild(newCmItem);
64 | newCmItem.onclick = () => saveAs(url, filename, fileExtension);
65 | }
66 |
67 | // contains code modified from https://github.com/Metalloriff/BetterDiscordPlugins/blob/master/SaveTo.plugin.js
68 |
69 | module.exports = new Plugin({
70 | name: 'Direct Download',
71 | author: 'Joe 🎸#7070',
72 | description: `Download files Steal memes without opening a browser.`,
73 | color: '#18770e',
74 |
75 | load: async function() {
76 | this._cmClass = EDApi.findModule('hideInteraction').menu;
77 | this._contClass = EDApi.findModule('embedWrapper').container;
78 | ttM = EDApi.findModule('tooltipPointer');
79 | iteM = EDApi.findModule('hideInteraction');
80 | Dispatcher = EDApi.findModule("dispatch");
81 | Dispatcher.subscribe("CONTEXT_MENU_OPEN", this.listener);
82 | },
83 | listener(e) {
84 | if (document.getElementsByClassName(this._cmClass).length == 0) setTimeout(() => module.exports.onContextMenu(e), 0);
85 | else this.onContextMenu(e);
86 | },
87 | onContextMenu(e) {
88 | e = e.contextMenu;
89 | const messageGroup = e.target.closest('.'+this._contClass);
90 | const parentElem = e.target.parentElement;
91 | const guildWrapper = EDApi.findModule('childWrapper').wrapper;
92 | const memberAvatar = EDApi.findModule('nameAndDecorators').avatar;
93 |
94 | if (e.target.localName != 'a' && e.target.localName != 'img' && e.target.localName != 'video' && !messageGroup && !e.target.className.includes(guildWrapper) && !parentElem.className.includes(memberAvatar) && !e.target.className.includes('avatar-')) return;
95 | let saveLabel = 'Download',
96 | url = e.target.poster || e.target.style.backgroundImage.substring(e.target.style.backgroundImage.indexOf(`'`) + 1, e.target.style.backgroundImage.lastIndexOf(`'`)) || e.target.href || e.target.src;
97 |
98 | if (e.target.className.includes(guildWrapper)) {
99 | saveLabel = 'Download Icon';
100 | if (e.target.firstChild.src) { // Make sure guild box has an icon
101 | url = e.target.firstChild.src;
102 | }
103 | }
104 | else if (e.target.className.includes('avatar-') || (parentElem.nodeName == 'DIV' && parentElem.className.includes(memberAvatar))) {
105 | saveLabel = 'Download Avatar';
106 |
107 | if (parentElem.className.includes(memberAvatar)) {
108 | url = e.target.firstChild.firstChild.firstChild.src;
109 | }
110 | }
111 |
112 | if (!url || e.target.classList.contains('emote') || url.includes('youtube.com/watch?v=') || url.includes('youtu.be/') || url.lastIndexOf('/') > url.lastIndexOf('.')) return;
113 |
114 | url = url.split('?')[0];
115 |
116 | url = url.replace('.webp', '.png');
117 |
118 | let fileName = url.substring(url.lastIndexOf('/') + 1, url.lastIndexOf('.'));
119 | const fileExtension = url.substr(url.lastIndexOf('.') + 1, url.length);
120 |
121 | if (saveLabel.includes('Avatar') || saveLabel.includes('Icon')) url += '?size=2048';
122 |
123 | if (e.target.classList.contains('emoji')) {
124 | saveLabel = 'Download Emoji';
125 | fileName = e.target.alt.replace(/[^A-Za-z_-]/g, '');
126 | }
127 | //console.log({url, saveLabel, fileName, fileExtension});
128 |
129 | setTimeout(() => addMenuItem(url, saveLabel, fileName, fileExtension), 5);
130 | },
131 | unload: function() {
132 | Dispatcher.unsubscribe("CONTEXT_MENU_OPEN", this.listener);
133 | }
134 | });
135 |
--------------------------------------------------------------------------------
/bd.css:
--------------------------------------------------------------------------------
1 | #bd-settingspane-container h2.ui-form-title {
2 | font-size: 16px;
3 | font-weight: 600;
4 | line-height: 20px;
5 | text-transform: uppercase;
6 | display: inline-block;
7 | margin-bottom: 20px;
8 | }
9 | #bd-settingspane-container h2.ui-form-title {
10 | color: #f6f6f7;
11 | }
12 | .theme-light #bd-settingspane-container h2.ui-form-title {
13 | color: #4f545c;
14 | }
15 |
16 | #bd-settingspane-container .ui-switch-item {
17 | flex-direction: column;
18 | margin-top: 8px;
19 | }
20 |
21 | #bd-settingspane-container .ui-switch-item h3 {
22 | font-size: 16px;
23 | font-weight: 500;
24 | line-height: 24px;
25 | flex: 1;
26 | }
27 | #bd-settingspane-container .ui-switch-item h3 {
28 | color: #f6f6f7;
29 | }
30 | .theme-light #bd-settingspane-container .ui-switch-item h3 {
31 | color: #4f545c;
32 | }
33 |
34 | #bd-settingspane-container .ui-switch-item .style-description {
35 | font-size: 14px;
36 | font-weight: 500;
37 | line-height: 20px;
38 | margin-bottom: 10px;
39 | padding-bottom: 10px;
40 | border-bottom: 1px solid hsla(218,5%,47%,.3);
41 | }
42 | #bd-settingspane-container .ui-switch-item .style-description {
43 | color: #72767d;
44 | }
45 | .theme-light #bd-settingspane-container .ui-switch-item .style-description {
46 | color: rgba(114,118,125,.6);
47 | }
48 |
49 | #bd-settingspane-container .ui-switch-item .ui-switch-wrapper {
50 | -webkit-user-select: none;
51 | -moz-user-select: none;
52 | -ms-user-select: none;
53 | user-select: none;
54 | position: relative;
55 | width: 44px;
56 | height: 24px;
57 | display: block;
58 | flex: 0 0 auto;
59 | }
60 |
61 | #bd-settingspane-container .ui-switch-item .ui-switch-wrapper input {
62 | position: absolute;
63 | opacity: 0;
64 | cursor: pointer;
65 | width: 100%;
66 | height: 100%;
67 | z-index: 1;
68 | }
69 |
70 | #bd-settingspane-container .ui-switch-item .ui-switch-wrapper .ui-switch {
71 | background: #7289da;
72 | position: absolute;
73 | top: 0;
74 | right: 0;
75 | bottom: 0;
76 | left: 0;
77 | background: #72767d;
78 | border-radius: 14px;
79 | transition: background .15s ease-in-out,box-shadow .15s ease-in-out,border .15s ease-in-out;
80 | }
81 |
82 | #bd-settingspane-container .ui-switch-item .ui-switch-wrapper .ui-switch:before {
83 | content: '';
84 | display: block;
85 | width: 18px;
86 | height: 18px;
87 | position: absolute;
88 | top: 3px;
89 | left: 3px;
90 | bottom: 3px;
91 | background: #f6f6f7;
92 | border-radius: 10px;
93 | transition: all .15s ease;
94 | box-shadow: 0 3px 1px 0 rgba(0,0,0,.05),0 2px 2px 0 rgba(0,0,0,.1),0 3px 3px 0 rgba(0,0,0,.05);
95 | }
96 |
97 | #bd-settingspane-container .ui-switch-item .ui-switch-wrapper .ui-switch.checked {
98 | background: #7289da;
99 | }
100 |
101 | #bd-settingspane-container .ui-switch-item .ui-switch-wrapper .ui-switch.checked:before {
102 | transform: translateX(20px);
103 | }
104 |
105 | #bd-settingspane-container .plugin-settings {
106 | padding: 0 12px 12px 20px;
107 | }
108 |
109 | @keyframes bd-modal-backdrop {
110 | to { opacity: 0.85; }
111 | }
112 |
113 | @keyframes bd-modal-anim {
114 | to { transform: scale(1); opacity: 1; }
115 | }
116 |
117 | @keyframes bd-modal-backdrop-closing {
118 | to { opacity: 0; }
119 | }
120 |
121 | @keyframes bd-modal-closing {
122 | to { transform: scale(0.7); opacity: 0; }
123 | }
124 |
125 | #bd-settingspane-container .backdrop {
126 | animation: bd-modal-backdrop 250ms ease;
127 | animation-fill-mode: forwards;
128 | background-color: rgb(0, 0, 0);
129 | transform: translateZ(0px);
130 | }
131 |
132 | #bd-settingspane-container.closing .backdrop {
133 | animation: bd-modal-backdrop-closing 200ms linear;
134 | animation-fill-mode: forwards;
135 | animation-delay: 50ms;
136 | opacity: 0.85;
137 | }
138 |
139 | #bd-settingspane-container.closing .modal {
140 | animation: bd-modal-closing 250ms cubic-bezier(0.19, 1, 0.22, 1);
141 | animation-fill-mode: forwards;
142 | opacity: 1;
143 | transform: scale(1);
144 | }
145 |
146 | #bd-settingspane-container .modal {
147 | animation: bd-modal-anim 250ms cubic-bezier(0.175, 0.885, 0.32, 1.275);
148 | animation-fill-mode: forwards;
149 | transform: scale(0.7);
150 | transform-origin: 50% 50%;
151 | }
152 | /* Toast CSS */
153 |
154 | .toasts {
155 | position: fixed;
156 | display: flex;
157 | top: 0;
158 | flex-direction: column;
159 | align-items: center;
160 | justify-content: flex-end;
161 | pointer-events: none;
162 | z-index: 4000;
163 | }
164 |
165 | @keyframes toast-up {
166 | from {
167 | transform: translateY(0);
168 | opacity: 0;
169 | }
170 | }
171 |
172 | .toast {
173 | animation: toast-up 300ms ease;
174 | transform: translateY(-10px);
175 | background: #36393F;
176 | padding: 10px;
177 | border-radius: 5px;
178 | box-shadow: 0 0 0 1px rgba(32,34,37,.6), 0 2px 10px 0 rgba(0,0,0,.2);
179 | font-weight: 500;
180 | color: #fff;
181 | user-select: text;
182 | font-size: 14px;
183 | opacity: 1;
184 | margin-top: 10px;
185 | pointer-events: none;
186 | user-select: none;
187 | }
188 |
189 | @keyframes toast-down {
190 | to {
191 | transform: translateY(0px);
192 | opacity: 0;
193 | }
194 | }
195 |
196 | .toast.closing {
197 | animation: toast-down 200ms ease;
198 | animation-fill-mode: forwards;
199 | opacity: 1;
200 | transform: translateY(-10px);
201 | }
202 |
203 |
204 | .toast.icon {
205 | padding-left: 30px;
206 | background-size: 20px 20px;
207 | background-repeat: no-repeat;
208 | background-position: 6px 50%;
209 | }
210 |
211 | .toast.toast-info {
212 | background-color: #4a90e2;
213 | }
214 |
215 | .toast.toast-info.icon {
216 | background-image: url();
217 | }
218 |
219 | .toast.toast-success {
220 | background-color: #43b581;
221 | }
222 |
223 | .toast.toast-success.icon {
224 | background-image: url();
225 | }
226 | .toast.toast-danger,
227 | .toast.toast-error {
228 | background-color: #f04747;
229 | }
230 |
231 | .toast.toast-danger.icon,
232 | .toast.toast-error.icon {
233 | background-image: url();
234 | }
235 |
236 | .toast.toast-warning,
237 | .toast.toast-warn {
238 | background-color: #FFA600;
239 | color: white;
240 | }
241 |
242 | .toast.toast-warning.icon,
243 | .toast.toast-warn.icon {
244 | background-image: url();
245 | }
246 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Properties/Resources.Designer.cs:
--------------------------------------------------------------------------------
1 | //------------------------------------------------------------------------------
2 | //
3 | // This code was generated by a tool.
4 | // Runtime Version:4.0.30319.42000
5 | //
6 | // Changes to this file may cause incorrect behavior and will be lost if
7 | // the code is regenerated.
8 | //
9 | //------------------------------------------------------------------------------
10 |
11 | namespace EnhancedDiscordUI.Properties {
12 | using System;
13 |
14 |
15 | ///
16 | /// A strongly-typed resource class, for looking up localized strings, etc.
17 | ///
18 | // This class was auto-generated by the StronglyTypedResourceBuilder
19 | // class via a tool like ResGen or Visual Studio.
20 | // To add or remove a member, edit your .ResX file then rerun ResGen
21 | // with the /str option, or rebuild your VS project.
22 | [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "15.0.0.0")]
23 | [global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
24 | [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
25 | internal class Resources {
26 |
27 | private static global::System.Resources.ResourceManager resourceMan;
28 |
29 | private static global::System.Globalization.CultureInfo resourceCulture;
30 |
31 | [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
32 | internal Resources() {
33 | }
34 |
35 | ///
36 | /// Returns the cached ResourceManager instance used by this class.
37 | ///
38 | [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
39 | internal static global::System.Resources.ResourceManager ResourceManager {
40 | get {
41 | if (object.ReferenceEquals(resourceMan, null)) {
42 | global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("EnhancedDiscordUI.Properties.Resources", typeof(Resources).Assembly);
43 | resourceMan = temp;
44 | }
45 | return resourceMan;
46 | }
47 | }
48 |
49 | ///
50 | /// Overrides the current thread's CurrentUICulture property for all
51 | /// resource lookups using this strongly typed resource class.
52 | ///
53 | [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
54 | internal static global::System.Globalization.CultureInfo Culture {
55 | get {
56 | return resourceCulture;
57 | }
58 | set {
59 | resourceCulture = value;
60 | }
61 | }
62 |
63 | ///
64 | /// Looks up a localized resource of type System.Drawing.Bitmap.
65 | ///
66 | internal static System.Drawing.Bitmap discord_canary_16 {
67 | get {
68 | object obj = ResourceManager.GetObject("discord_canary_16", resourceCulture);
69 | return ((System.Drawing.Bitmap)(obj));
70 | }
71 | }
72 |
73 | ///
74 | /// Looks up a localized resource of type System.Drawing.Bitmap.
75 | ///
76 | internal static System.Drawing.Bitmap discord_canary_32 {
77 | get {
78 | object obj = ResourceManager.GetObject("discord_canary_32", resourceCulture);
79 | return ((System.Drawing.Bitmap)(obj));
80 | }
81 | }
82 |
83 | ///
84 | /// Looks up a localized resource of type System.Drawing.Bitmap.
85 | ///
86 | internal static System.Drawing.Bitmap discord_canary_64 {
87 | get {
88 | object obj = ResourceManager.GetObject("discord_canary_64", resourceCulture);
89 | return ((System.Drawing.Bitmap)(obj));
90 | }
91 | }
92 |
93 | ///
94 | /// Looks up a localized resource of type System.Drawing.Bitmap.
95 | ///
96 | internal static System.Drawing.Bitmap discord_dev_16 {
97 | get {
98 | object obj = ResourceManager.GetObject("discord_dev_16", resourceCulture);
99 | return ((System.Drawing.Bitmap)(obj));
100 | }
101 | }
102 |
103 | ///
104 | /// Looks up a localized resource of type System.Drawing.Bitmap.
105 | ///
106 | internal static System.Drawing.Bitmap discord_dev_32 {
107 | get {
108 | object obj = ResourceManager.GetObject("discord_dev_32", resourceCulture);
109 | return ((System.Drawing.Bitmap)(obj));
110 | }
111 | }
112 |
113 | ///
114 | /// Looks up a localized resource of type System.Drawing.Bitmap.
115 | ///
116 | internal static System.Drawing.Bitmap discord_dev_64 {
117 | get {
118 | object obj = ResourceManager.GetObject("discord_dev_64", resourceCulture);
119 | return ((System.Drawing.Bitmap)(obj));
120 | }
121 | }
122 |
123 | ///
124 | /// Looks up a localized resource of type System.Drawing.Bitmap.
125 | ///
126 | internal static System.Drawing.Bitmap discord_stable_16 {
127 | get {
128 | object obj = ResourceManager.GetObject("discord_stable_16", resourceCulture);
129 | return ((System.Drawing.Bitmap)(obj));
130 | }
131 | }
132 |
133 | ///
134 | /// Looks up a localized resource of type System.Drawing.Bitmap.
135 | ///
136 | internal static System.Drawing.Bitmap discord_stable_32 {
137 | get {
138 | object obj = ResourceManager.GetObject("discord_stable_32", resourceCulture);
139 | return ((System.Drawing.Bitmap)(obj));
140 | }
141 | }
142 |
143 | ///
144 | /// Looks up a localized resource of type System.Drawing.Bitmap.
145 | ///
146 | internal static System.Drawing.Bitmap discord_stable_64 {
147 | get {
148 | object obj = ResourceManager.GetObject("discord_stable_64", resourceCulture);
149 | return ((System.Drawing.Bitmap)(obj));
150 | }
151 | }
152 |
153 | ///
154 | /// Looks up a localized resource of type System.Drawing.Bitmap.
155 | ///
156 | internal static System.Drawing.Bitmap ed_og {
157 | get {
158 | object obj = ResourceManager.GetObject("ed_og", resourceCulture);
159 | return ((System.Drawing.Bitmap)(obj));
160 | }
161 | }
162 |
163 | ///
164 | /// Looks up a localized string similar to require(`${process.env.injDir}/injection.js`);.
165 | ///
166 | internal static string injection {
167 | get {
168 | return ResourceManager.GetString("injection", resourceCulture);
169 | }
170 | }
171 |
172 | ///
173 | /// Looks up a localized string similar to https://codeload.github.com/joe27g/EnhancedDiscord/zip/.
174 | ///
175 | internal static string zipLink {
176 | get {
177 | return ResourceManager.GetString("zipLink", resourceCulture);
178 | }
179 | }
180 | }
181 | }
182 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Form1.resx:
--------------------------------------------------------------------------------
1 |
2 |
3 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
82 |
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
96 |
97 |
98 |
99 |
100 |
101 |
102 |
103 |
104 |
105 |
106 |
107 |
108 |
109 | text/microsoft-resx
110 |
111 |
112 | 2.0
113 |
114 |
115 | System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089
116 |
117 |
118 | System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089
119 |
120 |
121 | 19, 20
122 |
123 |
124 |
125 |
126 | AAABAAEAEBAAAAEAIABoBAAAFgAAACgAAAAQAAAAIAAAAAEAIAAAAAAAAAQAABMLAAATCwAAAAAAAAAA
127 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
128 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
129 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
130 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
131 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAQAAAAIAAAACAAAAAgAAAAIAAAACAAAAAAAAAAEAAAACAAAAAgAA
132 | AAIAAAACAAAAAQAAAAAAAAAAAAAAAAAON1cADTaWAA02lwANNpcADTaYAA0zkgAHHzgADTZUAA02mAAN
133 | NpcADTaXAA01lwAMMYQACCJFAAEDCQACCgAAFVOdABVW/wAVU/gAFVPwABVT8AAVU/EAEUWrABNNrAAV
134 | Vv8AFVT5ABVU8wAVVPUAFVX9ABRR7AAOO3gAAAAJABxvnQAccv8AF1y/ABRSgwAVVYQAE051ABFHOAAb
135 | baEAHHH/ABZXjQAWWTIAF1w2ABllbwAcce0AGmjmAA01QQAiip0AI47/ACOM+gAji/UAI4z2ACGE5wAV
136 | U1sAIomZACOM/wAaaGwAM80AABZaAAAXXwgAIoq5ACKK/AAWWmcAKaWdACqp/wAkkM0AIoeeACKJnwAi
137 | iJ4AHXNaACGFVAAhg5IAGGFQABFFIwATSiUAIYNWACqn4wApo+4AGmhJADDBnQAyxv8AL7vvAC643wAu
138 | ud8ALbTbAB97bAAml0UALrfQAC654gAuuOIALrnkADDA9gAxw/kAKqqOAA87CwA11WwANtiyADbZsgA2
139 | 2bMANtmzADfZtQAvu5IAHHAeADbWeQA32bUANtmzADbYswA10p4AMMBZACKIDAApowAAMsUDAC61BQAt
140 | tAUALbQFAC20BQAttAUANM4FACqoAQAxwgIALrYFAC20BQAqqQUAHXQCACqmAAAAAAAAAAAAAAAAAAAA
141 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
142 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
143 | AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
144 | AAAAAAAA//8AAP//AAD//wAA//8AAAIHAAAAAQAAAAAAAAAAAAAAMAAAAAAAAAAAAAAAAQAAAAcAAP//
145 | AAD//wAA//8AAA==
146 |
147 |
148 |
--------------------------------------------------------------------------------
/bd_shit.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 | const fs = require('fs');
3 | const Module = require('module').Module;
4 | const originalRequire = Module._extensions['.js'];
5 | const EDPlugin = require('./plugin');
6 | const electron = require('electron');
7 |
8 | const splitRegex = /[^\S\r\n]*?(?:\r\n|\n)[^\S\r\n]*?\*[^\S\r\n]?/;
9 | const escapedAtRegex = /^\\@/;
10 | module.exports = class BDManager {
11 |
12 | static async setup() {
13 | this.defineGlobals();
14 | this.jqueryElement = document.createElement('script');
15 | this.jqueryElement.src = `//ajax.googleapis.com/ajax/libs/jquery/2.0.0/jquery.min.js`;
16 | await new Promise(resolve => {
17 | this.jqueryElement.onload = resolve;
18 | document.head.appendChild(this.jqueryElement);
19 | });
20 |
21 | this.observer = new MutationObserver((mutations) => {
22 | for (let i = 0, mlen = mutations.length; i < mlen; i++) this.fireEvent('observer', mutations[i]);
23 | });
24 | this.observer.observe(document, {childList: true, subtree: true});
25 |
26 | //this.currentWindow.webContents.on('did-navigate-in-page', BDManager.onSwitch);
27 | try { electron.ipcRenderer.invoke('bd-navigate-page-listener', BDManager.onSwitch); }
28 | catch(err) { throw new Error(`Could not add navigate page listener. ${err}\n${err.stack}`); };
29 |
30 | fs.readFile(path.join(process.env.injDir, 'bd.css'), (err, text) => {
31 | if (err) return console.error(err);
32 | EDApi.injectCSS('BDManager', text);
33 | });
34 |
35 | Module._extensions['.js'] = this.pluginRequire();
36 | }
37 |
38 | static destroy() {
39 | EDApi.clearCSS('BDManager');
40 | this.observer.disconnect();
41 | //this.currentWindow.webContents.removeEventListener('did-navigate-in-page', BDManager.onSwitch);
42 | try { electron.ipcRenderer.invoke('remove-bd-navigate-page-listener', BDManager.onSwitch); }
43 | catch(err) { throw new Error(`Could not remove navigate page listener. ${err}\n${err.stack}`); };
44 | this.jqueryElement.remove();
45 | Module._extensions['.js'] = originalRequire;
46 | }
47 |
48 | static onSwitch() {
49 | BDManager.fireEvent('onSwitch');
50 | }
51 |
52 | static extractMeta(content) {
53 | const firstLine = content.split('\n')[0];
54 | const hasOldMeta = firstLine.includes('//META');
55 | if (hasOldMeta) return BDManager.parseOldMeta(content);
56 | const hasNewMeta = firstLine.includes('/**');
57 | if (hasNewMeta) return BDManager.parseNewMeta(content);
58 | throw new Error('META was not found.');
59 | }
60 |
61 | static parseOldMeta(content) {
62 | const meta = content.split('\n')[0];
63 | const rawMeta = meta.substring(meta.lastIndexOf('//META') + 6, meta.lastIndexOf('*//'));
64 | if (meta.indexOf('META') < 0) throw new Error('META was not found.');
65 | const parsed = EDApi.testJSON(rawMeta);
66 | if (!parsed) throw new Error('META could not be parsed.');
67 | if (!parsed.name) throw new Error('META missing name data.');
68 | parsed.format = 'json';
69 | return parsed;
70 | }
71 |
72 | static parseNewMeta(content) {
73 | const block = content.split('/**', 2)[1].split('*/', 1)[0];
74 | const out = {};
75 | let field = '';
76 | let accum = '';
77 | for (const line of block.split(splitRegex)) {
78 | if (line.length === 0) continue;
79 | if (line.charAt(0) === '@' && line.charAt(1) !== ' ') {
80 | out[field] = accum;
81 | const l = line.indexOf(' ');
82 | field = line.substr(1, l - 1);
83 | accum = line.substr(l + 1);
84 | }
85 | else {
86 | accum += ' ' + line.replace('\\n', '\n').replace(escapedAtRegex, '@');
87 | }
88 | }
89 | out[field] = accum.trim();
90 | delete out[''];
91 | out.format = 'jsdoc';
92 | return out;
93 | }
94 |
95 | static isEmpty(obj) {
96 | if (obj == null || obj == undefined || obj == '') return true;
97 | if (typeof(obj) !== 'object') return false;
98 | if (Array.isArray(obj)) return obj.length == 0;
99 | for (const key in obj) {
100 | if (obj.hasOwnProperty(key)) return false;
101 | }
102 | return true;
103 | }
104 |
105 | static pluginRequire() {
106 | return function(moduleWrap, filename) {
107 | if (!filename.endsWith('.plugin.js') || path.dirname(filename) !== path.resolve(process.env.injDir, 'plugins')) return Reflect.apply(originalRequire, this, arguments);
108 | let content = fs.readFileSync(filename, 'utf8');
109 | if (content.charCodeAt(0) === 0xFEFF) content = content.slice(1); // Strip BOM
110 | const meta = BDManager.extractMeta(content);
111 | meta.filename = path.basename(filename);
112 |
113 | moduleWrap._compile(content, filename);
114 | const noExport = BDManager.isEmpty(moduleWrap.exports);
115 | if (noExport) {
116 | content += `\nmodule.exports = ${meta.name};`;
117 | moduleWrap._compile(content, filename);
118 | }
119 | if (moduleWrap.exports.default) moduleWrap.exports = moduleWrap.exports.default;
120 | moduleWrap.exports = BDManager.convertPlugin(new moduleWrap.exports());
121 | };
122 | }
123 |
124 | static fireEvent(event, ...args) {
125 | const plugins = Object.values(ED.plugins);
126 | for (let p = 0; p < plugins.length; p++) {
127 | const plugin = plugins[p];
128 | if (!plugin[event] || typeof plugin[event] !== 'function') continue;
129 | try { plugin[event](...args); }
130 | catch (error) { throw new Error(`Could not fire ${event} for plugin ${plugin.name}.`); }
131 | }
132 | }
133 |
134 | static convertPlugin(plugin) {
135 | const newPlugin = new EDPlugin({
136 | name: plugin.getName(),
137 | load: function() {
138 | if (plugin.load) plugin.load();
139 | plugin.start();
140 | },
141 | unload: function() {plugin.stop();},
142 | config: {},
143 | bdplugin: plugin
144 | });
145 | Object.defineProperties(newPlugin, {
146 | name: {
147 | enumerable: true, configurable: true,
148 | get() {return plugin.getName();}
149 | },
150 | author: {
151 | enumerable: true, configurable: true,
152 | get() {return plugin.getAuthor();}
153 | },
154 | description: {
155 | enumerable: true, configurable: true,
156 | get() {return plugin.getDescription();}
157 | }
158 | });
159 | if (typeof plugin.getSettingsPanel == 'function') {
160 | newPlugin.settingsSectionName = plugin.getName();
161 | newPlugin.generateSettingsSection = function() {return plugin.getSettingsPanel();};
162 | newPlugin.getSettingsPanel = function() {return plugin.getSettingsPanel();};
163 | }
164 | if (typeof plugin.onSwitch == 'function') newPlugin.onSwitch = function() {return plugin.onSwitch();};
165 | if (typeof plugin.observer == 'function') newPlugin.observer = function(e) {return plugin.observer(e);};
166 | return newPlugin;
167 | }
168 |
169 | static defineGlobals() {
170 | window.bdConfig = {dataPath: process.env.injDir};
171 | window.bdplugins = window.bdthemes = window.pluginCookie = window.themeCookie = window.settingsCookie = {};
172 | window.bdpluginErrors = window.bdthemeErrors = [];
173 |
174 | window.bdPluginStorage = {get: EDApi.getData, set: EDApi.setData};
175 | window.Utils = {monkeyPatch: EDApi.monkeyPatch, suppressErrors: EDApi.suppressErrors, escapeID: EDApi.escapeID};
176 |
177 | window.BDV2 = class V2 {
178 | static get WebpackModules() {return {find: EDApi.findModule, findAll: EDApi.findAllModules, findByUniqueProperties: EDApi.findModuleByProps, findByDisplayName: EDApi.findModuleByDisplayName};}
179 | static getInternalInstance(node) {return EDApi.getInternalInstance(node);}
180 | static get react() {return EDApi.React;}
181 | static get reactDom() {return EDApi.ReactDOM;}
182 | };
183 | }
184 | };
185 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Properties/Resources.resx:
--------------------------------------------------------------------------------
1 |
2 |
3 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
82 |
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
96 |
97 |
98 |
99 |
100 |
101 |
102 |
103 |
104 |
105 |
106 |
107 |
108 |
109 | text/microsoft-resx
110 |
111 |
112 | 2.0
113 |
114 |
115 | System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089
116 |
117 |
118 | System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089
119 |
120 |
121 |
122 | ..\Resources\discord_canary_64.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
123 |
124 |
125 | ..\Resources\discord_dev_32.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
126 |
127 |
128 | ..\Resources\discord_canary_16.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
129 |
130 |
131 | ..\Resources\discord_stable_64.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
132 |
133 |
134 | ..\Resources\discord_canary_32.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
135 |
136 |
137 | ..\Resources\discord_dev_64.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
138 |
139 |
140 | ..\Resources\discord_stable_32.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
141 |
142 |
143 | ..\Resources\discord_dev_16.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
144 |
145 |
146 | ..\Resources\discord_stable_16.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
147 |
148 |
149 | ..\Resources\ed_og.png;System.Drawing.Bitmap, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a
150 |
151 |
152 | require(`${process.env.injDir}/injection.js`);
153 |
154 |
155 | https://codeload.github.com/joe27g/EnhancedDiscord/zip/
156 |
157 |
--------------------------------------------------------------------------------
/plugins/css_settings.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 | const fs = require('fs');
3 | const path = require('path');
4 | const propMap = {
5 | bg_color: 'bg',
6 | bg_opacity: 'bg-overlay'
7 | }
8 |
9 | module.exports = new Plugin({
10 | name: 'CSS Settings',
11 | author: 'Joe 🎸#7070',
12 | description: 'Allows you to modify options for the default theme.',
13 | color: 'blue',
14 |
15 | load: () => {},
16 | unload: () => {},
17 |
18 | customLoad: function(prop) {
19 | const st = ED.plugins.css_loader.settings;
20 | if (!prop || !st || !st.path) return;
21 | const cssPath = path.isAbsolute(st.path) ? st.path : path.join(process.env.injDir, st.path);
22 |
23 | let content;
24 | try {
25 | content = fs.readFileSync(cssPath, 'utf-8');
26 | } catch (e) {
27 | this.error(e);
28 | return null;
29 | }
30 |
31 | const lines = content.split(/[\r\n][\r\n]?/).filter(l => l);
32 | for (const i in lines) {
33 | if (lines[i] && lines[i].match(/^\s*(?:\/\*)?\s*--/)) {
34 | const reg = new RegExp(`--${propMap[prop] || prop}: ([^;]+);`);
35 | const matches = lines[i].match(reg);
36 | if (!matches) continue;
37 | const raw = matches[1];
38 | if (!raw) continue;
39 |
40 | if (prop === 'bg' && raw.startsWith('url(')) {
41 | return raw.substring(4, raw.length - 1)
42 | } else if (prop === 'bg') {
43 | return null;
44 | }
45 | if (prop === 'bg_color' && !raw.startsWith('url(')) {
46 | return raw;
47 | } else if (prop === 'bg_color') {
48 | return null;
49 | }
50 | if (prop === 'bg_opacity' && raw.startsWith('rgba(')) {
51 | const str = raw.substring(5, raw.length - 1).split(',').pop();
52 | if (!str) return null;
53 | return 100 - Math.round(parseFloat(str)*100);
54 | } else if (prop === 'bg_opacity') {
55 | return 100;
56 | } else if (prop === 'gift-button' || prop === 'gif-picker') {
57 | return raw === 'none' ? false : true;
58 | }
59 | // TODO: proper transparency support?
60 | return raw;
61 | }
62 | }
63 | },
64 | customSave: function(prop, data) {
65 | let varName = prop, finalValue = data;
66 | switch (prop) {
67 | case 'bg':
68 | if (data) {
69 | if (!data.startsWith('https://') && !data.startsWith('http://'))
70 | EDApi.showToast('Warning: This location probably won\'t work - it needs to be a remote URL, not a local file.')
71 | finalValue = `url(${data})`;
72 | } else {
73 | finalValue = 'transparent'
74 | }
75 | break;
76 | case 'bg_color':
77 | varName = 'bg';
78 | finalValue = data || 'transparent';
79 | break;
80 | case 'bg_opacity':
81 | varName = 'bg-overlay';
82 | finalValue = `rgba(0, 0, 0, ${(100 - data) / 100.0})`;
83 | break;
84 | case 'typing-height':
85 | finalValue = finalValue || 0;
86 | break;
87 | case 'gift-button':
88 | case 'gif-picker':
89 | finalValue = finalValue ? 'flex' : 'none';
90 | break;
91 | default:
92 | finalValue = data || 'transparent';
93 | // TODO: proper transparency support?
94 | }
95 | const reg = new RegExp(`--${varName}: ([^;]+);`);
96 |
97 | const st = ED.plugins.css_loader.settings;
98 | if (!st || !st.path) return;
99 | const cssPath = path.isAbsolute(st.path) ? st.path : path.join(process.env.injDir, st.path);
100 |
101 | fs.readFile(cssPath, 'utf-8', (err, content) => {
102 | if (err) return module.exports.error(err);
103 | const lines = content.split(/\r\n/g);
104 | let changed = false;
105 | for (const i in lines) {
106 | if (lines[i] && reg.test(lines[i])) {
107 | lines[i] = lines[i].replace(reg, `--${varName}: ${finalValue};`);
108 | changed = true;
109 | }
110 | }
111 | if (changed) {
112 | fs.writeFile(cssPath, lines.join('\n'), err => {
113 | if (err) {
114 | EDApi.showToast('Error saving: '+err);
115 | return module.exports.error(err);
116 | }
117 | EDApi.showToast('Saved.');
118 | return;
119 | })
120 | } else {
121 | return EDApi.showToast('Already saved.');
122 | }
123 | });
124 | },
125 |
126 | settingsSectionName: 'Theme Settings',
127 | generateSettingsSection: function() {
128 | if (!ED.plugins.css_loader || !ED.customCss || !ED.customCss.innerHTML || (!ED.customCss.innerHTML.includes("enhanceddiscord.com/theme") && !ED.customCss.innerHTML.includes("EnhancedDiscord Theme"))) return;
129 |
130 | const els = [{
131 | type: "input:text",
132 | configName: "bg",
133 | title: "Background Image",
134 | desc: "Must be a remote URL, not a local file."
135 | }, {
136 | type: "input:colorpicker",
137 | title: 'Background Color',
138 | configName: "bg_color",
139 | desc: "Set your background to a simple color rather than an image. See the list of [valid css colors](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value).",
140 | colors: [0x3C0D0D, 0x0D3C13, 0x0D193C, 0x250D3C, 0x4B1741, 0x552A13, 0x2D261F, 0x153538, 0x000000, 0x151515],
141 | defaultColor: 0x36393F
142 | }, {
143 | type: "input:slider",
144 | configName: "bg_opacity",
145 | title: "Background Opacity",
146 | desc: "Opacity of the background image underneath other elements. Lowering it increases readability.",
147 | defaultValue: 20,
148 | highlightDefaultValue: true,
149 | markers: [
150 | 0,10,20,30,40,50,60,69,80,90,100
151 | ],
152 | formatTooltip: e => e.toFixed(0)+'%',
153 | minValue: 0,
154 | maxValue: 100,
155 | }, {
156 | type: "input:boolean",
157 | title: "Nitro gift button",
158 | configName: "gift-button",
159 | note: "Show Nitro gift button in the chat textarea (next to emoji picker or gif picker)"
160 | }, {
161 | type: "input:boolean",
162 | title: "GIF picker",
163 | configName: "gif-picker",
164 | note: "Show GIF picker button in the chat textarea (next to emoji picker)"
165 | }, {
166 | type: "input:colorpicker",
167 | title: 'Accent Color',
168 | configName: "accent",
169 | desc: "Prominent color in the UI. Affects buttons, switches, mentions, etc.",
170 | colors: [0x7390DB], // TODO: need more defaults
171 | defaultColor: 0x990000
172 | }, {
173 | type: "input:colorpicker",
174 | title: 'Accent Background',
175 | configName: "accent-back",
176 | desc: "Background for mentions and other misc. things.",
177 | colors: [], // TODO: need more defaults
178 | defaultColor: 0x660000
179 | }, {
180 | type: "input:colorpicker",
181 | title: 'Bright Accent Color',
182 | configName: "accent-bright",
183 | desc: "Color of mentions while hovering and other misc. things.",
184 | colors: [0xFFFFFF], // TODO: need more defaults
185 | defaultColor: 0xFF0000
186 | }, {
187 | type: "input:colorpicker",
188 | title: 'Bright Accent Background',
189 | configName: "accent-back-bright",
190 | desc: "Background for mentions while hovering and other misc. things.",
191 | colors: [], // TODO: need more defaults
192 | defaultColor: 0x880000
193 | }, {
194 | type: "input:colorpicker",
195 | title: 'Icon Color',
196 | configName: "icon-color",
197 | desc: "Color of icons for channels, home sections, etc.",
198 | colors: [], // TODO: need more defaults
199 | defaultColor: 0xFAA61A
200 | }, {
201 | type: "input:colorpicker",
202 | title: 'Link Color',
203 | configName: "link-color",
204 | desc: "Color of links.",
205 | colors: [], // TODO: need more defaults
206 | defaultColor: 0xFAA61A
207 | }, {
208 | type: "input:colorpicker",
209 | title: 'Hovered Link Color',
210 | configName: "link-color-hover",
211 | desc: "Color of links while hovering over them.",
212 | colors: [], // TODO: need more defaults
213 | defaultColor: 0xFAD61A
214 | }, {
215 | type: "input:colorpicker",
216 | title: 'Popup Background',
217 | configName: "popup-background",
218 | desc: "Background color of modals and popups, such as pinned messages, context menus, and confirmation dialogs.",
219 | colors: [], // TODO: need more defaults
220 | defaultColor: 0x222222
221 | }, {
222 | type: "input:colorpicker",
223 | title: 'Popup Headers & Footers',
224 | configName: "popup-highlight",
225 | desc: "Background color of headers and footers on modals and popups.",
226 | colors: [], // TODO: need more defaults
227 | defaultColor: 0x333333
228 | }, {
229 | type: "input:colorpicker",
230 | title: 'Unread Color',
231 | configName: "unread-color",
232 | desc: "Color of channel/server unread or selected indicators.",
233 | colors: [0xFFFFFF], // TODO: need more defaults
234 | defaultColor: 0x990000
235 | }, {
236 | type: "input:text",
237 | configName: "typing-height",
238 | title: "Typing Height",
239 | desc: "Height of typing element and margin underneath chat to allow space for it."
240 | }];
241 | els.forEach(item => {
242 | let dat = this.customLoad(item.configName);
243 | if (dat && /^#\d{3}$/.test(dat)) {
244 | dat = dat[0]+dat[1]+dat[1]+dat[2]+dat[2]+dat[3]+dat[3];
245 | }
246 | // TODO: proper transparency support?
247 | if (item.type === "input:colorpicker")
248 | item.currentColor = dat ? parseInt(dat.substr(1), 16) : null;
249 | });
250 | return els;
251 | }
252 | });
253 |
--------------------------------------------------------------------------------
/plugins/hidden_channels.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 |
3 | let getChannel, g_dc, g_cat, ha, disp, chanM, fm, reb, sv, cs, csp, ghp, gs, gsr, pf, sw = {}, g = {}, ai = {};
4 |
5 | module.exports = new Plugin({
6 | name: 'Hidden Channels',
7 | description: 'Shows hidden channels and lets you view server permissions.',
8 | color: 'magenta',
9 | author: 'Joe 🎸#7070',
10 |
11 | load: async function() {
12 | disp = EDApi.findModule("dispatch");
13 | getChannel = EDApi.findModule('getChannel').getChannel;
14 | sw = EDApi.findModule('switchItem');
15 | g = EDApi.findModule(m => m.group && m.item);
16 | ai = EDApi.findModule('actionIcon');
17 |
18 | const getUser = EDApi.findModule('getCurrentUser').getCurrentUser;
19 | const getAllChannels = EDApi.findModule('getMutableGuildChannels').getMutableGuildChannels;
20 | const can = EDApi.findModule('computePermissions').can;
21 |
22 | g_dc = EDApi.findModule('getDefaultChannel');
23 | EDApi.monkeyPatch(g_dc, 'getChannels', b => {
24 | const og = b.callOriginalMethod(b.methodArguments);
25 | if (!b.methodArguments[0]) return og;
26 | const hidden = [], allChans = getAllChannels();
27 | for (const i in allChans) {
28 | if (allChans[i].guild_id === b.methodArguments[0]) {
29 | if (allChans[i].type !== 4 && !can({data: 1024n}, getUser(), getChannel(allChans[i].id))) {
30 | hidden.push(allChans[i]);
31 | }
32 | }
33 | }
34 | og.HIDDEN = hidden;
35 | return og;
36 | });
37 | chanM = EDApi.findModule(m => m.prototype && m.prototype.isManaged);
38 | chanM.prototype.isHidden = function() {
39 | return [0, 4, 5].includes(this.type) && !can({data: 1024n}, getUser(), this);
40 | }
41 |
42 | g_cat = EDApi.findModule(m => m.getCategories && !m.EMOJI_NAME_RE);
43 | EDApi.monkeyPatch(g_cat, 'getCategories', b => {
44 | const og = b.callOriginalMethod(b.methodArguments);
45 | const chs = g_dc.getChannels(b.methodArguments[0]);
46 | chs.HIDDEN.forEach(c => {
47 | const result = og[c.parent_id || "null"].filter(item => item.channel.id === c.id);
48 | if (result.length) return; // already added
49 | og[c.parent_id || "null"].push({channel: c, index: 0})
50 | });
51 | return og;
52 | });
53 |
54 | ha = EDApi.findModule('hasUnread').__proto__;
55 | EDApi.monkeyPatch(ha, 'hasUnread', function(b) {
56 | if (getChannel(b.methodArguments[0]) && getChannel(b.methodArguments[0]).isHidden())
57 | return false; // don't show hidden channels as unread.
58 | return b.callOriginalMethod(b.methodArguments);
59 | });
60 | EDApi.monkeyPatch(ha, 'hasUnreadPins', function(b) {
61 | if (getChannel(b.methodArguments[0]) && getChannel(b.methodArguments[0]).isHidden())
62 | return false; // don't show icon on hidden channel pins.
63 | return b.callOriginalMethod(b.methodArguments);
64 | });
65 |
66 | disp.subscribe("CHANNEL_SELECT", module.exports.dispatchSubscription);
67 |
68 | fm = EDApi.findModule("fetchMessages");
69 | EDApi.monkeyPatch(fm, "fetchMessages", function(b) {
70 | if (getChannel(b.methodArguments[0]) && getChannel(b.methodArguments[0]).isHidden()) return;
71 | return b.callOriginalMethod(b.methodArguments);
72 | });
73 |
74 | const clk = window.EDApi.findModuleByDisplayName("Clickable")
75 | const Tooltip = window.EDApi.findModule('TooltipContainer').TooltipContainer;
76 | const { Messages } = window.EDApi.findModule('Messages');
77 | const getIcon = window.EDApi.findModule(m => m.id && typeof m.keys === 'function' && m.keys().includes('./Gear'));
78 | const Gear = getIcon('./Gear').default;
79 |
80 | reb = window.EDApi.findModule(m => m.default && m.default.prototype && m.default.prototype.renderEditButton).default.prototype;
81 | window.EDApi.monkeyPatch(reb, "renderEditButton", function(b) {
82 | return window.EDApi.React.createElement(Tooltip, { text: Messages.EDIT_CHANNEL }, window.EDApi.React.createElement(clk, {
83 | className: ai.iconItem,
84 | onClick: function() {
85 | module.exports._editingGuild = null;
86 | module.exports._editingChannel = b.thisObject.props.channel.id;
87 | return b.thisObject.handleEditClick.apply(b.thisObject, arguments);
88 | },
89 | onMouseEnter: b.thisObject.props.onMouseEnter,
90 | onMouseLeave: b.thisObject.props.onMouseLeave
91 | }, window.EDApi.React.createElement(Gear, {
92 | width: 16,
93 | height: 16,
94 | className: ai.actionIcon
95 | })));
96 | });
97 |
98 | sv = EDApi.findModuleByDisplayName("SettingsView").prototype;
99 | EDApi.monkeyPatch(sv, 'getPredicateSections', {before: b => {
100 | const permSect = b.thisObject.props.sections.find(item => item.section === 'PERMISSIONS');
101 | if (permSect) permSect.predicate = () => true;
102 | }, silent: true});
103 |
104 | cs = EDApi.findModuleByDisplayName("FluxContainer(ChannelSettings)").prototype;
105 | EDApi.monkeyPatch(cs, 'render', b => {
106 | const egg = b.callOriginalMethod(b.methodArguments);
107 | egg.props.canManageRoles = true;
108 | return egg;
109 | });
110 |
111 | csp = EDApi.findModuleByDisplayName("FluxContainer(ChannelSettingsPermissions)").prototype;
112 | EDApi.monkeyPatch(csp, 'render', b => {
113 | const egg = b.callOriginalMethod(b.methodArguments);
114 | const chan = getChannel(egg.props.channel.id);
115 | if (!chan || !chan.isHidden()) return egg;
116 | egg.props.canSyncChannel = false;
117 | egg.props.locked = true;
118 | setTimeout(() => {
119 | document.querySelectorAll('.'+g.group).forEach(elem => elem.style = "opacity: 0.5; pointer-events: none;");
120 | });
121 | return egg;
122 | });
123 |
124 | /*ghp = EDApi.findModuleByDisplayName("FluxContainer(GuildHeaderPopout)").prototype;
125 | EDApi.monkeyPatch(ghp, 'render', b => {
126 | const egg = b.callOriginalMethod(b.methodArguments);
127 | egg.props.canAccessSettings = true;
128 | return egg;
129 | });
130 |
131 | gs = EDApi.findModuleByDisplayName("FluxContainer(GuildSettings)").prototype;
132 | EDApi.monkeyPatch(gs, 'render', b => {
133 | const egg = b.callOriginalMethod(b.methodArguments);
134 | module.exports._editingChannel = null;
135 | module.exports._editingGuild = egg.props.guild.id;
136 | egg.props.canManageRoles = true;
137 | return egg;
138 | });*/
139 |
140 | const cancan = EDApi.findModuleByProps('can').can;
141 | gsr = EDApi.findModuleByDisplayName("FluxContainer(GuildSettingsRoles)").prototype;
142 | EDApi.monkeyPatch(gsr, 'render', b => {
143 | const egg = b.callOriginalMethod(b.methodArguments);
144 | const hasPerm = cancan({data: 268435456n}, { guildId: egg.props.guild.id });
145 | if (hasPerm) return;
146 | setTimeout(() => {
147 | document.querySelectorAll('.'+sw.switchItem).forEach(elem => elem.classList.add(sw.disabled));
148 | });
149 | return egg;
150 | });
151 |
152 | const getGuild = EDApi.findModule('getGuild').getGuild;
153 | pf = EDApi.findModuleByDisplayName("PermissionsForm").prototype;
154 | EDApi.monkeyPatch(pf, 'render', b => {
155 | const egg = b.callOriginalMethod(b.methodArguments);
156 | const guild = module.exports._editingGuild ? getGuild(module.exports._editingGuild) : null;
157 | const channel = module.exports._editingChannel ? getChannel(module.exports._editingChannel) : null;
158 | if (!guild && !channel) return egg;
159 | const hasPerm = cancan({data: 268435456n}, guild ? { guildId: guild.id } : { channelId: channel.id });
160 | if (hasPerm) return egg;
161 |
162 | if (!egg.props.children || !egg.props.children[1]) return egg;
163 | egg.props.children[1].forEach(item => {item.disabled = true; item.props.disabled = true;});
164 | return egg;
165 | });
166 | },
167 | unload: function() {
168 | g_dc.getChannels.unpatch();
169 | g_cat.getCategories.unpatch();
170 | ha.hasUnread.unpatch();
171 | ha.hasUnreadPins.unpatch();
172 | fm.fetchMessages.unpatch();
173 | reb.renderEditButton.unpatch();
174 |
175 | for (const mod of [sv, cs, csp, ghp, gs, gsr, pf])
176 | if (mod && mod.render && mod.render.unpatch) mod.render.unpatch();
177 |
178 | disp.unsubscribe("CHANNEL_SELECT", module.exports.dispatchSubscription);
179 | },
180 | dispatchSubscription: function (data) {
181 | if (data.type !== "CHANNEL_SELECT") return;
182 |
183 | if (getChannel(data.channelId) && getChannel(data.channelId).isHidden()) {
184 | setTimeout(module.exports.attachHiddenChanNotice);
185 | }
186 | },
187 | attachHiddenChanNotice: function () {
188 | const messagesWrapper = document.querySelector(`.${EDApi.findModule("messagesWrapper").messagesWrapper}`);
189 | if (!messagesWrapper) return;
190 |
191 | messagesWrapper.firstChild.style.display = "none"; // Remove messages shit.
192 | messagesWrapper.parentElement.children[1].style.display = "none"; // Remove message box.
193 | messagesWrapper.parentElement.parentElement.children[1].style.display = "none"; // Remove user list.
194 |
195 | const toolbar = document.querySelector("."+EDApi.findModule(m => {
196 | if (m instanceof Window) return;
197 | if (m.toolbar && m.selected) return m;
198 | }).toolbar);
199 |
200 | toolbar.style.display = "none";
201 |
202 | const hiddenChannelNotif = document.createElement("div");
203 |
204 | // Class name modules
205 | const txt = EDApi.findModule("h5");
206 | const flx = EDApi.findModule("flex");
207 |
208 | hiddenChannelNotif.className = flx.flexCenter;
209 | hiddenChannelNotif.style.width = "100%";
210 |
211 | hiddenChannelNotif.innerHTML = `
212 |
213 |
This is a hidden channel.
214 | You cannot see the contents of this channel. However, you may see its name and topic.
215 | `;
216 |
217 | messagesWrapper.appendChild(hiddenChannelNotif);
218 | }
219 | });
220 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Form1.Designer.cs:
--------------------------------------------------------------------------------
1 | namespace EnhancedDiscordUI
2 | {
3 | partial class EDInstaller
4 | {
5 | ///
6 | /// Required designer variable.
7 | ///
8 | private System.ComponentModel.IContainer components = null;
9 |
10 | ///
11 | /// Clean up any resources being used.
12 | ///
13 | /// true if managed resources should be disposed; otherwise, false.
14 | protected override void Dispose(bool disposing)
15 | {
16 | if (disposing && (components != null))
17 | {
18 | components.Dispose();
19 | }
20 | base.Dispose(disposing);
21 | }
22 |
23 | #region Windows Form Designer generated code
24 |
25 | ///
26 | /// Required method for Designer support - do not modify
27 | /// the contents of this method with the code editor.
28 | ///
29 | private void InitializeComponent()
30 | {
31 | this.components = new System.ComponentModel.Container();
32 | System.ComponentModel.ComponentResourceManager resources = new System.ComponentModel.ComponentResourceManager(typeof(EDInstaller));
33 | this.InstallButton = new System.Windows.Forms.Button();
34 | this.Title = new System.Windows.Forms.Label();
35 | this.UninstallButton = new System.Windows.Forms.Button();
36 | this.UpdateButton = new System.Windows.Forms.Button();
37 | this.InstallProgress = new System.Windows.Forms.ProgressBar();
38 | this.StatusLabel2 = new System.Windows.Forms.Label();
39 | this.StatusLabel = new System.Windows.Forms.Label();
40 | this.StatusCloseButton = new System.Windows.Forms.Button();
41 | this.StatusText = new System.Windows.Forms.TextBox();
42 | this.toolTip1 = new System.Windows.Forms.ToolTip(this.components);
43 | this.DevButton = new System.Windows.Forms.Button();
44 | this.CanaryButton = new System.Windows.Forms.Button();
45 | this.PTBButton = new System.Windows.Forms.Button();
46 | this.StableButton = new System.Windows.Forms.Button();
47 | this.ReinjectButton = new System.Windows.Forms.Button();
48 | this.pictureBox1 = new System.Windows.Forms.PictureBox();
49 | this.OpenFolderButton = new System.Windows.Forms.Button();
50 | this.BetaRadio = new System.Windows.Forms.RadioButton();
51 | ((System.ComponentModel.ISupportInitialize)(this.pictureBox1)).BeginInit();
52 | this.SuspendLayout();
53 | //
54 | // InstallButton
55 | //
56 | this.InstallButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
57 | this.InstallButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
58 | this.InstallButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
59 | this.InstallButton.ForeColor = System.Drawing.Color.WhiteSmoke;
60 | this.InstallButton.Location = new System.Drawing.Point(155, 127);
61 | this.InstallButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
62 | this.InstallButton.Name = "InstallButton";
63 | this.InstallButton.Size = new System.Drawing.Size(131, 30);
64 | this.InstallButton.TabIndex = 0;
65 | this.InstallButton.Text = "Install";
66 | this.toolTip1.SetToolTip(this.InstallButton, "Downloads and injects ED into your Discord client.");
67 | this.InstallButton.UseVisualStyleBackColor = true;
68 | this.InstallButton.Click += new System.EventHandler(this.InstallButton_Click);
69 | //
70 | // Title
71 | //
72 | this.Title.AutoSize = true;
73 | this.Title.Font = new System.Drawing.Font("Segoe UI Semibold", 18F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
74 | this.Title.ForeColor = System.Drawing.Color.WhiteSmoke;
75 | this.Title.Location = new System.Drawing.Point(131, 11);
76 | this.Title.Name = "Title";
77 | this.Title.Size = new System.Drawing.Size(255, 41);
78 | this.Title.TabIndex = 1;
79 | this.Title.Text = "EnhancedDiscord";
80 | //
81 | // UninstallButton
82 | //
83 | this.UninstallButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
84 | this.UninstallButton.Enabled = false;
85 | this.UninstallButton.FlatAppearance.BorderColor = System.Drawing.Color.WhiteSmoke;
86 | this.UninstallButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
87 | this.UninstallButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
88 | this.UninstallButton.ForeColor = System.Drawing.Color.WhiteSmoke;
89 | this.UninstallButton.Location = new System.Drawing.Point(155, 162);
90 | this.UninstallButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
91 | this.UninstallButton.Name = "UninstallButton";
92 | this.UninstallButton.Size = new System.Drawing.Size(131, 30);
93 | this.UninstallButton.TabIndex = 2;
94 | this.UninstallButton.Text = "Uninstall";
95 | this.toolTip1.SetToolTip(this.UninstallButton, "Uninjects ED and prompts you to delete ED\'s files.");
96 | this.UninstallButton.UseVisualStyleBackColor = true;
97 | this.UninstallButton.Click += new System.EventHandler(this.UninstallButton_Click);
98 | //
99 | // UpdateButton
100 | //
101 | this.UpdateButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
102 | this.UpdateButton.Enabled = false;
103 | this.UpdateButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
104 | this.UpdateButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
105 | this.UpdateButton.ForeColor = System.Drawing.Color.WhiteSmoke;
106 | this.UpdateButton.Location = new System.Drawing.Point(87, 197);
107 | this.UpdateButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
108 | this.UpdateButton.Name = "UpdateButton";
109 | this.UpdateButton.Size = new System.Drawing.Size(131, 30);
110 | this.UpdateButton.TabIndex = 3;
111 | this.UpdateButton.Text = "Update";
112 | this.toolTip1.SetToolTip(this.UpdateButton, "Replaces the ED files with the most recent ones.");
113 | this.UpdateButton.UseVisualStyleBackColor = true;
114 | this.UpdateButton.Click += new System.EventHandler(this.UpdateButton_Click);
115 | //
116 | // InstallProgress
117 | //
118 | this.InstallProgress.Location = new System.Drawing.Point(12, 218);
119 | this.InstallProgress.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
120 | this.InstallProgress.Name = "InstallProgress";
121 | this.InstallProgress.Size = new System.Drawing.Size(415, 23);
122 | this.InstallProgress.Style = System.Windows.Forms.ProgressBarStyle.Continuous;
123 | this.InstallProgress.TabIndex = 5;
124 | this.InstallProgress.Visible = false;
125 | //
126 | // StatusLabel2
127 | //
128 | this.StatusLabel2.AutoSize = true;
129 | this.StatusLabel2.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
130 | this.StatusLabel2.ForeColor = System.Drawing.Color.WhiteSmoke;
131 | this.StatusLabel2.Location = new System.Drawing.Point(9, 190);
132 | this.StatusLabel2.Name = "StatusLabel2";
133 | this.StatusLabel2.Size = new System.Drawing.Size(89, 19);
134 | this.StatusLabel2.TabIndex = 8;
135 | this.StatusLabel2.Text = "Lorem ipsum";
136 | this.StatusLabel2.Visible = false;
137 | //
138 | // StatusLabel
139 | //
140 | this.StatusLabel.AutoSize = true;
141 | this.StatusLabel.Font = new System.Drawing.Font("Segoe UI", 12F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
142 | this.StatusLabel.ForeColor = System.Drawing.Color.WhiteSmoke;
143 | this.StatusLabel.Location = new System.Drawing.Point(8, 167);
144 | this.StatusLabel.Name = "StatusLabel";
145 | this.StatusLabel.Size = new System.Drawing.Size(165, 28);
146 | this.StatusLabel.TabIndex = 9;
147 | this.StatusLabel.Text = "Installation failed.";
148 | this.StatusLabel.Visible = false;
149 | //
150 | // StatusCloseButton
151 | //
152 | this.StatusCloseButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
153 | this.StatusCloseButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
154 | this.StatusCloseButton.ForeColor = System.Drawing.Color.WhiteSmoke;
155 | this.StatusCloseButton.Location = new System.Drawing.Point(251, 128);
156 | this.StatusCloseButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
157 | this.StatusCloseButton.Name = "StatusCloseButton";
158 | this.StatusCloseButton.Size = new System.Drawing.Size(64, 30);
159 | this.StatusCloseButton.TabIndex = 10;
160 | this.StatusCloseButton.Text = "Close";
161 | this.StatusCloseButton.UseVisualStyleBackColor = true;
162 | this.StatusCloseButton.Visible = false;
163 | this.StatusCloseButton.Click += new System.EventHandler(this.StatusCloseButton_Click);
164 | //
165 | // StatusText
166 | //
167 | this.StatusText.BackColor = System.Drawing.Color.FromArgb(((int)(((byte)(31)))), ((int)(((byte)(36)))), ((int)(((byte)(36)))));
168 | this.StatusText.BorderStyle = System.Windows.Forms.BorderStyle.None;
169 | this.StatusText.Font = new System.Drawing.Font("Segoe UI", 10.2F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
170 | this.StatusText.ForeColor = System.Drawing.Color.WhiteSmoke;
171 | this.StatusText.Location = new System.Drawing.Point(11, 68);
172 | this.StatusText.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
173 | this.StatusText.Multiline = true;
174 | this.StatusText.Name = "StatusText";
175 | this.StatusText.ReadOnly = true;
176 | this.StatusText.Size = new System.Drawing.Size(416, 26);
177 | this.StatusText.TabIndex = 11;
178 | this.StatusText.Text = "Make sure to launch Discord before installing!";
179 | this.StatusText.TextAlign = System.Windows.Forms.HorizontalAlignment.Center;
180 | //
181 | // DevButton
182 | //
183 | this.DevButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
184 | this.DevButton.FlatAppearance.BorderSize = 2;
185 | this.DevButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
186 | this.DevButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
187 | this.DevButton.ForeColor = System.Drawing.Color.White;
188 | this.DevButton.Image = global::EnhancedDiscordUI.Properties.Resources.discord_dev_64;
189 | this.DevButton.ImageAlign = System.Drawing.ContentAlignment.TopCenter;
190 | this.DevButton.Location = new System.Drawing.Point(320, 98);
191 | this.DevButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
192 | this.DevButton.Name = "DevButton";
193 | this.DevButton.RightToLeft = System.Windows.Forms.RightToLeft.Yes;
194 | this.DevButton.Size = new System.Drawing.Size(91, 110);
195 | this.DevButton.TabIndex = 15;
196 | this.DevButton.Text = "Dev";
197 | this.DevButton.TextAlign = System.Drawing.ContentAlignment.BottomCenter;
198 | this.toolTip1.SetToolTip(this.DevButton, "Discord Development (aka Local.)");
199 | this.DevButton.UseVisualStyleBackColor = true;
200 | this.DevButton.Visible = false;
201 | this.DevButton.Click += new System.EventHandler(this.DevButton_Click);
202 | //
203 | // CanaryButton
204 | //
205 | this.CanaryButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
206 | this.CanaryButton.FlatAppearance.BorderSize = 2;
207 | this.CanaryButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
208 | this.CanaryButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
209 | this.CanaryButton.ForeColor = System.Drawing.Color.Gold;
210 | this.CanaryButton.Image = global::EnhancedDiscordUI.Properties.Resources.discord_canary_64;
211 | this.CanaryButton.ImageAlign = System.Drawing.ContentAlignment.TopCenter;
212 | this.CanaryButton.Location = new System.Drawing.Point(224, 98);
213 | this.CanaryButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
214 | this.CanaryButton.Name = "CanaryButton";
215 | this.CanaryButton.RightToLeft = System.Windows.Forms.RightToLeft.Yes;
216 | this.CanaryButton.Size = new System.Drawing.Size(91, 110);
217 | this.CanaryButton.TabIndex = 14;
218 | this.CanaryButton.Text = "Canary";
219 | this.CanaryButton.TextAlign = System.Drawing.ContentAlignment.BottomCenter;
220 | this.toolTip1.SetToolTip(this.CanaryButton, "Discord Canary");
221 | this.CanaryButton.UseVisualStyleBackColor = true;
222 | this.CanaryButton.Visible = false;
223 | this.CanaryButton.Click += new System.EventHandler(this.CanaryButton_Click);
224 | //
225 | // PTBButton
226 | //
227 | this.PTBButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
228 | this.PTBButton.FlatAppearance.BorderSize = 2;
229 | this.PTBButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
230 | this.PTBButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
231 | this.PTBButton.ForeColor = System.Drawing.Color.SteelBlue;
232 | this.PTBButton.Image = global::EnhancedDiscordUI.Properties.Resources.discord_stable_64;
233 | this.PTBButton.ImageAlign = System.Drawing.ContentAlignment.TopCenter;
234 | this.PTBButton.Location = new System.Drawing.Point(128, 98);
235 | this.PTBButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
236 | this.PTBButton.Name = "PTBButton";
237 | this.PTBButton.RightToLeft = System.Windows.Forms.RightToLeft.Yes;
238 | this.PTBButton.Size = new System.Drawing.Size(91, 110);
239 | this.PTBButton.TabIndex = 13;
240 | this.PTBButton.Text = "PTB";
241 | this.PTBButton.TextAlign = System.Drawing.ContentAlignment.BottomCenter;
242 | this.toolTip1.SetToolTip(this.PTBButton, "Discord PTB (Public Test Build)");
243 | this.PTBButton.UseVisualStyleBackColor = true;
244 | this.PTBButton.Visible = false;
245 | this.PTBButton.Click += new System.EventHandler(this.PTBButton_Click);
246 | //
247 | // StableButton
248 | //
249 | this.StableButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
250 | this.StableButton.FlatAppearance.BorderSize = 2;
251 | this.StableButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
252 | this.StableButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
253 | this.StableButton.ForeColor = System.Drawing.Color.SteelBlue;
254 | this.StableButton.Image = global::EnhancedDiscordUI.Properties.Resources.discord_stable_64;
255 | this.StableButton.ImageAlign = System.Drawing.ContentAlignment.TopCenter;
256 | this.StableButton.Location = new System.Drawing.Point(32, 98);
257 | this.StableButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
258 | this.StableButton.Name = "StableButton";
259 | this.StableButton.RightToLeft = System.Windows.Forms.RightToLeft.Yes;
260 | this.StableButton.Size = new System.Drawing.Size(91, 110);
261 | this.StableButton.TabIndex = 12;
262 | this.StableButton.Text = "Stable";
263 | this.StableButton.TextAlign = System.Drawing.ContentAlignment.BottomCenter;
264 | this.toolTip1.SetToolTip(this.StableButton, "Normal version of Discord.");
265 | this.StableButton.UseVisualStyleBackColor = true;
266 | this.StableButton.Visible = false;
267 | this.StableButton.Click += new System.EventHandler(this.StableButton_Click);
268 | //
269 | // ReinjectButton
270 | //
271 | this.ReinjectButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
272 | this.ReinjectButton.Enabled = false;
273 | this.ReinjectButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
274 | this.ReinjectButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
275 | this.ReinjectButton.ForeColor = System.Drawing.Color.WhiteSmoke;
276 | this.ReinjectButton.Location = new System.Drawing.Point(225, 197);
277 | this.ReinjectButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
278 | this.ReinjectButton.Name = "ReinjectButton";
279 | this.ReinjectButton.Size = new System.Drawing.Size(131, 30);
280 | this.ReinjectButton.TabIndex = 17;
281 | this.ReinjectButton.Text = "Reinject";
282 | this.toolTip1.SetToolTip(this.ReinjectButton, "Reinjects without changing your ED folder; useful after Discord updates.");
283 | this.ReinjectButton.UseVisualStyleBackColor = true;
284 | this.ReinjectButton.Click += new System.EventHandler(this.ReinjectButton_Click);
285 | //
286 | // pictureBox1
287 | //
288 | this.pictureBox1.Image = global::EnhancedDiscordUI.Properties.Resources.ed_og;
289 | this.pictureBox1.Location = new System.Drawing.Point(41, 12);
290 | this.pictureBox1.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
291 | this.pictureBox1.Name = "pictureBox1";
292 | this.pictureBox1.Size = new System.Drawing.Size(87, 50);
293 | this.pictureBox1.SizeMode = System.Windows.Forms.PictureBoxSizeMode.StretchImage;
294 | this.pictureBox1.TabIndex = 4;
295 | this.pictureBox1.TabStop = false;
296 | //
297 | // OpenFolderButton
298 | //
299 | this.OpenFolderButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
300 | this.OpenFolderButton.Font = new System.Drawing.Font("Segoe UI", 7.8F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
301 | this.OpenFolderButton.ForeColor = System.Drawing.Color.WhiteSmoke;
302 | this.OpenFolderButton.Location = new System.Drawing.Point(113, 128);
303 | this.OpenFolderButton.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
304 | this.OpenFolderButton.Name = "OpenFolderButton";
305 | this.OpenFolderButton.Size = new System.Drawing.Size(125, 30);
306 | this.OpenFolderButton.TabIndex = 16;
307 | this.OpenFolderButton.Text = "Open Folder";
308 | this.OpenFolderButton.UseVisualStyleBackColor = true;
309 | this.OpenFolderButton.Visible = false;
310 | this.OpenFolderButton.Click += new System.EventHandler(this.OpenFolderButton_Click);
311 | //
312 | // BetaRadio
313 | //
314 | this.BetaRadio.AutoSize = true;
315 | this.BetaRadio.Font = new System.Drawing.Font("Segoe UI", 10.2F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
316 | this.BetaRadio.ForeColor = System.Drawing.SystemColors.ControlLightLight;
317 | this.BetaRadio.Location = new System.Drawing.Point(115, 95);
318 | this.BetaRadio.Name = "BetaRadio";
319 | this.BetaRadio.Size = new System.Drawing.Size(200, 27);
320 | this.BetaRadio.TabIndex = 18;
321 | this.BetaRadio.TabStop = true;
322 | this.BetaRadio.Text = "Opt-in to beta version";
323 | this.BetaRadio.UseVisualStyleBackColor = true;
324 | //
325 | // EDInstaller
326 | //
327 | this.AutoScaleDimensions = new System.Drawing.SizeF(8F, 16F);
328 | this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
329 | this.BackColor = System.Drawing.Color.FromArgb(((int)(((byte)(31)))), ((int)(((byte)(36)))), ((int)(((byte)(36)))));
330 | this.ClientSize = new System.Drawing.Size(439, 254);
331 | this.Controls.Add(this.BetaRadio);
332 | this.Controls.Add(this.ReinjectButton);
333 | this.Controls.Add(this.OpenFolderButton);
334 | this.Controls.Add(this.UninstallButton);
335 | this.Controls.Add(this.StatusCloseButton);
336 | this.Controls.Add(this.StatusLabel);
337 | this.Controls.Add(this.StatusLabel2);
338 | this.Controls.Add(this.InstallProgress);
339 | this.Controls.Add(this.pictureBox1);
340 | this.Controls.Add(this.UpdateButton);
341 | this.Controls.Add(this.Title);
342 | this.Controls.Add(this.InstallButton);
343 | this.Controls.Add(this.StatusText);
344 | this.Controls.Add(this.StableButton);
345 | this.Controls.Add(this.DevButton);
346 | this.Controls.Add(this.CanaryButton);
347 | this.Controls.Add(this.PTBButton);
348 | this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedSingle;
349 | this.Icon = ((System.Drawing.Icon)(resources.GetObject("$this.Icon")));
350 | this.Margin = new System.Windows.Forms.Padding(3, 2, 3, 2);
351 | this.Name = "EDInstaller";
352 | this.Text = "EnhancedDiscord Installer";
353 | ((System.ComponentModel.ISupportInitialize)(this.pictureBox1)).EndInit();
354 | this.ResumeLayout(false);
355 | this.PerformLayout();
356 |
357 | }
358 |
359 | #endregion
360 |
361 | private System.Windows.Forms.Button InstallButton;
362 | private System.Windows.Forms.Label Title;
363 | private System.Windows.Forms.Button UninstallButton;
364 | private System.Windows.Forms.Button UpdateButton;
365 | private System.Windows.Forms.PictureBox pictureBox1;
366 | private System.Windows.Forms.ProgressBar InstallProgress;
367 | private System.Windows.Forms.Label StatusLabel2;
368 | private System.Windows.Forms.Label StatusLabel;
369 | private System.Windows.Forms.Button StatusCloseButton;
370 | private System.Windows.Forms.TextBox StatusText;
371 | private System.Windows.Forms.ToolTip toolTip1;
372 | private System.Windows.Forms.Button StableButton;
373 | private System.Windows.Forms.Button PTBButton;
374 | private System.Windows.Forms.Button CanaryButton;
375 | private System.Windows.Forms.Button DevButton;
376 | private System.Windows.Forms.Button OpenFolderButton;
377 | private System.Windows.Forms.Button ReinjectButton;
378 | private System.Windows.Forms.RadioButton BetaRadio;
379 | }
380 | }
381 |
382 |
--------------------------------------------------------------------------------
/dom_shit.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 | const fs = require('fs');
3 | const electron = require('electron');
4 |
5 | const mainProcessInfo = {
6 | originalNodeModulesPath: electron.ipcRenderer.sendSync('main-process-info', 'original-node-modules-path'),
7 | originalPreloadScript: electron.ipcRenderer.sendSync('main-process-info', 'original-preload-script')
8 | };
9 | const Module = require('module');
10 | Module.globalPaths.push(mainProcessInfo.originalNodeModulesPath);
11 | if (mainProcessInfo.originalPreloadScript) {
12 | process.electronBinding('command_line').appendSwitch('preload', mainProcessInfo.originalPreloadScript);
13 | // This hack is no longer needed due to context isolation having to be on
14 | //electron.contextBridge.exposeInMainWorld = (key, val) => window[key] = val; // Expose DiscordNative
15 | require(mainProcessInfo.originalPreloadScript);
16 | }
17 |
18 | //electron.ipcRenderer.sendSync('current-web-contents');
19 |
20 | //Get inject directory
21 | if (!process.env.injDir) process.env.injDir = __dirname;
22 |
23 | //set up global functions
24 | const c = {
25 | log: function(msg, plugin) {
26 | if (plugin && plugin.name)
27 | console.log(`%c[EnhancedDiscord] %c[${plugin.name}]`, 'color: red;', `color: ${plugin.color}`, msg);
28 | else console.log('%c[EnhancedDiscord]', 'color: red;', msg);
29 | },
30 | info: function(msg, plugin) {
31 | if (plugin && plugin.name)
32 | console.info(`%c[EnhancedDiscord] %c[${plugin.name}]`, 'color: red;', `color: ${plugin.color}`, msg);
33 | else console.info('%c[EnhancedDiscord]', 'color: red;', msg);
34 | },
35 | warn: function(msg, plugin) {
36 | if (plugin && plugin.name)
37 | console.warn(`%c[EnhancedDiscord] %c[${plugin.name}]`, 'color: red;', `color: ${plugin.color}`, msg);
38 | else console.warn('%c[EnhancedDiscord]', 'color: red;', msg);
39 | },
40 | error: function(msg, plugin) {
41 | if (plugin && plugin.name)
42 | console.error(`%c[EnhancedDiscord] %c[${plugin.name}]`, 'color: red;', `color: ${plugin.color}`, msg);
43 | else console.error('%c[EnhancedDiscord]', 'color: red;', msg);
44 | },
45 | sleep: function(ms) {
46 | return new Promise(resolve => {
47 | setTimeout(resolve, ms);
48 | });
49 | }
50 | };
51 |
52 | // config util
53 | window.ED = { plugins: {}, version: '2.8.1' };
54 | Object.defineProperty(ED, 'config', {
55 | get: function() {
56 | let conf;
57 | try{
58 | conf = require('./config.json');
59 | } catch (err) {
60 | if(err.code !== 'MODULE_NOT_FOUND')
61 | c.error(err);
62 | conf = {};
63 | }
64 | return conf;
65 | },
66 | set: function(newSets = {}) {
67 | let confPath;
68 | let bDelCache;
69 | try{
70 | confPath = require.resolve('./config.json');
71 | bDelCache = true;
72 | } catch (err) {
73 | if(err.code !== 'MODULE_NOT_FOUND')
74 | c.error(err);
75 | confPath = path.join(process.env.injDir, 'config.json');
76 | bDelCache = false;
77 | }
78 |
79 | try {
80 | fs.writeFileSync(confPath, JSON.stringify(newSets));
81 | if(bDelCache)
82 | delete require.cache[confPath];
83 | } catch(err) {
84 | c.error(err);
85 | }
86 | return this.config;
87 | }
88 | });
89 |
90 | function loadPlugin(plugin) {
91 | try {
92 | if (plugin.preload)
93 | console.log(`%c[EnhancedDiscord] %c[PRELOAD] %cLoading plugin %c${plugin.name}`, 'color: red;', 'color: yellow;', '', `color: ${plugin.color}`, `by ${plugin.author}...`);
94 | else console.log(`%c[EnhancedDiscord] %cLoading plugin %c${plugin.name}`, 'color: red;', '', `color: ${plugin.color}`, `by ${plugin.author}...`);
95 | plugin.load();
96 | } catch(err) {
97 | c.error(`Failed to load:\n${err.stack}`, plugin);
98 | }
99 | }
100 |
101 | ED.localStorage = window.localStorage;
102 |
103 | process.once('loaded', async () => {
104 | c.log(`v${ED.version} is running. Validating plugins...`);
105 |
106 | const pluginFiles = fs.readdirSync(path.join(process.env.injDir, 'plugins'));
107 | const plugins = {};
108 | for (const i in pluginFiles) {
109 | if (!pluginFiles[i].endsWith('.js') || pluginFiles[i].endsWith('.plugin.js')) continue;
110 | let p;
111 | const pName = pluginFiles[i].replace(/\.js$/, '');
112 | try {
113 | p = require(path.join(process.env.injDir, 'plugins', pName));
114 | if (typeof p.name !== 'string' || typeof p.load !== 'function') {
115 | throw new Error('Plugin must have a name and load() function.');
116 | }
117 | plugins[pName] = Object.assign(p, {id: pName});
118 | }
119 | catch (err) {
120 | c.warn(`Failed to load ${pluginFiles[i]}: ${err}\n${err.stack}`, p);
121 | }
122 | }
123 | for (const id in plugins) {
124 | if (!plugins[id] || !plugins[id].name || typeof plugins[id].load !== 'function') {
125 | c.info(`Skipping invalid plugin: ${id}`); delete plugins[id]; continue;
126 | }
127 | plugins[id].settings; // this will set default settings in config if necessary
128 | }
129 | ED.plugins = plugins;
130 | c.log(`Plugins validated.`);
131 |
132 | // work-around to wait for webpack
133 | while (true) {
134 | await c.sleep(100);
135 | if(electron.webFrame.top.context.window && electron.webFrame.top.context.window.webpackJsonp) break;
136 | };
137 |
138 | ED.webSocket = window._ws;
139 |
140 | /* Add helper functions that make plugins easy to create */
141 | window.req = electron.webFrame.top.context.window.webpackJsonp.push([[], {
142 | '__extra_id__': (module, exports, req) => module.exports = req
143 | }, [['__extra_id__']]]);
144 | delete window.req.m['__extra_id__'];
145 | delete window.req.c['__extra_id__'];
146 |
147 | window.findModule = EDApi.findModule;
148 | window.findModules = EDApi.findAllModules;
149 | window.findRawModule = EDApi.findRawModule;
150 | window.monkeyPatch = EDApi.monkeyPatch;
151 |
152 | while (!EDApi.findModule('dispatch'))
153 | await c.sleep(100);
154 |
155 | c.log(`Loading preload plugins...`);
156 | for (const id in plugins) {
157 | if (ED.config[id] && ED.config[id].enabled == false) continue;
158 | if (!plugins[id].preload) continue;
159 | loadPlugin(plugins[id]);
160 | }
161 |
162 | const d = {resolve: () => {}};
163 | window.monkeyPatch(window.findModule('dispatch'), 'dispatch', {before: b => {
164 | // modules seem to all be loaded when RPC server loads
165 | if (b.methodArguments[0].type === 'RPC_SERVER_READY') {
166 | window.findModule('dispatch').dispatch.unpatch();
167 | d.resolve();
168 | }
169 | }});
170 |
171 | await new Promise(resolve => {
172 | d.resolve = resolve;
173 | });
174 | c.log(`Modules done loading (${Object.keys(window.req.c).length})`);
175 |
176 | if (ED.config.bdPlugins) {
177 | try {
178 | await require('./bd_shit').setup();
179 | c.log(`Preparing BD plugins...`);
180 | for (const i in pluginFiles) {
181 | if (!pluginFiles[i].endsWith('.js') || !pluginFiles[i].endsWith('.plugin.js')) continue;
182 | let p;
183 | const pName = pluginFiles[i].replace(/\.js$/, '');
184 | try {
185 | p = require(path.join(process.env.injDir, 'plugins', pName));
186 | if (typeof p.name !== 'string' || typeof p.load !== 'function') {
187 | throw new Error('Plugin must have a name and load() function.');
188 | }
189 | plugins[pName] = Object.assign(p, {id: pName});
190 | }
191 | catch (err) {
192 | c.warn(`Failed to load ${pluginFiles[i]}: ${err}\n${err.stack}`, p);
193 | }
194 | }
195 | for (const id in plugins) {
196 | if (!plugins[id] || !plugins[id].name || typeof plugins[id].load !== 'function') {
197 | c.info(`Skipping invalid plugin: ${id}`); delete plugins[id]; continue;
198 | }
199 | }
200 | }
201 | catch (err) {
202 | c.warn(`Failed to load BD plugins support: ${err}\n${err.stack}`);
203 | }
204 | }
205 |
206 | c.log(`Loading plugins...`);
207 | for (const id in plugins) {
208 | if (ED.config[id] && ED.config[id].enabled == false) continue;
209 | if (plugins[id].preload) continue;
210 | if (ED.config[id] && ED.config[id].enabled !== true && plugins[id].disabledByDefault) {
211 | plugins[id].settings.enabled = false; continue;
212 | }
213 | loadPlugin(plugins[id]);
214 | }
215 |
216 |
217 | const ht = EDApi.findModule('hideToken');
218 | // prevent client from removing token from localstorage when dev tools is opened, or reverting your token if you change it
219 | EDApi.monkeyPatch(ht, 'hideToken', () => {});
220 | window.fixedShowToken = () => {
221 | // Only allow this to add a token, not replace it. This allows for changing of the token in dev tools.
222 | if (!ED.localStorage || ED.localStorage.getItem('token')) return;
223 | return ED.localStorage.setItem('token', '"' + ht.getToken() + '"');
224 | };
225 | EDApi.monkeyPatch(ht, 'showToken', window.fixedShowToken);
226 | if (!ED.localStorage.getItem('token') && ht.getToken())
227 | window.fixedShowToken(); // prevent you from being logged out for no reason
228 |
229 | // change the console warning to be more fun
230 | electron.ipcRenderer.invoke('custom-devtools-warning');
231 |
232 | // expose stuff for devtools
233 | Object.assign(electron.webFrame.top.context.window, {ED, EDApi, BdApi});
234 | });
235 |
236 |
237 |
238 | /* BD/ED joint api */
239 | window.EDApi = window.BdApi = class EDApi {
240 | static get React() { return this.findModuleByProps('createElement'); }
241 | static get ReactDOM() { return this.findModuleByProps('findDOMNode'); }
242 |
243 | static escapeID(id) {
244 | return id.replace(/^[^a-z]+|[^\w-]+/gi, '');
245 | }
246 |
247 | static injectCSS(id, css) {
248 | const style = document.createElement('style');
249 | style.id = this.escapeID(id);
250 | style.innerHTML = css;
251 | document.head.append(style);
252 | }
253 |
254 | static clearCSS(id) {
255 | const element = document.getElementById(this.escapeID(id));
256 | if (element) element.remove();
257 | }
258 |
259 | static linkJS(id, url) {
260 | return new Promise(resolve => {
261 | const script = document.createElement('script');
262 | script.id = this.escapeID(id);
263 | script.src = url;
264 | script.type = 'text/javascript';
265 | script.onload = resolve;
266 | document.head.append(script);
267 | });
268 | }
269 |
270 | static unlinkJS(id) {
271 | const element = document.getElementById(this.escapeID(id));
272 | if (element) element.remove();
273 | }
274 |
275 | static getPlugin(name) {
276 | const plugin = Object.values(ED.plugins).find(p => p.name == name);
277 | if (!plugin) return null;
278 | return plugin.bdplugin ? plugin.bdplugin : plugin;
279 | }
280 |
281 | static alert(title, body) {
282 | return this.showConfirmationModal(title, body, {cancelText: null});
283 | }
284 |
285 | /**
286 | * Shows a generic but very customizable confirmation modal with optional confirm and cancel callbacks.
287 | * @param {string} title - title of the modal
288 | * @param {(string|ReactElement|Array)} children - a single or mixed array of react elements and strings. Every string is wrapped in Discord's `Markdown` component so strings will show and render properly.
289 | * @param {object} [options] - options to modify the modal
290 | * @param {boolean} [options.danger=false] - whether the main button should be red or not
291 | * @param {string} [options.confirmText=Okay] - text for the confirmation/submit button
292 | * @param {string} [options.cancelText=Cancel] - text for the cancel button
293 | * @param {callable} [options.onConfirm=NOOP] - callback to occur when clicking the submit button
294 | * @param {callable} [options.onCancel=NOOP] - callback to occur when clicking the cancel button
295 | * @param {string} [options.key] - key used to identify the modal. If not provided, one is generated and returned
296 | * @returns {string} - the key used for this modal
297 | */
298 | static showConfirmationModal(title, content, options = {}) {
299 | const ModalActions = this.findModuleByProps('openModal', 'updateModal');
300 | const Markdown = this.findModuleByDisplayName('Markdown');
301 | const ConfirmationModal = this.findModuleByDisplayName('ConfirmModal');
302 | if (!ModalActions || !ConfirmationModal || !Markdown) return window.alert(content);
303 |
304 | const emptyFunction = () => {};
305 | const {onConfirm = emptyFunction, onCancel = emptyFunction, confirmText = 'Okay', cancelText = 'Cancel', danger = false, key = undefined} = options;
306 |
307 | if (!Array.isArray(content)) content = [content];
308 | content = content.map(c => typeof(c) === 'string' ? this.React.createElement(Markdown, null, c) : c);
309 | return ModalActions.openModal(props => {
310 | return this.React.createElement(ConfirmationModal, Object.assign({
311 | header: title,
312 | red: danger,
313 | confirmText: confirmText,
314 | cancelText: cancelText,
315 | onConfirm: onConfirm,
316 | onCancel: onCancel
317 | }, props), content);
318 | }, {modalKey: key});
319 | }
320 |
321 | static loadPluginSettings(pluginName) {
322 | const pl = ED.plugins[pluginName];
323 | if (!pl) return null;
324 |
325 | if (!ED.config[pluginName]) {
326 | this.savePluginSettings(pluginName, pl.defaultSettings || {enabled: !pl.disabledByDefault});
327 | }
328 | return ED.config[pluginName];
329 | }
330 |
331 | static savePluginSettings(pluginName, data) {
332 | const pl = ED.plugins[pluginName];
333 | if (!pl) return null;
334 | ED.config[pluginName] = data;
335 | ED.config = ED.config; // eslint-disable-line no-self-assign
336 | }
337 |
338 | static loadData(pluginName, key) {
339 | const pl = ED.plugins[pluginName] || Object.values(ED.plugins).find(p => p.name === pluginName);
340 | if (!pl) return null;
341 | const id = pl.id;
342 |
343 | if (!ED.plugins[id]) return null;
344 | return this.loadPluginSettings(id)[key];
345 | }
346 |
347 | static saveData(pluginName, key, data) {
348 | const pl = ED.plugins[pluginName] || Object.values(ED.plugins).find(p => p.name === pluginName);
349 | if (!pl) return null;
350 | const id = pl.id;
351 |
352 | const obj = this.loadPluginSettings(id);
353 | obj[key] = data;
354 | return this.savePluginSettings(id, obj);
355 | }
356 |
357 | static getData(pluginName, key) {
358 | return this.loadData(pluginName, key);
359 | }
360 |
361 | static setData(pluginName, key, data) {
362 | this.saveData(pluginName, key, data);
363 | }
364 |
365 | static getInternalInstance(node) {
366 | if (!(node instanceof window.jQuery) && !(node instanceof Element)) return undefined;
367 | if (node instanceof window.jQuery) node = node[0];
368 | return node[Object.keys(node).find(k => k.startsWith('__reactInternalInstance'))];
369 | }
370 |
371 | static showToast(content, options = {}) {
372 | if (!document.querySelector('.toasts')) {
373 | const container = document.querySelector('.sidebar-2K8pFh + div') || null;
374 | const memberlist = container ? container.querySelector('.membersWrap-2h-GB4') : null;
375 | const form = container ? container.querySelector('form') : null;
376 | const left = container ? container.getBoundingClientRect().left : 310;
377 | const right = memberlist ? memberlist.getBoundingClientRect().left : 0;
378 | const width = right ? right - container.getBoundingClientRect().left : Math.max(document.documentElement.clientWidth, window.innerWidth || 0) - left - 240;
379 | const bottom = form ? form.offsetHeight : 80;
380 | const toastWrapper = document.createElement('div');
381 | toastWrapper.classList.add('toasts');
382 | toastWrapper.style.setProperty('left', left + 'px');
383 | toastWrapper.style.setProperty('width', width + 'px');
384 | toastWrapper.style.setProperty('bottom', bottom + 'px');
385 | document.querySelector('#app-mount').appendChild(toastWrapper);
386 | }
387 | const {type = '', icon = true, timeout = 3000} = options;
388 | const toastElem = document.createElement('div');
389 | toastElem.classList.add('toast');
390 | if (type) toastElem.classList.add('toast-' + type);
391 | if (type && icon) toastElem.classList.add('icon');
392 | toastElem.innerText = content;
393 | document.querySelector('.toasts').appendChild(toastElem);
394 | setTimeout(() => {
395 | toastElem.classList.add('closing');
396 | setTimeout(() => {
397 | toastElem.remove();
398 | if (!document.querySelectorAll('.toasts .toast').length) document.querySelector('.toasts').remove();
399 | }, 300);
400 | }, timeout);
401 | }
402 |
403 | static findModule(filter, silent = true) {
404 | const moduleName = typeof filter === 'string' ? filter : null;
405 | for (const i in window.req.c) {
406 | if (window.req.c.hasOwnProperty(i)) {
407 | const m = window.req.c[i].exports;
408 | if (m && m.__esModule && m.default && (moduleName ? m.default[moduleName] : filter(m.default))) return m.default;
409 | if (m && (moduleName ? m[moduleName] : filter(m))) return m;
410 | }
411 | }
412 | if (!silent) c.warn(`Could not find module ${module}.`, {name: 'Modules', color: 'black'});
413 | return null;
414 | }
415 |
416 | static findRawModule(filter, silent = true) {
417 | const moduleName = typeof filter === 'string' ? filter : null;
418 | for (const i in window.req.c) {
419 | if (window.req.c.hasOwnProperty(i)) {
420 | const m = window.req.c[i].exports;
421 | if (m && m.__esModule && m.default && (moduleName ? m.default[moduleName] : filter(m.default)))
422 | return window.req.c[i];
423 | if (m && (moduleName ? m[moduleName] : filter(m)))
424 | return window.req.c[i];
425 | }
426 | }
427 | if (!silent) c.warn(`Could not find module ${module}.`, {name: 'Modules', color: 'black'});
428 | return null;
429 | }
430 |
431 | static findAllModules(filter) {
432 | const moduleName = typeof filter === 'string' ? filter : null;
433 | const modules = [];
434 | for (const i in window.req.c) {
435 | if (window.req.c.hasOwnProperty(i)) {
436 | const m = window.req.c[i].exports;
437 | if (m && m.__esModule && m.default && (moduleName ? m.default[moduleName] : filter(m.default))) modules.push(m.default);
438 | else if (m && (moduleName ? m[moduleName] : filter(m))) modules.push(m);
439 | }
440 | }
441 | return modules;
442 | }
443 |
444 | static findModuleByProps(...props) {
445 | return this.findModule(module => props.every(prop => module[prop] !== undefined));
446 | }
447 |
448 | static findModuleByDisplayName(name) {
449 | return this.findModule(module => module.displayName === name);
450 | }
451 |
452 | static monkeyPatch(what, methodName, options) {
453 | if (typeof options === 'function') {
454 | const newOptions = {instead: options, silent: true};
455 | options = newOptions;
456 | }
457 | const {before, after, instead, once = false, silent = false, force = false} = options;
458 | const displayName = options.displayName || what.displayName || what.name || what.constructor ? (what.constructor.displayName || what.constructor.name) : null;
459 | if (!silent) console.log(`%c[EnhancedDiscord] %c[Modules]`, 'color: red;', `color: black;`, `Patched ${methodName} in module ${displayName || ''}:`, what); // eslint-disable-line no-console
460 | if (!what[methodName]) {
461 | if (force) what[methodName] = function() {};
462 | else return console.warn(`%c[EnhancedDiscord] %c[Modules]`, 'color: red;', `color: black;`, `Method ${methodName} doesn't exist in module ${displayName || ''}`, what); // eslint-disable-line no-console
463 | }
464 | const origMethod = what[methodName];
465 | const cancel = () => {
466 | if (!silent) console.log(`%c[EnhancedDiscord] %c[Modules]`, 'color: red;', `color: black;`, `Unpatched ${methodName} in module ${displayName || ''}:`, what); // eslint-disable-line no-console
467 | what[methodName] = origMethod;
468 | };
469 | what[methodName] = function() {
470 | const data = {
471 | thisObject: this,
472 | methodArguments: arguments,
473 | cancelPatch: cancel,
474 | originalMethod: origMethod,
475 | callOriginalMethod: () => data.returnValue = data.originalMethod.apply(data.thisObject, data.methodArguments)
476 | };
477 | if (instead) {
478 | const tempRet = EDApi.suppressErrors(instead, '`instead` callback of ' + what[methodName].displayName)(data);
479 | if (tempRet !== undefined) data.returnValue = tempRet;
480 | }
481 | else {
482 | if (before) EDApi.suppressErrors(before, '`before` callback of ' + what[methodName].displayName)(data);
483 | data.callOriginalMethod();
484 | if (after) EDApi.suppressErrors(after, '`after` callback of ' + what[methodName].displayName)(data);
485 | }
486 | if (once) cancel();
487 | return data.returnValue;
488 | };
489 | what[methodName].__monkeyPatched = true;
490 | what[methodName].displayName = 'patched ' + (what[methodName].displayName || methodName);
491 | what[methodName].unpatch = cancel;
492 | return cancel;
493 | }
494 |
495 | static testJSON(data) {
496 | try {
497 | return JSON.parse(data);
498 | }
499 | catch (err) {
500 | return false;
501 | }
502 | }
503 |
504 | static suppressErrors(method, description) {
505 | return (...params) => {
506 | try { return method(...params); }
507 | catch (e) { console.error('Error occurred in ' + description, e); }
508 | };
509 | }
510 |
511 | static formatString(string, values) {
512 | for (const val in values) {
513 | string = string.replace(new RegExp(`\\{\\{${val}\\}\\}`, 'g'), values[val]);
514 | }
515 | return string;
516 | }
517 |
518 | static isPluginEnabled(name) {
519 | const plugins = Object.values(ED.plugins);
520 | const plugin = plugins.find(p => p.id == name || p.name == name);
521 | if (!plugin) return false;
522 | return !(plugin.settings.enabled === false);
523 | }
524 |
525 | static isThemeEnabled() {
526 | return false;
527 | }
528 |
529 | static isSettingEnabled(id) {
530 | return ED.config[id];
531 | }
532 | };
533 |
534 | window.BdApi.Plugins = new class AddonAPI {
535 |
536 | get folder() {return path.join(process.env.injDir, 'plugins');}
537 |
538 | isEnabled(name) {
539 | const plugins = Object.values(ED.plugins);
540 | const plugin = plugins.find(p => p.id == name || p.name == name);
541 | if (!plugin) return false;
542 | return !(plugin.settings.enabled === false);
543 | }
544 |
545 | enable(name) {
546 | const plugin = ED.plugins[name];
547 | if (!plugin || plugin.settings.enabled !== false) return;
548 | plugin.settings.enabled = true;
549 | ED.plugins[name].settings = plugin.settings;
550 | plugin.load();
551 | }
552 |
553 | disable(name) {
554 | const plugin = ED.plugins[name];
555 | if (!plugin || plugin.settings.enabled === false) return;
556 | plugin.settings.enabled = false;
557 | ED.plugins[name].settings = plugin.settings;
558 | plugin.unload();
559 | }
560 |
561 | toggle(name) {
562 | if (this.isEnabled(name)) this.disable(name);
563 | else this.enable(name);
564 | }
565 |
566 | reload(name) {
567 | const plugin = ED.plugins[name];
568 | if (!plugin) return;
569 | plugin.reload();
570 | }
571 |
572 | get(name) {
573 | return ED.plugins[name] || null;
574 | }
575 |
576 | getAll() {
577 | return Object.values(ED.plugins);
578 | }
579 | };
580 |
581 | window.BdApi.Themes = new class AddonAPI {
582 | get folder() {return '';}
583 | isEnabled() {}
584 | enable() {}
585 | disable() {}
586 | toggle() {}
587 | reload() {}
588 | get() {return null;}
589 | getAll() {return [];}
590 | };
--------------------------------------------------------------------------------
/plugins/ed_settings.js:
--------------------------------------------------------------------------------
1 | const Plugin = require('../plugin');
2 | const BD = require('../bd_shit');
3 |
4 | const edSettingsID = require("path").parse(__filename).name;
5 |
6 | module.exports = new Plugin({
7 | name: 'ED Settings (React)',
8 | author: 'jakuski#9191',
9 | description: 'Adds an EnhancedDiscord tab in user settings.',
10 | color: 'darkred',
11 | async load () {
12 | const discordConstants = EDApi.findModule("API_HOST");
13 | const UserSettings = module.exports.utils.getComponentFromFluxContainer(
14 | EDApi.findModule('getUserSettingsSections').default
15 | );
16 |
17 | if (!ED.classMaps) {
18 | ED.classMaps = {};
19 | }
20 |
21 | if (!ED.discordComponents) {
22 | ED.discordComponents = {};
23 | }
24 |
25 | this._initClassMaps(ED.classMaps);
26 | this._initDiscordComponents(ED.discordComponents);
27 | this.components = this._initReactComponents();
28 | this.settingsSections = this._getDefaultSections(); // Easily allow plugins to add in their own sections if need be.
29 |
30 | this.unpatch = EDApi.monkeyPatch(
31 | UserSettings.prototype,
32 | "generateSections",
33 | data => {
34 | const sections = data.originalMethod.call(data.thisObject);
35 | // We use the devIndex as a base so that should Discord add more sections later on, our sections shouldn't move possibly fucking up the UI.
36 | const devIndex = this._getDevSectionIndex(sections, discordConstants);
37 | let workingIndex = devIndex + 2;
38 |
39 | sections.splice(workingIndex, 0, ...this.settingsSections);
40 | workingIndex += this.settingsSections.length;
41 |
42 | const customSections = this._getPluginSections();
43 | if (customSections.length) {
44 | sections.splice(workingIndex, 0, ...customSections);
45 | workingIndex += customSections.length;
46 | }
47 |
48 | sections.splice(workingIndex, 0, { section: "DIVIDER" });
49 |
50 | return sections;
51 | }
52 | )
53 | },
54 | unload () {
55 | if (this.unpatch && typeof this.unpatch === "function") this.unpatch();
56 | },
57 | utils: {
58 | join (...args) {
59 | return args.join(" ")
60 | },
61 | getComponentFromFluxContainer (component) {
62 | return (new component({})).render().type;
63 | },
64 | shouldPluginRender (id) {
65 | let shouldRender = true;
66 |
67 | // BetterDiscord plugins settings render in their own modal activated in their listing.
68 | if (ED.plugins[id].getSettingsPanel && typeof ED.plugins[id].getSettingsPanel == 'function') {
69 | shouldRender = false;
70 | }
71 |
72 | if (ED.plugins[id].settings.enabled === false || !ED.plugins[id].generateSettings) {
73 | shouldRender = false;
74 | }
75 |
76 | return shouldRender;
77 | }
78 | },
79 | _getDevSectionIndex(sections, constants) {
80 | const indexOf = sections.indexOf(
81 | sections.find(sect => sect.section === constants.UserSettingsSections.DEVELOPER_OPTIONS)
82 | );
83 |
84 | if (indexOf !== -1) return indexOf;
85 | else return 28; // Hardcoded index fallback incase Discord mess with something
86 | },
87 | _getDefaultSections() {
88 | /*
89 |
90 | For future reference:
91 |
92 | normal sections / pages
93 | section: [string] an id string of some sort, must be unique.
94 | label: [string] self-explanatory
95 | element: [optional] [react-renderable] the page that will be rendered on the right when the section is clicked
96 | color: [optional] [string (hex)] a colour to be applied to the section (see the log out / discord nitro btn)
97 | onClick: [optional] [function] a function to be executed whenever the element is clicked
98 |
99 | special sections
100 | headers
101 | section: "HEADER"
102 | label: [string]
103 | divider
104 | section: "DIVIDER"
105 | custom element
106 | section: "CUSTOM"
107 | element: [react-renderable]
108 |
109 | all sections regardless of type can have the following
110 | predicate: [function => boolean] determine whether the section should be shown
111 |
112 | */
113 | return [{
114 | section: "CUSTOM",
115 | element: () => {
116 | const { join } = module.exports.utils;
117 | const { header } = EDApi.findModule("topPill");
118 |
119 | return EDApi.React.createElement("div", { className: join(header, "ed-settings") }, "EnhancedDiscord")
120 | }
121 | },{
122 | section: "ED/Plugins",
123 | label: "Plugins",
124 | element: this.components.PluginsPage
125 | },{
126 | section: "ED/Settings",
127 | label: "Settings",
128 | element: this.components.SettingsPage
129 | }];
130 | },
131 | _getPluginSections() {
132 | const arr = [];
133 | for (const key in ED.plugins) {
134 | if (typeof ED.plugins[key].generateSettingsSection !== 'function') continue;
135 |
136 | const label = ED.plugins[key].settingsSectionName || ED.plugins[key].name;
137 | arr.push({
138 | section: 'ED/'+key,
139 | label,
140 | element: () => EDApi.React.createElement(this.components.PluginSection, {id: key, label})
141 | });
142 | }
143 | return arr;
144 | },
145 | _initClassMaps(obj) {
146 | const divM = EDApi.findModule(m => m.divider && Object.keys(m).length === 1)
147 | obj.headers = EDApi.findModule('defaultMarginh2');
148 | obj.margins = EDApi.findModule('marginBottom8');
149 | obj.divider = divM ? divM.divider : '';
150 | obj.checkbox = EDApi.findModule('checkboxEnabled');
151 | obj.buttons = EDApi.findModule('lookFilled');
152 | obj.switch = EDApi.findModule('switch');
153 | obj.alignment = EDApi.findModule('horizontalReverse');
154 | obj.description = EDApi.findModule('formText');
155 | // New
156 | obj.shadows = EDApi.findModule("elevationHigh");
157 | },
158 | _initDiscordComponents(obj) {
159 | obj.Textbox = EDApi.findModuleByDisplayName("TextInput");
160 | obj.Select = EDApi.findModuleByDisplayName("SelectTempWrapper");
161 | obj.RadioGroup = EDApi.findModuleByDisplayName("RadioGroup");
162 | obj.Title = EDApi.findModuleByDisplayName("FormTitle");
163 | obj.Text = EDApi.findModuleByDisplayName("FormText");
164 | obj.FormSection = EDApi.findModuleByDisplayName("FormSection");
165 | obj.Icon = EDApi.findModuleByDisplayName("Icon");
166 | obj.LoadingSpinner = EDApi.findModuleByDisplayName("Spinner");
167 | obj.Card = EDApi.findModuleByDisplayName("FormNotice");
168 | obj.Flex = EDApi.findModuleByDisplayName("Flex");
169 | obj.Switch = EDApi.findModuleByDisplayName("Switch");
170 | obj.SwitchItem = EDApi.findModuleByDisplayName("SwitchItem");
171 | obj.Slider = EDApi.findModuleByDisplayName("Slider");
172 | obj.Select = EDApi.findModuleByDisplayName("SelectTempWrapper");
173 | obj.Tooltip = EDApi.findModuleByDisplayName("Tooltip");
174 | obj.Button = EDApi.findModule("Sizes");
175 | //obj.ColorPicker = EDApi.findModuleByDisplayName("ColorPicker");
176 |
177 | /*
178 | Props: any valid props you can apply to a div element
179 | */
180 | obj.Divider = props => {
181 | props.className = props.className ? props.className + " " + ED.classMaps.divider : ED.classMaps.divider
182 | return EDApi.React.createElement("div", Object.assign({}, props))
183 | }
184 | },
185 | _initReactComponents () {
186 | const { createElement:e, Component, Fragment, useState, useEffect, useReducer, createRef, isValidElement } = EDApi.React;
187 | const { FormSection, Divider, Flex, Switch, Title, Text, Button, SwitchItem, Textbox, RadioGroup, Select, Slider /*, ColorPicker*/ } = ED.discordComponents;
188 | const { margins } = ED.classMaps;
189 | const { join } = module.exports.utils;
190 |
191 | const PluginsPage = () => {
192 | return e(FormSection, {title: "EnhancedDiscord Plugins", tag: "h2"},
193 | e(Flex, {},
194 | e(OpenPluginDirBtn)
195 | ),
196 | e(Divider, {className: join(margins.marginTop20, margins.marginBottom20)}),
197 | Object
198 | .keys(ED.plugins)
199 | .map(id => e(PluginListing, {id, plugin: ED.plugins[id]}))
200 | )
201 | }
202 |
203 | const SettingsPage = () => {
204 | return e(Fragment, null,
205 | e(FormSection, {title: "EnhancedDiscord Settings", tag: "h2"},
206 | e(BDPluginToggle),
207 | Object
208 | .keys(ED.plugins)
209 | .filter(module.exports.utils.shouldPluginRender)
210 | .map((id, index) => e(PluginSettings, {id, index})),
211 | )
212 | )
213 | }
214 |
215 | class PluginListing extends Component {
216 | constructor(props) {
217 | super(props);
218 |
219 | this.state = {
220 | reloadBtnText: "Reload"
221 | }
222 |
223 | this.canBeManagedByUser = this.canBeManagedByUser.bind(this);
224 | this.isPluginEnabled = this.isPluginEnabled.bind(this);
225 | this.handleReload = this.handleReload.bind(this);
226 | this.handleToggle = this.handleToggle.bind(this);
227 | this.handleLoad = this.handleLoad.bind(this);
228 | this.handleUnload = this.handleUnload.bind(this);
229 | }
230 | isPluginEnabled() {
231 | return this.props.plugin.settings.enabled !== false;
232 | }
233 | canBeManagedByUser () {
234 | return !(this.props.id === edSettingsID)
235 | }
236 | handleReload () {
237 | this.setState({reloadBtnText: "Reloading..."});
238 | try {
239 | this.props.plugin.reload();
240 | this.setState({reloadBtnText: "Reloaded!"});
241 | } catch (err) {
242 | module.exports.error(err);
243 | this.setState({reloadBtnText: `Failed to reload (${err.name} - see console.)`})
244 | }
245 |
246 | setTimeout(
247 | setState => setState({reloadBtnText: "Reload"}),
248 | 3000,
249 | this.setState.bind(this)
250 | )
251 | }
252 | handleToggle(newStatus) {
253 | if (newStatus) this.handleLoad();
254 | else this.handleUnload();
255 |
256 | this.forceUpdate();
257 | }
258 | handleLoad () {
259 | if (this.isPluginEnabled()) return;
260 |
261 | this.props.plugin.settings.enabled = true;
262 | ED.plugins[this.props.id].settings = this.props.plugin.settings;
263 | this.props.plugin.load();
264 | }
265 | handleUnload () {
266 | if (!this.isPluginEnabled()) return;
267 |
268 | this.props.plugin.settings.enabled = false;
269 | ED.plugins[this.props.id].settings = this.props.plugin.settings;
270 | this.props.plugin.unload();
271 | }
272 | render () {
273 | const { plugin } = this.props;
274 | const { MarginRight, ColorBlob } = this.constructor;
275 |
276 | return e(Fragment, null,
277 | e(Flex, { direction: Flex.Direction.VERTICAL},
278 | e(Flex, {align: Flex.Align.CENTER},
279 | e(Title, {tag: "h3", className: ""}, plugin.name),
280 | e(ColorBlob, {color: plugin.color || "orange"}),
281 | e(MarginRight, null,
282 | e(Button, {size: Button.Sizes.NONE, onClick: this.handleReload}, this.state.reloadBtnText)
283 | ),
284 | this.canBeManagedByUser() && e(Switch, {checked: this.isPluginEnabled(), onChange: this.handleToggle})
285 | ),
286 | e(Text, {type: Text.Types.DESCRIPTION},
287 | VariableTypeRenderer.render(plugin.description)
288 | )
289 | ),
290 | e(Divider, {className: join(margins.marginTop20, margins.marginBottom20)})
291 | )
292 | }
293 | }
294 |
295 | PluginListing.MarginRight = props => e("div", {style:{marginRight: "10px"}}, props.children);
296 | PluginListing.ColorBlob = props => e("div", {style: {
297 | backgroundColor: props.color,
298 | boxShadow: `0 0 5px 2px ${props.color}`,
299 | borderRadius: "50%",
300 | height: "10px",
301 | width: "10px",
302 | marginRight: "8px"
303 | }});
304 |
305 | const PluginSettings = props => {
306 | const plugin = ED.plugins[props.id];
307 |
308 | return e(Fragment, null,
309 | e(Divider, { style:{ marginTop: props.index === 0 ? "0px" : undefined}, className: join(margins.marginTop8, margins.marginBottom20)}),
310 | e(Title, {tag: "h2"}, plugin.name),
311 | VariableTypeRenderer.render(
312 | plugin.generateSettings(),
313 | plugin.settingsListeners,
314 | props.id
315 | )
316 | )
317 | }
318 |
319 | const PluginSection = props => {
320 | const plugin = ED.plugins[props.id];
321 |
322 | return e(Fragment, null,
323 | e(Title, {tag: "h2"}, props.label),
324 | VariableTypeRenderer.render(
325 | plugin.generateSettingsSection(),
326 | plugin.settingsListeners,
327 | props.id
328 | )
329 | )
330 | }
331 |
332 | const BDPluginToggle = () => {
333 | const [ enabled, setEnabled ] = useState(ED.config.bdPlugins);
334 |
335 | useEffect(() => {
336 | if (enabled === ED.config.bdPlugins) return; // Prevent unneccesary file write
337 |
338 | ED.config.bdPlugins = enabled;
339 | ED.config = ED.config;
340 | });
341 |
342 | return e(SwitchItem, {
343 | onChange: () => setEnabled(!enabled),
344 | value: enabled,
345 | hideBorder: true,
346 | note: "Allows EnhancedDiscord to load BetterDiscord plugins natively. Reload (ctrl+r) for changes to take effect."
347 | },
348 | "BetterDiscord Plugins"
349 | );
350 | }
351 |
352 | const OpenPluginDirBtn = () => {
353 | const [ string, setString ] = useState("Open Plugins Directory");
354 |
355 | return e(Button, {size: Button.Sizes.SMALL, color: Button.Colors.GREEN, style: {margin: '0 5px', display: 'inline-block'}, onClick: e => {
356 | setString("Opening...");
357 | const sucess = require("electron").shell.openPath(
358 | e.shiftKey ?
359 | process.env.injDir :
360 | require("path").join(process.env.injDir, "plugins")
361 | );
362 |
363 | if (sucess) setString("Opened!");
364 | else setString("Failed to open...");
365 |
366 | setTimeout(() => {
367 | setString("Open Plugins Directory")
368 | }, 1500)
369 | }}, string)
370 | }
371 |
372 | class VariableTypeRenderer {
373 | static render (value, listeners, plugin) {
374 | const { DOMStringRenderer, HTMLElementInstanceRenderer } = this;
375 |
376 | let typeOf = null;
377 |
378 | if (typeof value === "string") typeOf = "domstring";
379 | if (isValidElement(value)) typeOf = "react";
380 | if (value instanceof HTMLElement) typeOf = "htmlelement-instance";
381 | if (Array.isArray(value)) typeOf = "auto";
382 | if (value == null) typeOf = "blank";
383 |
384 | if (typeOf === null) return module.exports.error("Unable to figure out how to render value ", value);
385 |
386 | switch(typeOf) {
387 | case "domstring": return e(DOMStringRenderer, {html: value, listeners});
388 | case "react": return value;
389 | case "htmlelement-instance": return e(HTMLElementInstanceRenderer, {instance: value});
390 | case "auto": return DiscordUIGenerator.render(value, plugin);
391 | case "blank": return e(Fragment);
392 | }
393 | }
394 | }
395 | VariableTypeRenderer.DOMStringRenderer = class DOMString extends Component {
396 | componentDidMount() {
397 | if (!this.props.listeners) return;
398 |
399 | this.props.listeners.forEach(listener => {
400 | document.querySelector(listener.el).addEventListener(listener.type, listener.eHandler)
401 | });
402 | }
403 | render () {
404 | return e("div", {dangerouslySetInnerHTML:{__html: this.props.html}});
405 | }
406 | }
407 | VariableTypeRenderer.HTMLElementInstanceRenderer = class HTMLElementInstance extends Component {
408 | constructor() {
409 | super();
410 | this.ref = createRef();
411 | }
412 | componentDidMount() {
413 | this.ref.current.appendChild(this.props.instance)
414 | }
415 | render () {
416 | return e("div", {ref: this.ref});
417 | }
418 | }
419 |
420 | const DiscordUIGenerator = {
421 | reactMarkdownRules: (() => {
422 | const simpleMarkdown = EDApi.findModule("markdownToReact");
423 | const rules = require("electron").webFrame.top.context.window._.clone(simpleMarkdown.defaultRules);
424 |
425 | rules.paragraph.react = (node, output, state) => {
426 | return e(Fragment, null, output(node.content, state))
427 | }
428 |
429 | rules.em.react = (node, output, state) => {
430 | return e("i", null, output(node.content, state))
431 | }
432 |
433 | rules.link.react = (node, output, state) => {
434 | return e("a", {
435 | href: node.target,
436 | target: "_blank",
437 | rel: "noreferrer noopener"
438 | }, output(node.content, state))
439 | }
440 |
441 | return rules;
442 | })(),
443 | render (ui, pluginID) {
444 | const { _types } = DiscordUIGenerator;
445 |
446 | return e("div", {},
447 | ui.map(element => {
448 | if (isValidElement(element)) return element;
449 |
450 | const component = _types[element.type];
451 | if (!component) return module.exports.error("[DiscordUIGenerator] Invalid element type:", element.type);
452 |
453 | return e(component, Object.assign({}, element, {pluginID}))
454 | })
455 | )
456 | },
457 | _parseMD (content) {
458 | const { reactMarkdownRules } = DiscordUIGenerator;
459 | const { markdownToReact } = EDApi.findModule("markdownToReact");
460 |
461 | return markdownToReact(content, reactMarkdownRules);
462 | },
463 | _loadData (props) {
464 | const plug = ED.plugins[props.pluginID];
465 | if (!plug) return;
466 | if (plug.customLoad) { // custom function for loading settings
467 | return plug.customLoad(props.configName)
468 | }
469 | return EDApi.loadData(props.pluginID, props.configName)
470 | },
471 | _saveData (props, data) {
472 | const plug = ED.plugins[props.pluginID];
473 | if (!plug) return;
474 | if (plug.customSave) { // custom function for saving settings
475 | return plug.customSave(props.configName, data)
476 | }
477 | const r = EDApi.saveData(props.pluginID, props.configName, data)
478 | if (plug.onSettingsUpdate) { // custom function to run after settings have updated
479 | plug.onSettingsUpdate(props.configName, data)
480 | }
481 | return r;
482 | },
483 | _cfgNameCheck(props, name) {
484 | if (!props.configName || typeof props.configName !== "string") {
485 | module.exports.error(`[DiscordUIGenerator] Input component (${name}) was not passed a configName value!`);
486 | throw new Error("Stopping react render. Please fix above error");
487 | }
488 | },
489 | _inputWrapper (props) {
490 | const { _parseMD } = DiscordUIGenerator;
491 |
492 | return e(Fragment, null,
493 | props.title && e(Title, { tag: "h5"}, props.title),
494 | props.children,
495 | props.desc && e(Text, {type: Text.Types.DESCRIPTION, className: join(margins.marginTop8, margins.marginBottom20)}, _parseMD(props.desc))
496 | )
497 | },
498 | _types: {
499 | "std:title": props => {
500 | return e(Title, { id: props.id, tag: props.tag || "h5" },
501 | props.content
502 | )
503 | },
504 | "std:description": props => {
505 | const {_parseMD} = DiscordUIGenerator;
506 |
507 | return e(Text, { id: props.id, type: props.descriptionType || "description" }, _parseMD(props.content))
508 | },
509 | "std:divider": props => {
510 | return e(Divider, {id: props.id, style: {marginTop: props.top, marginBottom: props.bottom}})
511 | },
512 | "std:spacer": props => {
513 | return e("div", {id: props.id, style: {marginBottom: props.space}})
514 | },
515 | "input:text": props => {
516 | const {_loadData: load, _saveData: save, _cfgNameCheck, _inputWrapper} = DiscordUIGenerator;
517 |
518 | _cfgNameCheck(props, "input:text");
519 |
520 | const [value, setValue] = useState(load(props));
521 |
522 | return e(_inputWrapper, {title: props.title, desc: props.desc},
523 | e(Textbox, {
524 | id: props.id,
525 | onChange: val => setValue(val),
526 | onBlur: e => save(props, e.currentTarget.value),
527 | value,
528 | placeholder: props.placeholder,
529 | size: props.mini ? "mini" : "default",
530 | disabled: props.disabled,
531 | type: props.number ? "number" : "text"
532 | })
533 | )
534 | },
535 | "input:button": props => {
536 | const [ string, setString ] = useState(props.name);
537 |
538 | return e(Button, {
539 | id: props.id,
540 | size: Button.Sizes[(props.size || '').toUpperCase() || 'SMALL'],
541 | color: Button.Colors[(props.color || '').toUpperCase() || 'BRAND'],
542 | look: Button.Looks[(props.look || '').toUpperCase() || 'FILLED'],
543 | style: {margin: '0 5px', display: 'inline-block'},
544 | disabled: props.disabled,
545 | onClick: () => props.onClick(setString)
546 | }, string)
547 | },
548 | "input:colorpicker": props => {
549 | // TODO: proper transparency support? would need to use different/modified component
550 | const {_loadData: load, _saveData: save, _cfgNameCheck, _inputWrapper} = DiscordUIGenerator;
551 |
552 | _cfgNameCheck(props, "input:text");
553 |
554 | const [value, setValue] = useState(load(props));
555 |
556 | return e("div",
557 | e(Title, { tag: "h5" }, props.title),
558 | /*e(ColorPicker, {
559 | onChange: value => {
560 | const hexValue = '#'+value.toString(16).padStart(6, '0');
561 | const inp = document.getElementById('ed_'+props.configName);
562 | if (inp) inp.value = hexValue;
563 | save(props, hexValue);
564 | },
565 | colors: props.colors || [],
566 | defaultColor: props.defaultColor,
567 | customColor: props.currentColor,
568 | value: props.currentColor
569 | }),*/
570 | e("div", {style: {marginBottom: 10}}),
571 | e(_inputWrapper, {desc: props.desc},
572 | e(Textbox, {
573 | id: 'ed_'+props.configName,
574 | onChange: val => setValue(val),
575 | onBlur: e => save(props, e.currentTarget.value),
576 | value,
577 | placeholder: props.placeholder,
578 | size: "mini",
579 | disabled: props.disabled
580 | })
581 | )
582 | )
583 | },
584 | "input:boolean": props => {
585 | const {_loadData: load, _saveData: save, _cfgNameCheck, _parseMD} = DiscordUIGenerator;
586 |
587 | _cfgNameCheck(props, "input:boolean");
588 |
589 | const [ enabled, toggle ] = useReducer((state) => {
590 | const newState = !state;
591 |
592 | save(props, newState);
593 |
594 | return newState;
595 | }, load(props))
596 |
597 | return e(SwitchItem, {
598 | id: props.id,
599 | onChange: toggle,
600 | value: enabled,
601 | hideBorder: props.hideBorder,
602 | note: _parseMD(props.note),
603 | disabled: props.disabled
604 | },
605 | props.title
606 | )
607 | },
608 | "input:radio": props => {
609 | const {_loadData: load, _saveData: save, _cfgNameCheck, _inputWrapper} = DiscordUIGenerator;
610 |
611 | _cfgNameCheck(props, "input:radio");
612 |
613 | const [ currentSetting, setSetting ] = useReducer((state, data) => {
614 | const newState = data.value;
615 |
616 | save(props, newState);
617 |
618 | return newState;
619 | }, load(props))
620 |
621 | return e(_inputWrapper, {title: props.title, desc: props.desc},
622 | e(RadioGroup, {
623 | id: props.id,
624 | onChange: setSetting,
625 | value: currentSetting,
626 | size: props.size ? props.size : "10px",
627 | disabled: props.disabled,
628 | options: props.options
629 | })
630 | )
631 | },
632 | "input:select": props => {
633 | const {_loadData: load, _saveData: save, _cfgNameCheck, _inputWrapper} = DiscordUIGenerator;
634 |
635 | _cfgNameCheck(props, "input:select");
636 |
637 | const [ currentSetting, setSetting ] = useReducer((state, data) => {
638 | const newState = data.value;
639 |
640 | save(props, newState);
641 |
642 | return newState;
643 | }, load(props))
644 |
645 | return e(_inputWrapper, {title: props.title, desc: props.desc},
646 | e(Select, {
647 | id: props.id,
648 | onChange: setSetting,
649 | value: currentSetting,
650 | disabled: props.disabled,
651 | options: props.options,
652 | searchable: props.searchable || false
653 | })
654 | )
655 | },
656 | "input:slider": props => {
657 | const {_loadData: load, _saveData: save, _cfgNameCheck, _inputWrapper} = DiscordUIGenerator;
658 |
659 | _cfgNameCheck(props, "input:slider");
660 |
661 | const [ currentSetting, setSetting ] = useReducer((state, data) => {
662 | const newState = data;
663 |
664 | save(props, newState);
665 |
666 | return newState;
667 | }, load(props) || props.defaultValue);
668 |
669 | const defaultOnValueRender = e => {
670 | return e.toFixed(0) + "%"
671 | }
672 |
673 | return e(_inputWrapper, {title: props.title, desc: props.desc},
674 | e(Slider, {
675 | id: props.id,
676 | onValueChange: setSetting,
677 | onValueRender: props.formatTooltip ? props.formatTooltip : defaultOnValueRender,
678 | initialValue: currentSetting,
679 | defaultValue: props.highlightDefaultValue ? props.defaultValue : null,
680 | minValue: props.minValue,
681 | maxValue: props.maxValue,
682 | disabled: props.disabled,
683 | markers: props.markers,
684 | stickToMarkers: props.stickToMarkers
685 | })
686 | )
687 | }
688 | }
689 | }
690 |
691 | return {
692 | PluginsPage,
693 | SettingsPage,
694 | PluginListing,
695 | PluginSettings,
696 | PluginSection,
697 | BDPluginToggle,
698 | OpenPluginDirBtn,
699 | __VariableTypeRenderer: VariableTypeRenderer,
700 | __DiscordUIGenerator: DiscordUIGenerator
701 | }
702 | }
703 | });
704 |
--------------------------------------------------------------------------------
/installer/EnhancedDiscordUI/Form1.cs:
--------------------------------------------------------------------------------
1 | using System;
2 | using System.Collections.Generic;
3 | using System.ComponentModel;
4 | using System.Data;
5 | using System.Drawing;
6 | using System.Linq;
7 | using System.Text;
8 | using System.Threading.Tasks;
9 | using System.Windows.Forms;
10 | using System.Diagnostics;
11 | using System.IO;
12 | using System.Net;
13 | using System.IO.Compression;
14 | using System.Runtime.InteropServices;
15 |
16 | namespace EnhancedDiscordUI
17 | {
18 | public partial class EDInstaller : Form
19 | {
20 | private Process stableProcess;
21 | private Process ptbProcess;
22 | private Process canaryProcess;
23 | private Process devProcess;
24 | private string operation = "INSTALL";
25 | private string platform;
26 | private string branch = "master";
27 |
28 | public EDInstaller()
29 | {
30 | Logger.MakeDivider();
31 | Logger.Log("Starting...");
32 | InitializeComponent();
33 | if (Directory.Exists("./EnhancedDiscord"))
34 | {
35 | UninstallButton.Enabled = true;
36 | UpdateButton.Enabled = true;
37 | ReinjectButton.Enabled = true;
38 | }
39 | }
40 | private void endInstallation(string reason, bool failed)
41 | {
42 | InstallProgress.Value = 100;
43 | BetaRadio.Hide();
44 | InstallButton.Hide();
45 | UninstallButton.Hide();
46 | UpdateButton.Hide();
47 | ReinjectButton.Hide();
48 | StatusText.Hide();
49 | StatusLabel.Show();
50 | StatusLabel.Text = operation == "UPDATE" ? "Update " + (failed ? "failed" : "complete") : (operation == "UNINSTALL" ? "Unin" : "In") + "stallation " + (failed ? " failed." : "completed!");
51 | StatusLabel.ForeColor = failed ? Color.Red : Color.Lime;
52 | StatusLabel2.Show();
53 | StatusLabel2.Text = reason;
54 | StatusCloseButton.Show();
55 | if (platform != "Linux")
56 | {
57 | OpenFolderButton.Show();
58 | }
59 | }
60 | private void InstallButton_Click(object sender, EventArgs e)
61 | {
62 | if (BetaRadio.Checked)
63 | {
64 | branch = "beta";
65 | }
66 | BetaRadio.Hide();
67 | InstallButton.Hide();
68 | UninstallButton.Hide();
69 | UpdateButton.Hide();
70 | ReinjectButton.Hide();
71 | StatusText.Show();
72 | InstallProgress.Show();
73 | StatusText.Text = "Finding Discord processes...";
74 |
75 | Process[] stable = Process.GetProcessesByName("Discord");
76 | Process[] canary = Process.GetProcessesByName("DiscordCanary");
77 | Process[] ptb = Process.GetProcessesByName("DiscordPtb");
78 | Process[] dev = Process.GetProcessesByName("DiscordDevelopment");
79 |
80 | List discordProcesses = new List();
81 | discordProcesses.AddRange(stable);
82 | discordProcesses.AddRange(canary);
83 | discordProcesses.AddRange(ptb);
84 | discordProcesses.AddRange(dev);
85 |
86 | if (discordProcesses.Count == 0)
87 | {
88 | endInstallation("No Discord processes found. Please open Discord and try again.", true); return;
89 | }
90 | List uniqueProcesses = new List();
91 | // First look for processes with unique filenames that have a title
92 | for (int i = 0; i < discordProcesses.Count; i++)
93 | {
94 | bool isUnique = true;
95 | for (int j = 0; j < uniqueProcesses.Count; j++)
96 | {
97 | if (uniqueProcesses[j].MainModule.FileName.Equals(discordProcesses[i].MainModule.FileName))
98 | {
99 | isUnique = false; break;
100 | }
101 | }
102 | if (!isUnique || discordProcesses[i].MainWindowTitle == "" || discordProcesses[i].MainWindowTitle.StartsWith("Developer Tools")) continue;
103 |
104 | uniqueProcesses.Add(discordProcesses[i]);
105 | }
106 | // Then look for all processes with unique filenames
107 | for (int i = 0; i < discordProcesses.Count; i++)
108 | {
109 | bool isUnique = true;
110 | for (int j = 0; j < uniqueProcesses.Count; j++)
111 | {
112 | if (uniqueProcesses[j].MainModule.FileName.Equals(discordProcesses[i].MainModule.FileName))
113 | {
114 | isUnique = false; break;
115 | }
116 | }
117 | if (!isUnique) continue;
118 | uniqueProcesses.Add(discordProcesses[i]);
119 | }
120 | StatusText.Text = "Found " + uniqueProcesses.Count + " Discord process" + (uniqueProcesses.Count == 1 ? "" : "es") + ".";
121 | InstallProgress.Value = 10;
122 | Process finalProcess = uniqueProcesses[0];
123 | if (uniqueProcesses.Count > 1)
124 | {
125 | // Enable selection buttons
126 | List