diff --git a/client/src/index.js b/client/src/index.js
index 060eb804..3939e608 100644
--- a/client/src/index.js
+++ b/client/src/index.js
@@ -8,7 +8,7 @@
* LICENSE file in the root directory of this source tree.
*/
-import { DOM, BdUI } from 'ui';
+import { DOM, BdUI, ProfileBadges } from 'ui';
import BdCss from './styles/index.scss';
import { Events, CssEditor, Globals, PluginManager, ThemeManager } from 'modules';
import { ClientLogger as Logger } from 'common';
@@ -16,6 +16,7 @@ import { ClientLogger as Logger } from 'common';
class BetterDiscord {
constructor() {
+ ProfileBadges.init(); // Not final way to do it
DOM.injectStyle(BdCss, 'bdmain');
Events.on('global-ready', this.globalReady.bind(this));
}
@@ -27,6 +28,7 @@ class BetterDiscord {
}
globalReady() {
+ BdUI.initUiEvents();
this.vueInstance = BdUI.injectUi();
(async () => {
this.init();
diff --git a/client/src/modules/modules.js b/client/src/modules/modules.js
index 1e5537cb..28626124 100644
--- a/client/src/modules/modules.js
+++ b/client/src/modules/modules.js
@@ -4,4 +4,5 @@ export { default as CssEditor } from './csseditor';
export { default as PluginManager } from './pluginmanager';
export { default as ThemeManager } from './thememanager';
export { default as Globals } from './globals';
-export { default as Vendor } from './vendor';
\ No newline at end of file
+export { default as Vendor } from './vendor';
+export { default as WebpackModules } from './webpackmodules';
diff --git a/client/src/styles/partials/index.scss b/client/src/styles/partials/index.scss
index d40d30a0..b2d7157c 100644
--- a/client/src/styles/partials/index.scss
+++ b/client/src/styles/partials/index.scss
@@ -5,5 +5,6 @@
@import './sidebarview/index.scss';
@import './bdsettings/index.scss';
@import './generic/index.scss';
+@import './profilebadges.scss';
@import './discordoverrides.scss';
diff --git a/client/src/styles/partials/profilebadges.scss b/client/src/styles/partials/profilebadges.scss
new file mode 100644
index 00000000..9af642ed
--- /dev/null
+++ b/client/src/styles/partials/profilebadges.scss
@@ -0,0 +1,30 @@
+.bd-profile-badges-wrap {
+ display: flex;
+ margin-top: 8px;
+ flex: 1 1 auto;
+}
+
+.bd-profile-badges {
+ display: flex;
+}
+
+.bd-profile-badge {
+ background-position: 50%;
+ background-repeat: no-repeat;
+ background-size: cover;
+ cursor: pointer;
+ height: 16px;
+ margin-right: 6px;
+}
+
+.bd-profile-badge-developer,
+.bd-profile-badge-contributor {
+ background-image: $logoSmallBw;
+ width: 16px;
+ filter: brightness(10);
+
+ .theme-light [class*="topSectionNormal-"] & {
+ background-image: url('');
+ filter: none;
+ }
+}
diff --git a/client/src/ui/bdui.js b/client/src/ui/bdui.js
index 48d5a690..dd6646dd 100644
--- a/client/src/ui/bdui.js
+++ b/client/src/ui/bdui.js
@@ -8,23 +8,36 @@
* LICENSE file in the root directory of this source tree.
*/
-import Dom from './dom';
-import Vue from 'vue';
-import VTooltip from 'v-tooltip';
+import DOM from './dom';
+import Vue from './vue';
import { BdSettingsWrapper } from './components';
+import { Events, WebpackModules } from 'modules';
+import { Utils } from 'common';
export default class {
+
+ static initUiEvents() {
+ //this.profilePopupModule.open
+ const defer = setInterval(() => {
+ if (!this.profilePopupModule) return;
+ clearInterval(defer);
+ this.profilePopupModule.open = Utils.overload(this.profilePopupModule.open, userid => {
+ Events.emit('ui-event', {
+ event: 'profile-popup-open',
+ data: { userid }
+ });
+ });
+ }, 100);
+
+ }
+
+ static get profilePopupModule() {
+ return WebpackModules.getModuleByProps(['fetchMutualFriends', 'setSection']);
+ }
+
static injectUi() {
- Vue.use(VTooltip, {
- defaultContainer: 'bdtooltips',
- defaultClass: 'bd-tooltip',
- defaultTargetClass: 'bd-has-tooltip',
- defaultInnerSelector: '.bd-tooltip-inner',
- defaultTemplate: '
'
- });
-
- Dom.createElement('div', null, 'bd-settings').appendTo(Dom.bdBody);
-
+ DOM.createElement('bdtooltips').appendTo(DOM.bdBody);
+ DOM.createElement('div', null, 'bd-settings').appendTo(DOM.bdBody);
const vueInstance = new Vue({
el: '#bd-settings',
components: { BdSettingsWrapper },
diff --git a/client/src/ui/components/bd/BdBadge.vue b/client/src/ui/components/bd/BdBadge.vue
new file mode 100644
index 00000000..c151627d
--- /dev/null
+++ b/client/src/ui/components/bd/BdBadge.vue
@@ -0,0 +1,34 @@
+/**
+ * BetterDiscord BD Badge Component
+ * Copyright (c) 2015-present Jiiks/JsSucks - https://github.com/Jiiks / https://github.com/JsSucks
+ * All rights reserved.
+ * https://betterdiscord.net
+ *
+ * This source code is licensed under the MIT license found in the
+ * LICENSE file in the root directory of this source tree.
+*/
+
+
+
+
+
diff --git a/client/src/ui/components/bd/index.js b/client/src/ui/components/bd/index.js
index dfa19573..a0019f2f 100644
--- a/client/src/ui/components/bd/index.js
+++ b/client/src/ui/components/bd/index.js
@@ -3,4 +3,5 @@ export { default as CoreSettings } from './CoreSettings.vue';
export { default as UISettings } from './UISettings.vue';
export { default as EmoteSettings } from './EmoteSettings.vue';
export { default as CssEditorView } from './CssEditor.vue';
-export { default as PluginsView } from './PluginsView.vue';
\ No newline at end of file
+export { default as PluginsView } from './PluginsView.vue';
+export { default as BdBadge } from './BdBadge.vue';
diff --git a/client/src/ui/profilebadges.js b/client/src/ui/profilebadges.js
new file mode 100644
index 00000000..948fe15a
--- /dev/null
+++ b/client/src/ui/profilebadges.js
@@ -0,0 +1,71 @@
+/**
+ * BetterDiscord Developer/Contributor Profile Badges
+ * Copyright (c) 2015-present Jiiks/JsSucks - https://github.com/Jiiks / https://github.com/JsSucks
+ * All rights reserved.
+ * https://betterdiscord.net
+ *
+ * This source code is licensed under the MIT license found in the
+ * LICENSE file in the root directory of this source tree.
+*/
+
+import DOM from './dom';
+import { BdBadge } from './components/bd';
+import VueInjector from './vueinjector';
+import { Events } from 'modules';
+
+export default class {
+
+ static init() {
+ Events.on('ui-event', this.uiEvent.bind(this));
+ }
+
+ static uiEvent(e) {
+ const { event, data } = e;
+ if (event !== 'profile-popup-open') return;
+ const { userid } = data;
+ if (!userid) return;
+
+ this.inject(userid);
+ }
+
+ static inject(userid) {
+ const contributor = this.contributors.find(c => c.id == userid);
+ if (!contributor) return;
+
+ setTimeout(() => {
+ let hasBadges = false;
+ let root = document.querySelector('[class*=profileBadges]');
+ if (root) {
+ hasBadges = true;
+ } else {
+ root = document.querySelector('[class*="headerInfo"]');
+ }
+
+ VueInjector.inject(
+ root,
+ DOM.createElement('div', null, 'bdprofilebadges'),
+ { BdBadge },
+ ``
+ );
+ }, 200);
+ }
+
+ static filter(mutation) {
+ return mutation.target.firstChild && mutation.target.className.includes('modal');
+ }
+
+ static get contributors() {
+ return [
+ { 'id': 81388395867156480, 'webdev': true, 'developer': true, 'contributor': true }, // Jiiks
+ { 'id': 98003542823944192, 'webdev': false, 'developer': true, 'contributor': true }, // Pohky
+ { 'id': 138850472541814784, 'webdev': true, 'developer': false, 'contributor': true }, // Hammock
+ { 'id': 249746236008169473, 'webdev': false, 'developer': true, 'contributor': true }, // Zerebos
+ { 'id': 125367412370440192, 'webdev': false, 'developer': true, 'contributor': true }, // Pierce
+ { 'id': 284056145272766465, 'webdev': false, 'developer': false, 'contributor': true }, // Samuel Elliott
+ { 'id': 184021060562321419, 'webdev': false, 'developer': false, 'contributor': true }, // Lilian Tedone
+ { 'id': 76052829285916672, 'webdev': false, 'developer': false, 'contributor': true }, // samfun123
+ { 'id': 171005991272316937, 'webdev': false, 'developer': false, 'contributor': true }, // samogot
+ ];
+ }
+
+}
diff --git a/client/src/ui/ui.js b/client/src/ui/ui.js
index 3f794fff..e17a542e 100644
--- a/client/src/ui/ui.js
+++ b/client/src/ui/ui.js
@@ -1,2 +1,4 @@
export { default as DOM } from './dom';
export { default as BdUI } from './bdui';
+export { default as VueInjector } from './vueinjector';
+export { default as ProfileBadges } from './profilebadges';
diff --git a/client/src/ui/vue.js b/client/src/ui/vue.js
new file mode 100644
index 00000000..a3ec554b
--- /dev/null
+++ b/client/src/ui/vue.js
@@ -0,0 +1,22 @@
+/**
+ * BetterDiscord Client UI Module
+ * Copyright (c) 2015-present Jiiks/JsSucks - https://github.com/Jiiks / https://github.com/JsSucks
+ * All rights reserved.
+ * https://betterdiscord.net
+ *
+ * This source code is licensed under the MIT license found in the
+ * LICENSE file in the root directory of this source tree.
+*/
+
+import Vue from 'vue';
+import VTooltip from 'v-tooltip';
+
+Vue.use(VTooltip, {
+ defaultContainer: 'bdtooltips',
+ defaultClass: 'bd-tooltip',
+ defaultTargetClass: 'bd-has-tooltip',
+ defaultInnerSelector: '.bd-tooltip-inner',
+ defaultTemplate: '
'
+});
+
+export default Vue;
diff --git a/client/src/ui/vueinjector.js b/client/src/ui/vueinjector.js
new file mode 100644
index 00000000..0dfc0d93
--- /dev/null
+++ b/client/src/ui/vueinjector.js
@@ -0,0 +1,25 @@
+/**
+ * BetterDiscord VUE Injector Module
+ * Copyright (c) 2015-present Jiiks/JsSucks - https://github.com/Jiiks / https://github.com/JsSucks
+ * All rights reserved.
+ * https://betterdiscord.net
+ *
+ * This source code is licensed under the MIT license found in the
+ * LICENSE file in the root directory of this source tree.
+*/
+
+import Vue from './vue';
+
+export default class {
+
+ static inject(root, bdnode, components, template) {
+ bdnode.appendTo(root);
+
+ return new Vue({
+ el: bdnode.element,
+ components,
+ template
+ });
+ }
+
+}
diff --git a/package.json b/package.json
index c9adbf2f..855c6af4 100644
--- a/package.json
+++ b/package.json
@@ -2,7 +2,7 @@
"name": "betterdiscord",
"description": "BetterDiscord",
"author": "Jiiks",
- "version": "0.4.0",
+ "version": "2.0.0a",
"homepage": "https://betterdiscord.net",
"license": "MIT",
"main": "./core/index.js",