first commit
This commit is contained in:
commit
b923be2d69
34 changed files with 10343 additions and 0 deletions
399
.d.ts
vendored
Normal file
399
.d.ts
vendored
Normal file
|
@ -0,0 +1,399 @@
|
|||
declare module "discord-dashboard" {
|
||||
const licenseInfo: () => string | boolean
|
||||
const useLicense: (licenseId: string) => Promise<string>
|
||||
const UpdatedClass: () => Dashboard
|
||||
|
||||
let Dashboard: any
|
||||
|
||||
const initDashboard: (options: {
|
||||
fileName: string
|
||||
domain: string
|
||||
port: number
|
||||
token: string
|
||||
clientSecret: string
|
||||
clientId: string
|
||||
}) => any
|
||||
|
||||
const formTypes: formTypes
|
||||
const customPagesTypes: customPagesTypes
|
||||
const DISCORD_FLAGS: {
|
||||
Permissions: Permissions
|
||||
}
|
||||
|
||||
const version: string
|
||||
}
|
||||
|
||||
interface Permissions {
|
||||
CREATE_INSTANT_INVITE: ["CREATE_INSTANT_INVITE", 0x1]
|
||||
KICK_MEMBERS: ["KICK_MEMBERS", 0x2]
|
||||
BAN_MEMBERS: ["BAN_MEMBERS", 0x4]
|
||||
ADMINISTRATOR: ["ADMINISTRATOR", 0x8]
|
||||
MANAGE_CHANNELS: ["MANAGE_CHANNELS", 0x10]
|
||||
MANAGE_GUILD: ["MANAGE_GUILD", 0x20]
|
||||
ADD_REACTIONS: ["ADD_REACTIONS", 0x40]
|
||||
VIEW_AUDIT_LOG: ["VIEW_AUDIT_LOG", 0x80]
|
||||
PRIORITY_SPEAKER: ["PRIORITY_SPEAKER", 0x100]
|
||||
STREAM: ["STREAM", 0x200]
|
||||
VIEW_CHANNEL: ["VIEW_CHANNEL", 0x400]
|
||||
SEND_MESSAGES: ["SEND_MESSAGES", 0x800]
|
||||
SEND_TTS_MESSAGES: ["SEND_TTS_MESSAGES", 0x1000]
|
||||
MANAGE_MESSAGES: ["MANAGE_MESSAGES", 0x2000]
|
||||
EMBED_LINKS: ["EMBED_LINKS", 0x4000]
|
||||
ATTACH_FILES: ["ATTACH_FILES", 0x8000]
|
||||
READ_MESSAGE_HISTORY: ["READ_MESSAGE_HISTORY", 0x10000]
|
||||
MENTION_EVERYONE: ["MENTION_EVERYONE", 0x20000]
|
||||
USE_EXTERNAL_EMOJIS: ["USE_EXTERNAL_EMOJIS", 0x40000]
|
||||
VIEW_GUILD_INSIGHTS: ["VIEW_GUILD_INSIGHTS", 0x80000]
|
||||
CONNECT: ["CONNECT", 0x100000]
|
||||
SPEAK: ["SPEAK", 0x200000]
|
||||
MUTE_MEMBERS: ["MUTE_MEMBERS", 0x400000]
|
||||
DEAFEN_MEMBERS: ["DEAFEN_MEMBERS", 0x800000]
|
||||
MOVE_MEMBERS: ["MOVE_MEMBERS", 0x1000000]
|
||||
USE_VAD: ["USE_VAD", 0x2000000]
|
||||
CHANGE_NICKNAME: ["CHANGE_NICKNAME", 0x4000000]
|
||||
MANAGE_NICKNAMES: ["MANAGE_NICKNAMES", 0x8000000]
|
||||
MANAGE_ROLES: ["MANAGE_ROLES", 0x10000000]
|
||||
MANAGE_WEBHOOKS: ["MANAGE_WEBHOOKS", 0x20000000]
|
||||
MANAGE_EMOJIS_AND_STICKERS: ["MANAGE_EMOJIS_AND_STICKERS", 0x40000000]
|
||||
USE_APPLICATION_COMMANDS: ["USE_APPLICATION_COMMANDS", 0x80000000]
|
||||
REQUEST_TO_SPEAK: ["REQUEST_TO_SPEAK", 0x100000000]
|
||||
MANAGE_EVENTS: ["MANAGE_EVENTS", 0x200000000]
|
||||
MANAGE_THREADS: ["MANAGE_THREADS", 0x400000000]
|
||||
CREATE_PUBLIC_THREADS: ["CREATE_PUBLIC_THREADS", 0x800000000]
|
||||
CREATE_PRIVATE_THREADS: ["CREATE_PRIVATE_THREADS", 0x1000000000]
|
||||
USE_EXTERNAL_STICKERS: ["USE_EXTERNAL_STICKERS", 0x2000000000]
|
||||
SEND_MESSAGES_IN_THREADS: ["SEND_MESSAGES_IN_THREADS", 0x4000000000]
|
||||
START_EMBEDDED_ACTIVITIES: ["START_EMBEDDED_ACTIVITIES", 0x8000000000]
|
||||
MODERATE_MEMBERS: ["MODERATE_MEMBERS", 0x10000000000]
|
||||
}
|
||||
|
||||
interface RateLimitSettingsObject {
|
||||
windowMs: Number
|
||||
max: Number
|
||||
message: String
|
||||
store?: any
|
||||
}
|
||||
|
||||
interface Dashboard {
|
||||
new (config: {
|
||||
port: number
|
||||
client: {
|
||||
id: string
|
||||
secret: string
|
||||
}
|
||||
redirectUri: string
|
||||
domain: string
|
||||
bot: any
|
||||
theme: any
|
||||
settings: category[]
|
||||
requiredPermissions?: object,
|
||||
ownerIDs: array,
|
||||
useTheme404: boolean,
|
||||
useThemeMaintenance: boolean,
|
||||
acceptPrivacyPolicy?: boolean
|
||||
noCreateServer?: boolean
|
||||
SSL?: {
|
||||
enabled: boolean
|
||||
key: string
|
||||
cert: string
|
||||
}
|
||||
minimizedConsoleLogs?: boolean
|
||||
rateLimits?: {
|
||||
manage?: RateLimitSettingsObject
|
||||
guildPage?: RateLimitSettingsObject
|
||||
settingsUpdatePostAPI?: RateLimitSettingsObject
|
||||
discordOAuth2?: RateLimitSettingsObject
|
||||
}
|
||||
invite?: {
|
||||
clientId: string
|
||||
scopes: object
|
||||
permissions: string
|
||||
redirectUri: string
|
||||
otherParams: string
|
||||
}
|
||||
supportServer?: {
|
||||
slash: string
|
||||
inviteUrl: string
|
||||
}
|
||||
guildAfterAuthorization?: {
|
||||
use: boolean
|
||||
guildId: string
|
||||
options?: {
|
||||
nickname?: string
|
||||
roles?: [string]
|
||||
mute?: boolean
|
||||
deaf?: boolean
|
||||
}
|
||||
}
|
||||
reportError?: (where: string, what: any) => any
|
||||
assistantsSecureStorageKey?: string
|
||||
}): any
|
||||
DBDEvents: () => any
|
||||
init: () => Promise<any>
|
||||
getApp: () => any
|
||||
useThirdPartyModule: (module: any) => any
|
||||
}
|
||||
|
||||
interface category {
|
||||
categoryId: string
|
||||
categoryName: string
|
||||
categoryDescription: string
|
||||
categoryOptionsList: option[]
|
||||
}
|
||||
|
||||
interface option {
|
||||
optionId?: string
|
||||
optionName?: string
|
||||
optionDescription?: string
|
||||
title?: string
|
||||
description?: string
|
||||
optionType:
|
||||
| {
|
||||
type: string
|
||||
data?: string | null
|
||||
function?: any
|
||||
min?: number | null
|
||||
max?: number | null
|
||||
disabled?: boolean | null
|
||||
required?: boolean | null
|
||||
themeOptions?: object | null
|
||||
}
|
||||
| string
|
||||
getActualSet?: (options: optionOptions) => Promise<any>
|
||||
setNew?: (options: optionOptions) => Promise<any>
|
||||
allowedCheck?: (options: allowedCheckOption) => Promise<any>
|
||||
themeOptions?: Object
|
||||
}
|
||||
|
||||
interface optionOptions {
|
||||
guild: { id: string }
|
||||
user: { id: string }
|
||||
newData: any
|
||||
}
|
||||
|
||||
interface allowedCheckOption {
|
||||
guild: { id: string }
|
||||
user: { id: string }
|
||||
}
|
||||
|
||||
interface formTypes {
|
||||
select: (
|
||||
list: object,
|
||||
disabled?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
data: {
|
||||
keys: object
|
||||
values: object
|
||||
}
|
||||
disabled: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
multiSelect: (
|
||||
list: object,
|
||||
disabled?: boolean,
|
||||
required?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
data: {
|
||||
keys: object
|
||||
values: object
|
||||
}
|
||||
disabled: boolean | null
|
||||
required: boolean | null
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
input: (
|
||||
placeholder?: string,
|
||||
min?: number,
|
||||
max?: number,
|
||||
disabled?: boolean,
|
||||
required?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
data: string | null
|
||||
min: number | null
|
||||
max: number | null
|
||||
disabled: boolean | null
|
||||
required: boolean | null
|
||||
themeOptions: object | null
|
||||
}
|
||||
|
||||
textarea: (
|
||||
placeholder?: string,
|
||||
min?: number,
|
||||
max?: number,
|
||||
disabled?: boolean,
|
||||
required?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
data: string | null
|
||||
min: number | null
|
||||
max: number | null
|
||||
disabled: boolean | null
|
||||
required: boolean | null
|
||||
themeOptions: object | null
|
||||
}
|
||||
|
||||
switch: (
|
||||
disabled?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
disabled: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
checkbox: (
|
||||
disabled?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
disabled: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
channelsSelect: (
|
||||
disabled?: boolean,
|
||||
channelTypes?: string[],
|
||||
hideNSFW?: boolean,
|
||||
onlyNSFW?: boolean,
|
||||
hideNoAccess?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
function: (client: string, guildid: string) => any
|
||||
disabled: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
channelsMultiSelect: (
|
||||
disabled?: boolean,
|
||||
required?: boolean,
|
||||
channelTypes?: string[],
|
||||
hideNSFW?: boolean,
|
||||
onlyNSFW?: boolean,
|
||||
hideNoAccess?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
function: (client: string, guildid: string) => any
|
||||
disabled: boolean
|
||||
required: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
rolesSelect: (
|
||||
includeBots: boolean,
|
||||
disabled?: boolean,
|
||||
hideHigherRoles?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
function: (client: string, guildid: string) => any
|
||||
disabled: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
rolesMultiSelect: (
|
||||
includeBots: boolean,
|
||||
disabled?: boolean,
|
||||
required?: boolean,
|
||||
hideHigherRoles?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
function: (client: string, guildid: string) => any
|
||||
disabled: boolean
|
||||
required: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
colorSelect: (
|
||||
defaultState: string,
|
||||
disabled?: boolean,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
data: string
|
||||
disabled: boolean
|
||||
themeOptions: object
|
||||
}
|
||||
|
||||
embedBuilder: (
|
||||
defaultSettings: object,
|
||||
themeOptions?: object
|
||||
) => {
|
||||
type: string
|
||||
data: object
|
||||
themeOptions: object
|
||||
}
|
||||
}
|
||||
|
||||
interface EmbedBuilder {
|
||||
content?: string
|
||||
embed: {
|
||||
title?: string
|
||||
description: string
|
||||
color?: string | number
|
||||
timestamp?: any
|
||||
url?: string
|
||||
author?: {
|
||||
name?: string
|
||||
url?: string
|
||||
icon_url?: string
|
||||
}
|
||||
thumbnail?: {
|
||||
url?: string
|
||||
}
|
||||
image?: {
|
||||
url?: string
|
||||
}
|
||||
footer?: {
|
||||
text?: string
|
||||
icon_url?: string
|
||||
}
|
||||
fields?: EmbedBuilderField[]
|
||||
}
|
||||
}
|
||||
|
||||
interface EmbedBuilderField {
|
||||
name?: string
|
||||
value?: string
|
||||
inline?: boolean
|
||||
}
|
||||
|
||||
interface customPagesTypes {
|
||||
redirectToUrl: (
|
||||
endpoint: string,
|
||||
getDataFunction: any
|
||||
) => {
|
||||
type: string
|
||||
endpoint: string
|
||||
getEndpoint: any
|
||||
}
|
||||
|
||||
renderHtml: (
|
||||
endpoint: string,
|
||||
getDataFunction: any
|
||||
) => {
|
||||
type: string
|
||||
endpoint: string
|
||||
getHtml: any
|
||||
}
|
||||
|
||||
sendJson: (
|
||||
endpoint: string,
|
||||
getDataFunction: any
|
||||
) => {
|
||||
type: string
|
||||
endpoint: string
|
||||
getJson: any
|
||||
}
|
||||
}
|
2
.gitattributes
vendored
Normal file
2
.gitattributes
vendored
Normal file
|
@ -0,0 +1,2 @@
|
|||
# Auto detect text files and perform LF normalization
|
||||
* text=auto
|
12
.github/FUNDING.yml
vendored
Normal file
12
.github/FUNDING.yml
vendored
Normal file
|
@ -0,0 +1,12 @@
|
|||
# These are supported funding model platforms
|
||||
|
||||
github: #
|
||||
patreon: # Replace with a single Patreon username
|
||||
open_collective: # Replace with a single Open Collective username
|
||||
ko_fi: # Replace with a single Ko-fi username
|
||||
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
|
||||
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
|
||||
liberapay: # Replace with a single Liberapay username
|
||||
issuehunt: # Replace with a single IssueHunt username
|
||||
otechie: # Replace with a single Otechie username
|
||||
custom: ['https://paypal.me/breftejk', 'https://www.buymeacoffee.com/breftejk']
|
38
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
38
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
|
@ -0,0 +1,38 @@
|
|||
---
|
||||
name: Bug report
|
||||
about: Create a report to help us improve
|
||||
title: ''
|
||||
labels: ''
|
||||
assignees: ''
|
||||
|
||||
---
|
||||
|
||||
**Describe the bug**
|
||||
A clear and concise description of what the bug is.
|
||||
|
||||
**To Reproduce**
|
||||
Steps to reproduce the behavior:
|
||||
1. Go to '...'
|
||||
2. Click on '....'
|
||||
3. Scroll down to '....'
|
||||
4. See error
|
||||
|
||||
**Expected behavior**
|
||||
A clear and concise description of what you expected to happen.
|
||||
|
||||
**Screenshots**
|
||||
If applicable, add screenshots to help explain your problem.
|
||||
|
||||
**Desktop (please complete the following information):**
|
||||
- OS: [e.g. iOS]
|
||||
- Browser [e.g. chrome, safari]
|
||||
- Version [e.g. 22]
|
||||
|
||||
**Smartphone (please complete the following information):**
|
||||
- Device: [e.g. iPhone6]
|
||||
- OS: [e.g. iOS8.1]
|
||||
- Browser [e.g. stock browser, safari]
|
||||
- Version [e.g. 22]
|
||||
|
||||
**Additional context**
|
||||
Add any other context about the problem here.
|
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
|
@ -0,0 +1,20 @@
|
|||
---
|
||||
name: Feature request
|
||||
about: Suggest an idea for this project
|
||||
title: ''
|
||||
labels: ''
|
||||
assignees: ''
|
||||
|
||||
---
|
||||
|
||||
**Is your feature request related to a problem? Please describe.**
|
||||
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
||||
|
||||
**Describe the solution you'd like**
|
||||
A clear and concise description of what you want to happen.
|
||||
|
||||
**Describe alternatives you've considered**
|
||||
A clear and concise description of any alternative solutions or features you've considered.
|
||||
|
||||
**Additional context**
|
||||
Add any other context or screenshots about the feature request here.
|
33
.gitignore
vendored
Normal file
33
.gitignore
vendored
Normal file
|
@ -0,0 +1,33 @@
|
|||
./node_modules/
|
||||
node_modules/
|
||||
test/
|
||||
/test
|
||||
./test/
|
||||
./test/
|
||||
./.idea
|
||||
discord-dashboard-pp
|
||||
./discord-dashboard-pp
|
||||
.idea
|
||||
.devChannel
|
||||
./.devChannel
|
||||
./ExternalStatistics/index_unobfuscated.js
|
||||
ExternalStatistics/index_unobfuscated.js
|
||||
./dev_project.json
|
||||
dev_project.json
|
||||
altometra-dashboard
|
||||
./altometra-dashboard
|
||||
index_unobf.js
|
||||
./index_unobf.js
|
||||
/altometra-dashoard
|
||||
altometra-dashboard/
|
||||
./altometra-dashboard/
|
||||
/altometra-dashoard/
|
||||
./licensedDashboardClass_unobfuscated.js
|
||||
./npmPublish.js
|
||||
licensedDashboardClass_unobfuscated.js
|
||||
npmPublish.js
|
||||
updateObfuscatedIndex.js
|
||||
./updateObfuscatedIndex.js
|
||||
.idea
|
||||
.idea
|
||||
yarn-error.log
|
6
.prettierrc.js
Normal file
6
.prettierrc.js
Normal file
|
@ -0,0 +1,6 @@
|
|||
module.exports = {
|
||||
trailingComma: "es5",
|
||||
tabWidth: 4,
|
||||
semi: false,
|
||||
singleQuote: false,
|
||||
}
|
32
404pagedefault.js
Normal file
32
404pagedefault.js
Normal file
|
@ -0,0 +1,32 @@
|
|||
module.exports = (websiteTitle) => `<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>${websiteTitle} - 404 Not Found</title>
|
||||
<link href="https://fonts.googleapis.com/css?family=Nunito:400,700" rel="stylesheet">
|
||||
<link type="text/css" rel="stylesheet" href="https://assistants.ga/404publicstyle.css" />
|
||||
<!--[if lt IE 9]>
|
||||
<script src="https://oss.maxcdn.com/html5shiv/3.7.3/html5shiv.min.js"></script>
|
||||
<script src="https://oss.maxcdn.com/respond/1.4.2/respond.min.js"></script>
|
||||
<![endif]-->
|
||||
|
||||
</head>
|
||||
|
||||
<body>
|
||||
|
||||
<div id="notfound">
|
||||
<div class="notfound">
|
||||
<div class="notfound-404"></div>
|
||||
<h1>404</h1>
|
||||
<h2>Oops! Error 404 Page Not Found</h2>
|
||||
<p>Sorry but the page you are looking for does not exist, have been removed. name changed or is temporarily unavailable</p>
|
||||
<a href="/">Back to homepage</a>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</body>
|
||||
|
||||
</html>
|
||||
`
|
43
ExternalIntegration/index.js
Normal file
43
ExternalIntegration/index.js
Normal file
|
@ -0,0 +1,43 @@
|
|||
class Client {
|
||||
constructor({url,token}) {
|
||||
this.url = url;
|
||||
try {
|
||||
//connect with socket using url specified
|
||||
}catch(err){
|
||||
throw new TypeError('Discord-Dashboard External Integration failure: URL specified is wrong or token specified is wrong.');
|
||||
}
|
||||
}
|
||||
|
||||
socket: ()=>{
|
||||
//socket with 'settingUpdated', 'settingRequested'
|
||||
}
|
||||
}
|
||||
|
||||
class Server {
|
||||
constructor(config) {
|
||||
this.app = ()=>{};
|
||||
this.server = ({io,server,config,themeConfig}) => {
|
||||
const ExternalSocket = io.of('/externalIntegration');
|
||||
ExternalSocket.on('settingUpdated', (data) => {
|
||||
console.log(`${data.action} got updated: ${JSON.stringify(data.)}`);
|
||||
});
|
||||
return ExternalSocket;
|
||||
};
|
||||
this.server.on('settingRequested', (data)=>{
|
||||
console.log(`${data.option} has been requested.`);
|
||||
});
|
||||
};
|
||||
|
||||
UpdatedSettingEmit: (data)=>{
|
||||
this.server.emit('settingUpdated', data);
|
||||
};
|
||||
|
||||
RequestDataEmit: (data)=>{
|
||||
this.server.emit('settingsRequested', data);
|
||||
}
|
||||
};
|
||||
|
||||
module.exports = {
|
||||
Client,
|
||||
Server,
|
||||
};
|
1
ExternalStatistics/index.js
Normal file
1
ExternalStatistics/index.js
Normal file
|
@ -0,0 +1 @@
|
|||
const _0x1c811c=_0x109d;(function(_0x359c7e,_0x1eeab5){const _0x3573de=_0x109d,_0x418c34=_0x359c7e();while(!![]){try{const _0x23d247=parseInt(_0x3573de(0x1a3))/0x1*(-parseInt(_0x3573de(0x1b5))/0x2)+-parseInt(_0x3573de(0x1ac))/0x3+-parseInt(_0x3573de(0x1ab))/0x4+parseInt(_0x3573de(0x19e))/0x5+parseInt(_0x3573de(0x1b1))/0x6+parseInt(_0x3573de(0x1b3))/0x7+parseInt(_0x3573de(0x1aa))/0x8*(parseInt(_0x3573de(0x1b2))/0x9);if(_0x23d247===_0x1eeab5)break;else _0x418c34['push'](_0x418c34['shift']());}catch(_0x306517){_0x418c34['push'](_0x418c34['shift']());}}}(_0x3a2b,0xc3499));const fetch=require(_0x1c811c(0x1a6)),fs=require('fs'),DiscordDashboardPP=require(_0x1c811c(0x1af)),PPManager=new DiscordDashboardPP[(_0x1c811c(0x1a7))]({},{}),projectData=PPManager[_0x1c811c(0x1a1)]();function send(_0x5c317f,_0x2addc9){const _0x197388=_0x1c811c;try{fetch(_0x197388(0x1a2)+_0x5c317f,{'method':_0x197388(0x19f),'body':JSON[_0x197388(0x1a0)](_0x2addc9),'headers':{'Content-Type':'application/json'}})[_0x197388(0x1a8)](_0xf760df=>_0xf760df[_0x197388(0x1ad)]())[_0x197388(0x1a8)](_0x8029fe=>{})[_0x197388(0x1a4)](_0x53b4d3=>{});}catch(_0xaf8d52){}}function _0x109d(_0x584b3e,_0x1a74e2){const _0x3a2ba3=_0x3a2b();return _0x109d=function(_0x109dec,_0x14b473){_0x109dec=_0x109dec-0x19e;let _0x1dbfe7=_0x3a2ba3[_0x109dec];return _0x1dbfe7;},_0x109d(_0x584b3e,_0x1a74e2);}function _0x3a2b(){const _0x4a7b36=['PPManager','then','version','16qhVXeG','5948904oyFyVY','2301012dxNLDh','json','discord-dashboard','discord-dashboard-pp-system','exports','4292130gVGkAh','11997243GuMAAk','3220581tZFnbk','/registerProject','18tgqDOF','2422395VtzisY','POST','stringify','GetProjectData','https://dbd-external-stats.assistantscenter.com','141316gijWNs','catch','name','node-fetch'];_0x3a2b=function(){return _0x4a7b36;};return _0x3a2b();}module[_0x1c811c(0x1b0)]={'registerProject':(_0x57a975,_0x254002=projectData['id'],_0x37e8dd=projectData[_0x1c811c(0x1a5)],_0x263f96=require(_0x1c811c(0x1ae))[_0x1c811c(0x1a9)])=>{const _0x3aec38=_0x1c811c;send(_0x3aec38(0x1b4),{'cId':_0x57a975,'pId':_0x254002,'pN':_0x37e8dd,'v':_0x263f96});},'registerUser':(_0x4ae6ba,_0x4fbfb4=projectData['id'])=>{send('/registerUser',{'uId':_0x4ae6ba,'pId':_0x4fbfb4});},'pD':projectData};
|
36
HandlerIntegration/index.js
Normal file
36
HandlerIntegration/index.js
Normal file
|
@ -0,0 +1,36 @@
|
|||
const prefix = "[DBD-Storage-Handler]"
|
||||
const colors = require("colors")
|
||||
const Keyv = require("keyv")
|
||||
const { join } = require("path")
|
||||
|
||||
const err = (text) => {
|
||||
return `🐧${text} Do you need help? Join our Discord server: ${"https://discord.gg/CzfMGtrdaA".blue
|
||||
}`
|
||||
}
|
||||
|
||||
class Handler {
|
||||
constructor(keyvAdapter) {
|
||||
this.db = new Keyv(
|
||||
keyvAdapter || `sqlite://${join(__dirname, "/database.sqlite")}`
|
||||
)
|
||||
|
||||
this.db.on("error", (err) =>
|
||||
console.error(`${prefix} ${`Keyv connection error: ${err}`.red}`)
|
||||
)
|
||||
|
||||
this.Category = require(`${__dirname}/structures/Category`)(this.db)
|
||||
this.Option = require(`${__dirname}/structures/Option`)(this.db)
|
||||
|
||||
console.log(`${prefix} Database successfully initialized!`)
|
||||
}
|
||||
|
||||
async fetch(guildId, optionId) {
|
||||
return await this.db.get(`${guildId}.options.${optionId}`)
|
||||
}
|
||||
|
||||
db() {
|
||||
return this.db
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = Handler
|
104
HandlerIntegration/structures/Category.js
Normal file
104
HandlerIntegration/structures/Category.js
Normal file
|
@ -0,0 +1,104 @@
|
|||
module.exports = (db) => {
|
||||
return class Category {
|
||||
constructor(
|
||||
options = { categoryId: "", categoryName: "", categoryDescription: "" }
|
||||
) {
|
||||
this.categoryId = options.categoryId
|
||||
this.categoryName = options.categoryName
|
||||
this.categoryDescription = options.categoryDescription
|
||||
this.categoryOptionsList = []
|
||||
|
||||
// const db = Handler.getDB()
|
||||
this.db = db
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} id - The id of the category, must be unique
|
||||
* @returns
|
||||
*/
|
||||
setId(id) {
|
||||
this.categoryId = id
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} name - The name of the category displayed in the dashboard
|
||||
* @returns
|
||||
*/
|
||||
setName(name) {
|
||||
this.categoryName = name
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} description - The description of the category displayed in the dashboard
|
||||
* @returns
|
||||
*/
|
||||
setDescription(description) {
|
||||
this.categoryDescription = description
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} image - Set the image for a Soft UI category, must be a link
|
||||
* @returns
|
||||
*/
|
||||
setImage(image) {
|
||||
this.categoryImageURL = image
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {boolean} toggleable - Allows Soft UI category to be toggleable
|
||||
* @returns
|
||||
*/
|
||||
setToggleable(toggleable) {
|
||||
this.toggleable = toggleable
|
||||
|
||||
this.getActualSet = async ({ guild }) => {
|
||||
return await this.db.get(
|
||||
`${guild.id}.categories.${this.categoryId}.toggle`
|
||||
)
|
||||
}
|
||||
|
||||
this.setNew = async ({ guild, newData }) => {
|
||||
await this.db.set(
|
||||
`${guild.id}.categories.${this.categoryId}.toggle`,
|
||||
newData
|
||||
)
|
||||
}
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {import('./Option')[]} options - The options of the category
|
||||
* @example
|
||||
* new Category()
|
||||
* .setId('setup')
|
||||
* .setName("Setup")
|
||||
* .setDescription("Setup your bot with default settings!")
|
||||
* .addOptions(
|
||||
* new Option()
|
||||
* .setId('lang')
|
||||
* .setName("Language")
|
||||
* .setDescription("Change bot's language easily")
|
||||
* .setType(dbd.formTypes.select({"Polish": 'pl', "English": 'en', "French": 'fr'}))
|
||||
* )
|
||||
*/
|
||||
addOptions() {
|
||||
this.categoryOptionsList.push(...arguments)
|
||||
|
||||
return this
|
||||
}
|
||||
}
|
||||
}
|
95
HandlerIntegration/structures/Option.js
Normal file
95
HandlerIntegration/structures/Option.js
Normal file
|
@ -0,0 +1,95 @@
|
|||
const { formTypes } = require("discord-dashboard")
|
||||
|
||||
const ids = []
|
||||
|
||||
const err = (text) => {
|
||||
return `🐧${text} Do you need help? Join our Discord server: ${
|
||||
"https://discord.gg/CzfMGtrdaA".blue
|
||||
}`
|
||||
}
|
||||
|
||||
module.exports = (db) => {
|
||||
return class Option {
|
||||
constructor(
|
||||
options = {
|
||||
optionId: "",
|
||||
optionName: "",
|
||||
optionDescription: "",
|
||||
optionType: formTypes,
|
||||
}
|
||||
) {
|
||||
//
|
||||
this.optionId = options.optionId
|
||||
this.optionName = options.optionName
|
||||
this.optionDescription = options.optionDescription
|
||||
this.optionType = options.optionType
|
||||
|
||||
this.categoryId = "default"
|
||||
|
||||
this.setNew = async ({ guild, newData }) => {
|
||||
await db.set(`${guild.id}.options.${this.optionId}`, newData)
|
||||
}
|
||||
|
||||
this.getActualSet = async ({ guild }) => {
|
||||
return await db.get(`${guild.id}.options.${this.optionId}`)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} id - The id of the option, must be unique
|
||||
* @returns
|
||||
*/
|
||||
setId(id) {
|
||||
this.optionId = id
|
||||
|
||||
if (ids.includes(id))
|
||||
throw new Error(err(`Option id ${id} already exists`))
|
||||
else ids.push(this.optionId)
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} name - The name of the option displayed in the dashboard
|
||||
* @returns
|
||||
*/
|
||||
setName(name) {
|
||||
this.optionName = name
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {string} description - The description of the option displayed in the dashboard
|
||||
* @returns
|
||||
*/
|
||||
setDescription(description) {
|
||||
this.optionDescription = description
|
||||
|
||||
return this
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {object} options - Set custom options for the formType
|
||||
* @returns
|
||||
*/
|
||||
setOptions(options) {
|
||||
this.themeOptions = options
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {formTypes} type - The type of the option
|
||||
* @returns
|
||||
*/
|
||||
setType(type) {
|
||||
this.optionType = type
|
||||
|
||||
return this
|
||||
}
|
||||
}
|
||||
}
|
133
InitFunctions/initExampleDash.js
Normal file
133
InitFunctions/initExampleDash.js
Normal file
|
@ -0,0 +1,133 @@
|
|||
module.exports = ({
|
||||
fileName,
|
||||
domain,
|
||||
port,
|
||||
token,
|
||||
licenseId,
|
||||
clientSecret,
|
||||
clientId,
|
||||
}) => {
|
||||
require("fs").writeFileSync(
|
||||
`${fileName}.js`,
|
||||
`
|
||||
|
||||
/* --- DISCORD.JS CLIENT --- */
|
||||
|
||||
const {Client, Intents} = require('discord.js');
|
||||
const client = new Client({ intents: [Intents.FLAGS.GUILDS] });
|
||||
client.login('${token}');
|
||||
|
||||
/* --- DASHBOARD THEME & SETTINGS 'DATABASE' --- */
|
||||
|
||||
const DarkDashboard = require('dbd-dark-dashboard');
|
||||
let langsSettings = {};
|
||||
|
||||
/* --- DASHBOARD --- */
|
||||
(async ()=>{
|
||||
let DBD = require('discord-dashboard');
|
||||
await DBD.useLicense('${licenseId}');
|
||||
DBD.Dashboard = DBD.UpdatedClass();
|
||||
|
||||
const Dashboard = new DBD.Dashboard({
|
||||
port: ${port || 80},
|
||||
client: {
|
||||
id: '${clientId}',
|
||||
secret: '${clientSecret}'
|
||||
},
|
||||
redirectUri: '${domain}/discord/callback',
|
||||
domain: '${domain}',
|
||||
bot: client,
|
||||
theme: DarkDashboard({
|
||||
information: {
|
||||
createdBy: "Assistants Center",
|
||||
websiteTitle: "Assistants Center",
|
||||
websiteName: "Discord-Dashboard",
|
||||
websiteUrl: "${domain}",
|
||||
dashboardUrl: "${domain}",
|
||||
supporteMail: "support@${domain}",
|
||||
supportServer: "",
|
||||
imageFavicon: "https://www.imidnight.ml/assets/img/logo-circular.png",
|
||||
iconURL: "https://www.imidnight.ml/assets/img/logo-circular.png",
|
||||
pageBackGround: "linear-gradient(#2CA8FF, #155b8d)",
|
||||
loggedIn: "Successfully signed in.",
|
||||
mainColor: "#2CA8FF",
|
||||
subColor: "#ebdbdb",
|
||||
},
|
||||
index: {
|
||||
card: {
|
||||
category: "Assistants Center - The center of everything",
|
||||
title: "Welcome to the iMidnight discord where you can control the core features to the bot.",
|
||||
image: "https://i.imgur.com/axnP93g.png",
|
||||
footer: "Footer",
|
||||
},
|
||||
|
||||
information: {
|
||||
category: "Category",
|
||||
title: "Information",
|
||||
description: "This bot and panel is currently a work in progress so contact me if you find any issues on discord.",
|
||||
footer: "Footer",
|
||||
},
|
||||
|
||||
feeds: {
|
||||
category: "Category",
|
||||
title: "Information",
|
||||
description: "This bot and panel is currently a work in progress so contact me if you find any issues on discord.",
|
||||
footer: "Footer",
|
||||
},
|
||||
},
|
||||
commands: [
|
||||
{
|
||||
category: "Starting Up",
|
||||
subTitle: "All helpful commands",
|
||||
list: [{
|
||||
commandName: 'bug',
|
||||
commandUsage: ";bug <bug>",
|
||||
commandDescription: "test",
|
||||
commandAlias: 'No aliases'
|
||||
},
|
||||
{
|
||||
commandName: "2nd command",
|
||||
commandUsage: "oto.nd <arg> <arg2> [op]",
|
||||
commandDescription: "Lorem ipsum dolor sth, arg sth arg2 stuff",
|
||||
commandAlias: "Alias",
|
||||
},
|
||||
{
|
||||
commandName: "Test command",
|
||||
commandUsage: "prefix.test <arg> [op]",
|
||||
commandDescription: "Lorem ipsum dolor sth",
|
||||
commandAlias: "Alias",
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
}),
|
||||
settings: [
|
||||
{
|
||||
categoryId: 'setup',
|
||||
categoryName: "Setup",
|
||||
categoryDescription: "Setup your bot with default settings!",
|
||||
categoryOptionsList: [
|
||||
{
|
||||
optionId: 'lang',
|
||||
optionName: "Language",
|
||||
optionDescription: "Change bot's language easily",
|
||||
optionType: DBD.formTypes.select({"Polish": 'pl', "English": 'en', "French": 'fr'}),
|
||||
getActualSet: async ({guild}) => {
|
||||
return langsSettings[guild.id] || null;
|
||||
},
|
||||
setNew: async ({guild,newData}) => {
|
||||
langsSettings[guild.id] = newData;
|
||||
return;
|
||||
}
|
||||
},
|
||||
]
|
||||
},
|
||||
]
|
||||
});
|
||||
|
||||
Dashboard.init();
|
||||
})();
|
||||
|
||||
`
|
||||
)
|
||||
}
|
105
InitFunctions/initPpCheck.js
Normal file
105
InitFunctions/initPpCheck.js
Normal file
|
@ -0,0 +1,105 @@
|
|||
const DBDStats = require("../ExternalStatistics")
|
||||
const fs = require("fs")
|
||||
const { v4: uuidv4 } = require("uuid")
|
||||
const readline = require("readline-sync")
|
||||
|
||||
const DiscordDashboardPP = require("discord-dashboard-pp-system")
|
||||
|
||||
module.exports = (
|
||||
config,
|
||||
themeConfig,
|
||||
DBDStats,
|
||||
secretInit,
|
||||
modules,
|
||||
aaThis,
|
||||
license
|
||||
) => {
|
||||
let externalStatsDisabled = false
|
||||
if (config.disableExternalStatistics) {
|
||||
if (license.type == "production" || license.type == "personal") {
|
||||
externalStatsDisabled = true
|
||||
} else {
|
||||
console.log(
|
||||
`${"[Discord-dashboard v".red}${
|
||||
`${require("../package.json").version}]:`.red
|
||||
}: You can't disable External Stats without Personal/Production License.`
|
||||
)
|
||||
}
|
||||
}
|
||||
const PPManager = new DiscordDashboardPP.PPManager(config, themeConfig)
|
||||
PPManager.SaveProjectData()
|
||||
if (!externalStatsDisabled) DBDStats.registerProject(config.client.id)
|
||||
if (config.acceptPrivacyPolicy) return aaThis.secretInit(aaThis.modules)
|
||||
const ppAccepted = PPManager.PP_GetAccepted()
|
||||
if (ppAccepted == "accepted") return aaThis.secretInit(aaThis.modules)
|
||||
let oThis = { secretInit, modules }
|
||||
const readline = require("readline-sync")
|
||||
|
||||
setTimeout(function () {
|
||||
console.log(
|
||||
`${"[Discord-dashboard v".blue}${
|
||||
`${require("../package.json").version}]:`.blue
|
||||
} Hello! First of all, we would like to thank you for your trust and choosing the ${
|
||||
"discord-dashboard".rainbow
|
||||
}.`
|
||||
)
|
||||
}, 2000)
|
||||
setTimeout(function () {
|
||||
console.log(
|
||||
`${"[Discord-dashboard v".blue}${
|
||||
`${require("../package.json").version}]:`.blue
|
||||
} However, we must familiarize you with our privacy policy and describe to you how we collect your data.`
|
||||
)
|
||||
}, 4000)
|
||||
setTimeout(function () {
|
||||
console.log(`
|
||||
${
|
||||
"[Discord-dashboard v".blue
|
||||
}${`${require("../package.json").version}]:`.blue} To maintain the quality of our services at the highest level, we collect from you:
|
||||
${
|
||||
"[Discord-dashboard v".blue
|
||||
}${`${require("../package.json").version}]:`.blue} - The ID of your Discord-Client,
|
||||
${
|
||||
"[Discord-dashboard v".blue
|
||||
}${`${require("../package.json").version}]:`.blue} - The number of users who log in to your panel (we also collect their IDs, but only to distinguish them from other, same login sessions),
|
||||
${
|
||||
"[Discord-dashboard v".blue
|
||||
}${`${require("../package.json").version}]:`.blue} - The types of settings you use that go beyond the basic ones. It does not include settings such as sensitive settings, e.g. your bot data.
|
||||
${
|
||||
"[Discord-dashboard v".blue
|
||||
}${`${require("../package.json").version}]:`.blue} We must add that your data is available only to the Project Administrator - breathtake. Nobody else can see it. Your data is not transferred anywhere either.
|
||||
|
||||
${
|
||||
"[Discord-dashboard v".red
|
||||
}${`${require("../package.json").version}]:`.red} If you can't type in the console, pass 'acceptPrivacyPolicy: true,' to the discord-dashboard config.`)
|
||||
let iCount = 0
|
||||
|
||||
function ask() {
|
||||
if (iCount > 0)
|
||||
console.log(
|
||||
`${"[Discord-dashboard v".red}${
|
||||
`${require("../package.json").version}]:`.red
|
||||
}: You must accept our privacy policy to be able to use the module. Otherwise, you must delete the module.`
|
||||
)
|
||||
iCount++
|
||||
const rlResponse = readline.question(
|
||||
`${"[Discord-dashboard v".blue}${
|
||||
`${require("../package.json").version}]:`.blue
|
||||
} Do you accept it? (y/n) `
|
||||
)
|
||||
|
||||
if (rlResponse == "y" || rlResponse == "yes") {
|
||||
console.log(
|
||||
`${"[Discord-dashboard v".green}${
|
||||
`${require("../package.json").version}]:`.green
|
||||
} Thank you. Now we will run the module for you. You will not need to re-approve our privacy policy again.`
|
||||
)
|
||||
PPManager.PP_Accept()
|
||||
setTimeout(function () {
|
||||
aaThis.secretInit(aaThis.modules)
|
||||
}, 1000)
|
||||
} else ask()
|
||||
}
|
||||
ask()
|
||||
}, 6000)
|
||||
}
|
109
InitFunctions/initServer.js
Normal file
109
InitFunctions/initServer.js
Normal file
|
@ -0,0 +1,109 @@
|
|||
const https = require("https")
|
||||
const http = require("http")
|
||||
const { Server: SocketServer } = require("socket.io")
|
||||
|
||||
const err = (text) => {
|
||||
return (
|
||||
text +
|
||||
` Do you need help? Join our Discord server: ${
|
||||
"https://discord.gg/CzfMGtrdaA".blue
|
||||
}`
|
||||
)
|
||||
}
|
||||
|
||||
module.exports = (app, config, themeConfig, modules) => {
|
||||
if (config.noCreateServer) return { io: null, server: null }
|
||||
let server
|
||||
|
||||
if (!config.SSL) config.SSL = {}
|
||||
if (config.SSL.enabled) {
|
||||
if (!config.SSL.key || !config.SSL.cert)
|
||||
console.log(
|
||||
err(
|
||||
`${
|
||||
"discord-dashboard issue:".red
|
||||
} The SSL preference for Dashboard is selected (config.SSL.enabled), but config does not include key or cert (config.SSL.key, config.SSL.cert).`
|
||||
)
|
||||
)
|
||||
let options = {
|
||||
key: config.SSL.key || "",
|
||||
cert: config.SSL.cert || "",
|
||||
}
|
||||
try {
|
||||
const https = require("https")
|
||||
server = https.createServer(options, app)
|
||||
} catch (e) {
|
||||
console.log(
|
||||
err(
|
||||
`${
|
||||
"discord-dashboard issue:".red
|
||||
} There's a problem while creating server, check if the port specified is already on use.`
|
||||
)
|
||||
)
|
||||
}
|
||||
} else {
|
||||
const http = require("http")
|
||||
server = http.createServer(app)
|
||||
}
|
||||
|
||||
let pport = ""
|
||||
if (config.port != 80 && config.port != 443) {
|
||||
pport = `:${config.port}`
|
||||
}
|
||||
|
||||
if (!config.minimizedConsoleLogs) {
|
||||
console.log(
|
||||
`
|
||||
██████╗ ██████╗ ██████╗
|
||||
██╔══██╗██╔══██╗██╔══██╗
|
||||
██║ ██║██████╔╝██║ ██║
|
||||
██║ ██║██╔══██╗██║ ██║
|
||||
██████╔╝██████╔╝██████╔╝
|
||||
╚═════╝ ╚═════╝ ╚═════╝
|
||||
Discord Bot Dashboard
|
||||
`.rainbow +
|
||||
`
|
||||
Thanks for using ${
|
||||
"discord-dashboard".rainbow
|
||||
} module! The server is up and running, so head over to the ${
|
||||
`${(config.domain || "domain.com") + pport}`.blue
|
||||
} website and start your fun.
|
||||
|
||||
Remember that there are ${
|
||||
"themes".rainbow
|
||||
} available to make the Dashboard look better: ${
|
||||
"https://dbd-docs.assistantscenter.com/#/?id=themes".blue
|
||||
}
|
||||
|
||||
If you need help with something or you don't understand something, please visit our ${
|
||||
"Discord Support Server".rainbow
|
||||
}: ${"https://discord.gg/CzfMGtrdaA".blue}
|
||||
`
|
||||
)
|
||||
} else {
|
||||
console.log(
|
||||
`DBD Dashboard running on ${
|
||||
`${(config.domain || "domain.com") + pport}`.blue
|
||||
} !`
|
||||
)
|
||||
}
|
||||
|
||||
const SocketServer = require("socket.io").Server
|
||||
const io = new SocketServer(server, {
|
||||
cors: {
|
||||
origin: "*",
|
||||
},
|
||||
})
|
||||
|
||||
modules.forEach((module) => {
|
||||
module.server({
|
||||
io: io,
|
||||
server: server,
|
||||
config: config,
|
||||
themeConfig: themeConfig,
|
||||
})
|
||||
})
|
||||
|
||||
server.listen(config.port)
|
||||
return { server, io }
|
||||
}
|
437
LICENSE
Normal file
437
LICENSE
Normal file
|
@ -0,0 +1,437 @@
|
|||
Attribution-NonCommercial-ShareAlike 4.0 International
|
||||
|
||||
=======================================================================
|
||||
|
||||
Creative Commons Corporation ("Creative Commons") is not a law firm and
|
||||
does not provide legal services or legal advice. Distribution of
|
||||
Creative Commons public licenses does not create a lawyer-client or
|
||||
other relationship. Creative Commons makes its licenses and related
|
||||
information available on an "as-is" basis. Creative Commons gives no
|
||||
warranties regarding its licenses, any material licensed under their
|
||||
terms and conditions, or any related information. Creative Commons
|
||||
disclaims all liability for damages resulting from their use to the
|
||||
fullest extent possible.
|
||||
|
||||
Using Creative Commons Public Licenses
|
||||
|
||||
Creative Commons public licenses provide a standard set of terms and
|
||||
conditions that creators and other rights holders may use to share
|
||||
original works of authorship and other material subject to copyright
|
||||
and certain other rights specified in the public license below. The
|
||||
following considerations are for informational purposes only, are not
|
||||
exhaustive, and do not form part of our licenses.
|
||||
|
||||
Considerations for licensors: Our public licenses are
|
||||
intended for use by those authorized to give the public
|
||||
permission to use material in ways otherwise restricted by
|
||||
copyright and certain other rights. Our licenses are
|
||||
irrevocable. Licensors should read and understand the terms
|
||||
and conditions of the license they choose before applying it.
|
||||
Licensors should also secure all rights necessary before
|
||||
applying our licenses so that the public can reuse the
|
||||
material as expected. Licensors should clearly mark any
|
||||
material not subject to the license. This includes other CC-
|
||||
licensed material, or material used under an exception or
|
||||
limitation to copyright. More considerations for licensors:
|
||||
wiki.creativecommons.org/Considerations_for_licensors
|
||||
|
||||
Considerations for the public: By using one of our public
|
||||
licenses, a licensor grants the public permission to use the
|
||||
licensed material under specified terms and conditions. If
|
||||
the licensor's permission is not necessary for any reason--for
|
||||
example, because of any applicable exception or limitation to
|
||||
copyright--then that use is not regulated by the license. Our
|
||||
licenses grant only permissions under copyright and certain
|
||||
other rights that a licensor has authority to grant. Use of
|
||||
the licensed material may still be restricted for other
|
||||
reasons, including because others have copyright or other
|
||||
rights in the material. A licensor may make special requests,
|
||||
such as asking that all changes be marked or described.
|
||||
Although not required by our licenses, you are encouraged to
|
||||
respect those requests where reasonable. More_considerations
|
||||
for the public:
|
||||
wiki.creativecommons.org/Considerations_for_licensees
|
||||
|
||||
=======================================================================
|
||||
|
||||
Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International
|
||||
Public License
|
||||
|
||||
By exercising the Licensed Rights (defined below), You accept and agree
|
||||
to be bound by the terms and conditions of this Creative Commons
|
||||
Attribution-NonCommercial-ShareAlike 4.0 International Public License
|
||||
("Public License"). To the extent this Public License may be
|
||||
interpreted as a contract, You are granted the Licensed Rights in
|
||||
consideration of Your acceptance of these terms and conditions, and the
|
||||
Licensor grants You such rights in consideration of benefits the
|
||||
Licensor receives from making the Licensed Material available under
|
||||
these terms and conditions.
|
||||
|
||||
|
||||
Section 1 -- Definitions.
|
||||
|
||||
a. Adapted Material means material subject to Copyright and Similar
|
||||
Rights that is derived from or based upon the Licensed Material
|
||||
and in which the Licensed Material is translated, altered,
|
||||
arranged, transformed, or otherwise modified in a manner requiring
|
||||
permission under the Copyright and Similar Rights held by the
|
||||
Licensor. For purposes of this Public License, where the Licensed
|
||||
Material is a musical work, performance, or sound recording,
|
||||
Adapted Material is always produced where the Licensed Material is
|
||||
synched in timed relation with a moving image.
|
||||
|
||||
b. Adapter's License means the license You apply to Your Copyright
|
||||
and Similar Rights in Your contributions to Adapted Material in
|
||||
accordance with the terms and conditions of this Public License.
|
||||
|
||||
c. BY-NC-SA Compatible License means a license listed at
|
||||
creativecommons.org/compatiblelicenses, approved by Creative
|
||||
Commons as essentially the equivalent of this Public License.
|
||||
|
||||
d. Copyright and Similar Rights means copyright and/or similar rights
|
||||
closely related to copyright including, without limitation,
|
||||
performance, broadcast, sound recording, and Sui Generis Database
|
||||
Rights, without regard to how the rights are labeled or
|
||||
categorized. For purposes of this Public License, the rights
|
||||
specified in Section 2(b)(1)-(2) are not Copyright and Similar
|
||||
Rights.
|
||||
|
||||
e. Effective Technological Measures means those measures that, in the
|
||||
absence of proper authority, may not be circumvented under laws
|
||||
fulfilling obligations under Article 11 of the WIPO Copyright
|
||||
Treaty adopted on December 20, 1996, and/or similar international
|
||||
agreements.
|
||||
|
||||
f. Exceptions and Limitations means fair use, fair dealing, and/or
|
||||
any other exception or limitation to Copyright and Similar Rights
|
||||
that applies to Your use of the Licensed Material.
|
||||
|
||||
g. License Elements means the license attributes listed in the name
|
||||
of a Creative Commons Public License. The License Elements of this
|
||||
Public License are Attribution, NonCommercial, and ShareAlike.
|
||||
|
||||
h. Licensed Material means the artistic or literary work, database,
|
||||
or other material to which the Licensor applied this Public
|
||||
License.
|
||||
|
||||
i. Licensed Rights means the rights granted to You subject to the
|
||||
terms and conditions of this Public License, which are limited to
|
||||
all Copyright and Similar Rights that apply to Your use of the
|
||||
Licensed Material and that the Licensor has authority to license.
|
||||
|
||||
j. Licensor means the individual(s) or entity(ies) granting rights
|
||||
under this Public License.
|
||||
|
||||
k. NonCommercial means not primarily intended for or directed towards
|
||||
commercial advantage or monetary compensation. For purposes of
|
||||
this Public License, the exchange of the Licensed Material for
|
||||
other material subject to Copyright and Similar Rights by digital
|
||||
file-sharing or similar means is NonCommercial provided there is
|
||||
no payment of monetary compensation in connection with the
|
||||
exchange.
|
||||
|
||||
l. Share means to provide material to the public by any means or
|
||||
process that requires permission under the Licensed Rights, such
|
||||
as reproduction, public display, public performance, distribution,
|
||||
dissemination, communication, or importation, and to make material
|
||||
available to the public including in ways that members of the
|
||||
public may access the material from a place and at a time
|
||||
individually chosen by them.
|
||||
|
||||
m. Sui Generis Database Rights means rights other than copyright
|
||||
resulting from Directive 96/9/EC of the European Parliament and of
|
||||
the Council of 11 March 1996 on the legal protection of databases,
|
||||
as amended and/or succeeded, as well as other essentially
|
||||
equivalent rights anywhere in the world.
|
||||
|
||||
n. You means the individual or entity exercising the Licensed Rights
|
||||
under this Public License. Your has a corresponding meaning.
|
||||
|
||||
|
||||
Section 2 -- Scope.
|
||||
|
||||
a. License grant.
|
||||
|
||||
1. Subject to the terms and conditions of this Public License,
|
||||
the Licensor hereby grants You a worldwide, royalty-free,
|
||||
non-sublicensable, non-exclusive, irrevocable license to
|
||||
exercise the Licensed Rights in the Licensed Material to:
|
||||
|
||||
a. reproduce and Share the Licensed Material, in whole or
|
||||
in part, for NonCommercial purposes only; and
|
||||
|
||||
b. produce, reproduce, and Share Adapted Material for
|
||||
NonCommercial purposes only.
|
||||
|
||||
2. Exceptions and Limitations. For the avoidance of doubt, where
|
||||
Exceptions and Limitations apply to Your use, this Public
|
||||
License does not apply, and You do not need to comply with
|
||||
its terms and conditions.
|
||||
|
||||
3. Term. The term of this Public License is specified in Section
|
||||
6(a).
|
||||
|
||||
4. Media and formats; technical modifications allowed. The
|
||||
Licensor authorizes You to exercise the Licensed Rights in
|
||||
all media and formats whether now known or hereafter created,
|
||||
and to make technical modifications necessary to do so. The
|
||||
Licensor waives and/or agrees not to assert any right or
|
||||
authority to forbid You from making technical modifications
|
||||
necessary to exercise the Licensed Rights, including
|
||||
technical modifications necessary to circumvent Effective
|
||||
Technological Measures. For purposes of this Public License,
|
||||
simply making modifications authorized by this Section 2(a)
|
||||
(4) never produces Adapted Material.
|
||||
|
||||
5. Downstream recipients.
|
||||
|
||||
a. Offer from the Licensor -- Licensed Material. Every
|
||||
recipient of the Licensed Material automatically
|
||||
receives an offer from the Licensor to exercise the
|
||||
Licensed Rights under the terms and conditions of this
|
||||
Public License.
|
||||
|
||||
b. Additional offer from the Licensor -- Adapted Material.
|
||||
Every recipient of Adapted Material from You
|
||||
automatically receives an offer from the Licensor to
|
||||
exercise the Licensed Rights in the Adapted Material
|
||||
under the conditions of the Adapter's License You apply.
|
||||
|
||||
c. No downstream restrictions. You may not offer or impose
|
||||
any additional or different terms or conditions on, or
|
||||
apply any Effective Technological Measures to, the
|
||||
Licensed Material if doing so restricts exercise of the
|
||||
Licensed Rights by any recipient of the Licensed
|
||||
Material.
|
||||
|
||||
6. No endorsement. Nothing in this Public License constitutes or
|
||||
may be construed as permission to assert or imply that You
|
||||
are, or that Your use of the Licensed Material is, connected
|
||||
with, or sponsored, endorsed, or granted official status by,
|
||||
the Licensor or others designated to receive attribution as
|
||||
provided in Section 3(a)(1)(A)(i).
|
||||
|
||||
b. Other rights.
|
||||
|
||||
1. Moral rights, such as the right of integrity, are not
|
||||
licensed under this Public License, nor are publicity,
|
||||
privacy, and/or other similar personality rights; however, to
|
||||
the extent possible, the Licensor waives and/or agrees not to
|
||||
assert any such rights held by the Licensor to the limited
|
||||
extent necessary to allow You to exercise the Licensed
|
||||
Rights, but not otherwise.
|
||||
|
||||
2. Patent and trademark rights are not licensed under this
|
||||
Public License.
|
||||
|
||||
3. To the extent possible, the Licensor waives any right to
|
||||
collect royalties from You for the exercise of the Licensed
|
||||
Rights, whether directly or through a collecting society
|
||||
under any voluntary or waivable statutory or compulsory
|
||||
licensing scheme. In all other cases the Licensor expressly
|
||||
reserves any right to collect such royalties, including when
|
||||
the Licensed Material is used other than for NonCommercial
|
||||
purposes.
|
||||
|
||||
|
||||
Section 3 -- License Conditions.
|
||||
|
||||
Your exercise of the Licensed Rights is expressly made subject to the
|
||||
following conditions.
|
||||
|
||||
a. Attribution.
|
||||
|
||||
1. If You Share the Licensed Material (including in modified
|
||||
form), You must:
|
||||
|
||||
a. retain the following if it is supplied by the Licensor
|
||||
with the Licensed Material:
|
||||
|
||||
i. identification of the creator(s) of the Licensed
|
||||
Material and any others designated to receive
|
||||
attribution, in any reasonable manner requested by
|
||||
the Licensor (including by pseudonym if
|
||||
designated);
|
||||
|
||||
ii. a copyright notice;
|
||||
|
||||
iii. a notice that refers to this Public License;
|
||||
|
||||
iv. a notice that refers to the disclaimer of
|
||||
warranties;
|
||||
|
||||
v. a URI or hyperlink to the Licensed Material to the
|
||||
extent reasonably practicable;
|
||||
|
||||
b. indicate if You modified the Licensed Material and
|
||||
retain an indication of any previous modifications; and
|
||||
|
||||
c. indicate the Licensed Material is licensed under this
|
||||
Public License, and include the text of, or the URI or
|
||||
hyperlink to, this Public License.
|
||||
|
||||
2. You may satisfy the conditions in Section 3(a)(1) in any
|
||||
reasonable manner based on the medium, means, and context in
|
||||
which You Share the Licensed Material. For example, it may be
|
||||
reasonable to satisfy the conditions by providing a URI or
|
||||
hyperlink to a resource that includes the required
|
||||
information.
|
||||
3. If requested by the Licensor, You must remove any of the
|
||||
information required by Section 3(a)(1)(A) to the extent
|
||||
reasonably practicable.
|
||||
|
||||
b. ShareAlike.
|
||||
|
||||
In addition to the conditions in Section 3(a), if You Share
|
||||
Adapted Material You produce, the following conditions also apply.
|
||||
|
||||
1. The Adapter's License You apply must be a Creative Commons
|
||||
license with the same License Elements, this version or
|
||||
later, or a BY-NC-SA Compatible License.
|
||||
|
||||
2. You must include the text of, or the URI or hyperlink to, the
|
||||
Adapter's License You apply. You may satisfy this condition
|
||||
in any reasonable manner based on the medium, means, and
|
||||
context in which You Share Adapted Material.
|
||||
|
||||
3. You may not offer or impose any additional or different terms
|
||||
or conditions on, or apply any Effective Technological
|
||||
Measures to, Adapted Material that restrict exercise of the
|
||||
rights granted under the Adapter's License You apply.
|
||||
|
||||
|
||||
Section 4 -- Sui Generis Database Rights.
|
||||
|
||||
Where the Licensed Rights include Sui Generis Database Rights that
|
||||
apply to Your use of the Licensed Material:
|
||||
|
||||
a. for the avoidance of doubt, Section 2(a)(1) grants You the right
|
||||
to extract, reuse, reproduce, and Share all or a substantial
|
||||
portion of the contents of the database for NonCommercial purposes
|
||||
only;
|
||||
|
||||
b. if You include all or a substantial portion of the database
|
||||
contents in a database in which You have Sui Generis Database
|
||||
Rights, then the database in which You have Sui Generis Database
|
||||
Rights (but not its individual contents) is Adapted Material,
|
||||
including for purposes of Section 3(b); and
|
||||
|
||||
c. You must comply with the conditions in Section 3(a) if You Share
|
||||
all or a substantial portion of the contents of the database.
|
||||
|
||||
For the avoidance of doubt, this Section 4 supplements and does not
|
||||
replace Your obligations under this Public License where the Licensed
|
||||
Rights include other Copyright and Similar Rights.
|
||||
|
||||
|
||||
Section 5 -- Disclaimer of Warranties and Limitation of Liability.
|
||||
|
||||
a. UNLESS OTHERWISE SEPARATELY UNDERTAKEN BY THE LICENSOR, TO THE
|
||||
EXTENT POSSIBLE, THE LICENSOR OFFERS THE LICENSED MATERIAL AS-IS
|
||||
AND AS-AVAILABLE, AND MAKES NO REPRESENTATIONS OR WARRANTIES OF
|
||||
ANY KIND CONCERNING THE LICENSED MATERIAL, WHETHER EXPRESS,
|
||||
IMPLIED, STATUTORY, OR OTHER. THIS INCLUDES, WITHOUT LIMITATION,
|
||||
WARRANTIES OF TITLE, MERCHANTABILITY, FITNESS FOR A PARTICULAR
|
||||
PURPOSE, NON-INFRINGEMENT, ABSENCE OF LATENT OR OTHER DEFECTS,
|
||||
ACCURACY, OR THE PRESENCE OR ABSENCE OF ERRORS, WHETHER OR NOT
|
||||
KNOWN OR DISCOVERABLE. WHERE DISCLAIMERS OF WARRANTIES ARE NOT
|
||||
ALLOWED IN FULL OR IN PART, THIS DISCLAIMER MAY NOT APPLY TO YOU.
|
||||
|
||||
b. TO THE EXTENT POSSIBLE, IN NO EVENT WILL THE LICENSOR BE LIABLE
|
||||
TO YOU ON ANY LEGAL THEORY (INCLUDING, WITHOUT LIMITATION,
|
||||
NEGLIGENCE) OR OTHERWISE FOR ANY DIRECT, SPECIAL, INDIRECT,
|
||||
INCIDENTAL, CONSEQUENTIAL, PUNITIVE, EXEMPLARY, OR OTHER LOSSES,
|
||||
COSTS, EXPENSES, OR DAMAGES ARISING OUT OF THIS PUBLIC LICENSE OR
|
||||
USE OF THE LICENSED MATERIAL, EVEN IF THE LICENSOR HAS BEEN
|
||||
ADVISED OF THE POSSIBILITY OF SUCH LOSSES, COSTS, EXPENSES, OR
|
||||
DAMAGES. WHERE A LIMITATION OF LIABILITY IS NOT ALLOWED IN FULL OR
|
||||
IN PART, THIS LIMITATION MAY NOT APPLY TO YOU.
|
||||
|
||||
c. The disclaimer of warranties and limitation of liability provided
|
||||
above shall be interpreted in a manner that, to the extent
|
||||
possible, most closely approximates an absolute disclaimer and
|
||||
waiver of all liability.
|
||||
|
||||
|
||||
Section 6 -- Term and Termination.
|
||||
|
||||
a. This Public License applies for the term of the Copyright and
|
||||
Similar Rights licensed here. However, if You fail to comply with
|
||||
this Public License, then Your rights under this Public License
|
||||
terminate automatically.
|
||||
|
||||
b. Where Your right to use the Licensed Material has terminated under
|
||||
Section 6(a), it reinstates:
|
||||
|
||||
1. automatically as of the date the violation is cured, provided
|
||||
it is cured within 30 days of Your discovery of the
|
||||
violation; or
|
||||
|
||||
2. upon express reinstatement by the Licensor.
|
||||
|
||||
For the avoidance of doubt, this Section 6(b) does not affect any
|
||||
right the Licensor may have to seek remedies for Your violations
|
||||
of this Public License.
|
||||
|
||||
c. For the avoidance of doubt, the Licensor may also offer the
|
||||
Licensed Material under separate terms or conditions or stop
|
||||
distributing the Licensed Material at any time; however, doing so
|
||||
will not terminate this Public License.
|
||||
|
||||
d. Sections 1, 5, 6, 7, and 8 survive termination of this Public
|
||||
License.
|
||||
|
||||
|
||||
Section 7 -- Other Terms and Conditions.
|
||||
|
||||
a. The Licensor shall not be bound by any additional or different
|
||||
terms or conditions communicated by You unless expressly agreed.
|
||||
|
||||
b. Any arrangements, understandings, or agreements regarding the
|
||||
Licensed Material not stated herein are separate from and
|
||||
independent of the terms and conditions of this Public License.
|
||||
|
||||
|
||||
Section 8 -- Interpretation.
|
||||
|
||||
a. For the avoidance of doubt, this Public License does not, and
|
||||
shall not be interpreted to, reduce, limit, restrict, or impose
|
||||
conditions on any use of the Licensed Material that could lawfully
|
||||
be made without permission under this Public License.
|
||||
|
||||
b. To the extent possible, if any provision of this Public License is
|
||||
deemed unenforceable, it shall be automatically reformed to the
|
||||
minimum extent necessary to make it enforceable. If the provision
|
||||
cannot be reformed, it shall be severed from this Public License
|
||||
without affecting the enforceability of the remaining terms and
|
||||
conditions.
|
||||
|
||||
c. No term or condition of this Public License will be waived and no
|
||||
failure to comply consented to unless expressly agreed to by the
|
||||
Licensor.
|
||||
|
||||
d. Nothing in this Public License constitutes or may be interpreted
|
||||
as a limitation upon, or waiver of, any privileges and immunities
|
||||
that apply to the Licensor or You, including from the legal
|
||||
processes of any jurisdiction or authority.
|
||||
|
||||
=======================================================================
|
||||
|
||||
Creative Commons is not a party to its public
|
||||
licenses. Notwithstanding, Creative Commons may elect to apply one of
|
||||
its public licenses to material it publishes and in those instances
|
||||
will be considered the “Licensor.” The text of the Creative Commons
|
||||
public licenses is dedicated to the public domain under the CC0 Public
|
||||
Domain Dedication. Except for the limited purpose of indicating that
|
||||
material is shared under a Creative Commons public license or as
|
||||
otherwise permitted by the Creative Commons policies published at
|
||||
creativecommons.org/policies, Creative Commons does not authorize the
|
||||
use of the trademark "Creative Commons" or any other trademark or logo
|
||||
of Creative Commons without its prior written consent including,
|
||||
without limitation, in connection with any unauthorized modifications
|
||||
to any of its public licenses or any other arrangements,
|
||||
understandings, or agreements concerning use of licensed material. For
|
||||
the avoidance of doubt, this paragraph does not form part of the
|
||||
public licenses.
|
||||
|
||||
Creative Commons may be contacted at creativecommons.org.
|
2
LicenseSystem/index.js
Normal file
2
LicenseSystem/index.js
Normal file
|
@ -0,0 +1,2 @@
|
|||
const _0x11e2b8=_0x4ef1;function _0x4ef1(_0x25c410,_0x358db5){const _0x29a8e6=_0x29a8();return _0x4ef1=function(_0x4ef111,_0x4067f7){_0x4ef111=_0x4ef111-0x14c;let _0x4796d9=_0x29a8e6[_0x4ef111];return _0x4796d9;},_0x4ef1(_0x25c410,_0x358db5);}function _0x29a8(){const _0x77775c=['2904430BCDYuv','9ceBTKm','731330jzyhAt','1137520MhgEHw','376cQveQv','POST','Discord-dashboard\x20License\x20ID\x20is\x20not\x20valid.','licenseId','ValidateLicense','node-fetch','https://licenses.assistantscenter.com/validate','176xSOpMV','466968xQLxio','6Uyrrnc','Discord-Dashboard\x20License\x20ID\x20is\x20not\x20defined!\x20Go\x20to\x20https://licenses.assistants.ga/\x20and\x20generate/buy\x20one\x20for\x20you.','35756yBzcPP','then','304918QtlMjH','exports','170581JeXJRS','6QtFvLt'];_0x29a8=function(){return _0x77775c;};return _0x29a8();}(function(_0x576ba9,_0x27eee6){const _0x45302b=_0x4ef1,_0x59594e=_0x576ba9();while(!![]){try{const _0x43186f=parseInt(_0x45302b(0x150))/0x1+parseInt(_0x45302b(0x14e))/0x2*(-parseInt(_0x45302b(0x15f))/0x3)+parseInt(_0x45302b(0x155))/0x4+parseInt(_0x45302b(0x154))/0x5*(parseInt(_0x45302b(0x151))/0x6)+parseInt(_0x45302b(0x14c))/0x7*(parseInt(_0x45302b(0x156))/0x8)+-parseInt(_0x45302b(0x153))/0x9*(-parseInt(_0x45302b(0x152))/0xa)+-parseInt(_0x45302b(0x15d))/0xb*(parseInt(_0x45302b(0x15e))/0xc);if(_0x43186f===_0x27eee6)break;else _0x59594e['push'](_0x59594e['shift']());}catch(_0x59c126){_0x59594e['push'](_0x59594e['shift']());}}}(_0x29a8,0x31dac));const fetch=require(_0x11e2b8(0x15b));class License{constructor(_0x1dec84){const _0x31f658=_0x11e2b8;if(!_0x1dec84)throw new TypeError(_0x31f658(0x160));this[_0x31f658(0x159)]=_0x1dec84;}async[_0x11e2b8(0x15a)](){const _0x115f1c=_0x11e2b8;let _0x469e87;return await fetch(_0x115f1c(0x15c),{'method':_0x115f1c(0x157),'headers':{'Content-Type':'application/json'},'body':JSON['stringify']({'licenseId':this[_0x115f1c(0x159)]})})[_0x115f1c(0x14d)](_0x264618=>_0x264618['json']())[_0x115f1c(0x14d)](_0x1adf24=>{const _0x2b7298=_0x115f1c;if(_0x1adf24['error'])throw new TypeError(_0x2b7298(0x158));_0x469e87=_0x1adf24;}),_0x469e87;}}module[_0x11e2b8(0x14f)]=License;
|
||||
module.exports = License;
|
23
ModuleExportsFunctions/customPagesTypes.js
Normal file
23
ModuleExportsFunctions/customPagesTypes.js
Normal file
|
@ -0,0 +1,23 @@
|
|||
module.exports = {
|
||||
redirectToUrl: (endpoint, getDataFunction) => {
|
||||
return {
|
||||
type: "redirect",
|
||||
endpoint: endpoint,
|
||||
getEndpoint: getDataFunction,
|
||||
}
|
||||
},
|
||||
renderHtml: (endpoint, getDataFunction) => {
|
||||
return {
|
||||
type: "html",
|
||||
endpoint: endpoint,
|
||||
getHtml: getDataFunction,
|
||||
}
|
||||
},
|
||||
sendJson: (endpoint, getDataFunction) => {
|
||||
return {
|
||||
type: "json",
|
||||
endpoint: endpoint,
|
||||
getJson: getDataFunction,
|
||||
}
|
||||
},
|
||||
}
|
43
ModuleExportsFunctions/discordPermissions.js
Normal file
43
ModuleExportsFunctions/discordPermissions.js
Normal file
|
@ -0,0 +1,43 @@
|
|||
module.exports = {
|
||||
CREATE_INSTANT_INVITE: ["CREATE_INSTANT_INVITE", 0x1],
|
||||
KICK_MEMBERS: ["KICK_MEMBERS", 0x2],
|
||||
BAN_MEMBERS: ["BAN_MEMBERS", 0x4],
|
||||
ADMINISTRATOR: ["ADMINISTRATOR", 0x8],
|
||||
MANAGE_CHANNELS: ["MANAGE_CHANNELS", 0x10],
|
||||
MANAGE_GUILD: ["MANAGE_GUILD", 0x20],
|
||||
ADD_REACTIONS: ["ADD_REACTIONS", 0x40],
|
||||
VIEW_AUDIT_LOG: ["VIEW_AUDIT_LOG", 0x80],
|
||||
PRIORITY_SPEAKER: ["PRIORITY_SPEAKER", 0x100],
|
||||
STREAM: ["STREAM", 0x200],
|
||||
VIEW_CHANNEL: ["VIEW_CHANNEL", 0x400],
|
||||
SEND_MESSAGES: ["SEND_MESSAGES", 0x800],
|
||||
SEND_TTS_MESSAGES: ["SEND_TTS_MESSAGES", 0x1000],
|
||||
MANAGE_MESSAGES: ["MANAGE_MESSAGES", 0x2000],
|
||||
EMBED_LINKS: ["EMBED_LINKS", 0x4000],
|
||||
ATTACH_FILES: ["ATTACH_FILES", 0x8000],
|
||||
READ_MESSAGE_HISTORY: ["READ_MESSAGE_HISTORY", 0x10000],
|
||||
MENTION_EVERYONE: ["MENTION_EVERYONE", 0x20000],
|
||||
USE_EXTERNAL_EMOJIS: ["USE_EXTERNAL_EMOJIS", 0x40000],
|
||||
VIEW_GUILD_INSIGHTS: ["VIEW_GUILD_INSIGHTS", 0x80000],
|
||||
CONNECT: ["CONNECT", 0x100000],
|
||||
SPEAK: ["SPEAK", 0x200000],
|
||||
MUTE_MEMBERS: ["MUTE_MEMBERS", 0x400000],
|
||||
DEAFEN_MEMBERS: ["DEAFEN_MEMBERS", 0x800000],
|
||||
MOVE_MEMBERS: ["MOVE_MEMBERS", 0x1000000],
|
||||
USE_VAD: ["USE_VAD", 0x2000000],
|
||||
CHANGE_NICKNAME: ["CHANGE_NICKNAME", 0x4000000],
|
||||
MANAGE_NICKNAMES: ["MANAGE_NICKNAMES", 0x8000000],
|
||||
MANAGE_ROLES: ["MANAGE_ROLES", 0x10000000],
|
||||
MANAGE_WEBHOOKS: ["MANAGE_WEBHOOKS", 0x20000000],
|
||||
MANAGE_EMOJIS_AND_STICKERS: ["MANAGE_EMOJIS_AND_STICKERS", 0x40000000],
|
||||
USE_APPLICATION_COMMANDS: ["USE_APPLICATION_COMMANDS", 0x80000000],
|
||||
REQUEST_TO_SPEAK: ["REQUEST_TO_SPEAK", 0x100000000],
|
||||
MANAGE_EVENTS: ["MANAGE_EVENTS", 0x200000000],
|
||||
MANAGE_THREADS: ["MANAGE_THREADS", 0x400000000],
|
||||
CREATE_PUBLIC_THREADS: ["CREATE_PUBLIC_THREADS", 0x800000000],
|
||||
CREATE_PRIVATE_THREADS: ["CREATE_PRIVATE_THREADS", 0x1000000000],
|
||||
USE_EXTERNAL_STICKERS: ["USE_EXTERNAL_STICKERS", 0x2000000000],
|
||||
SEND_MESSAGES_IN_THREADS: ["SEND_MESSAGES_IN_THREADS", 0x4000000000],
|
||||
START_EMBEDDED_ACTIVITIES: ["START_EMBEDDED_ACTIVITIES", 0x8000000000],
|
||||
MODERATE_MEMBERS: ["MODERATE_MEMBERS", 0x10000000000],
|
||||
}
|
346
ModuleExportsFunctions/formTypes.js
Normal file
346
ModuleExportsFunctions/formTypes.js
Normal file
|
@ -0,0 +1,346 @@
|
|||
const discordPermissions = require("./discordPermissions")
|
||||
|
||||
module.exports = {
|
||||
select: (list, disabled, themeOptions = {}) => {
|
||||
if (!list)
|
||||
throw new Error(
|
||||
err("List in the 'select' form type cannot be empty.")
|
||||
)
|
||||
if (typeof list != "object")
|
||||
throw new Error(
|
||||
err("List in the 'select' form type should be an JSON object.")
|
||||
)
|
||||
let keys = Object.keys(list)
|
||||
let values = Object.values(list)
|
||||
return {
|
||||
type: "select",
|
||||
data: {
|
||||
keys,
|
||||
values,
|
||||
},
|
||||
disabled: disabled || false,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
multiSelect: (list, disabled, required, themeOptions = {}) => {
|
||||
if (!list)
|
||||
throw new Error(
|
||||
err("List in the 'select' form type cannot be empty.")
|
||||
)
|
||||
if (typeof list != "object")
|
||||
throw new Error(
|
||||
err("List in the 'select' form type should be an JSON object.")
|
||||
)
|
||||
let keys = Object.keys(list)
|
||||
let values = Object.values(list)
|
||||
return {
|
||||
type: "multiSelect",
|
||||
data: {
|
||||
keys,
|
||||
values,
|
||||
},
|
||||
disabled: disabled || false,
|
||||
required: required || false,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
input: (placeholder, min, max, disabled, required, themeOptions = {}) => {
|
||||
if (min) {
|
||||
if (isNaN(min))
|
||||
throw new Error(
|
||||
err("'min' in the 'input' form type should be an number.")
|
||||
)
|
||||
}
|
||||
if (max) {
|
||||
if (isNaN(max))
|
||||
throw new Error(
|
||||
err("'max' in the 'input' form type should be an number.")
|
||||
)
|
||||
}
|
||||
if (min && max) {
|
||||
if (min > max)
|
||||
throw new Error(
|
||||
err(
|
||||
"'min' in the 'input' form type cannot be higher than 'max'."
|
||||
)
|
||||
)
|
||||
}
|
||||
return {
|
||||
type: "input",
|
||||
data: placeholder,
|
||||
min: min || null,
|
||||
max: max || null,
|
||||
disabled: disabled || false,
|
||||
required: required || false,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
textarea: (
|
||||
placeholder,
|
||||
min,
|
||||
max,
|
||||
disabled,
|
||||
required,
|
||||
themeOptions = {}
|
||||
) => {
|
||||
if (min) {
|
||||
if (isNaN(min))
|
||||
throw new Error(
|
||||
err("'min' in the 'input' form type should be an number.")
|
||||
)
|
||||
}
|
||||
if (max) {
|
||||
if (isNaN(max))
|
||||
throw new Error(
|
||||
err("'max' in the 'input' form type should be an number.")
|
||||
)
|
||||
}
|
||||
if (min && max) {
|
||||
if (min > max)
|
||||
throw new Error(
|
||||
err(
|
||||
"'min' in the 'input' form type cannot be higher than 'max'."
|
||||
)
|
||||
)
|
||||
}
|
||||
return {
|
||||
type: "textarea",
|
||||
data: placeholder,
|
||||
min: min || null,
|
||||
max: max || null,
|
||||
disabled: disabled || false,
|
||||
required: required || false,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
switch: (disabled, themeOptions = {}) => {
|
||||
return {
|
||||
type: "switch",
|
||||
disabled: disabled,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
checkbox: (disabled, themeOptions = {}) => {
|
||||
return {
|
||||
type: "checkbox",
|
||||
disabled: disabled,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
channelsSelect: (
|
||||
disabled,
|
||||
channelTypes = ["GUILD_TEXT"],
|
||||
hideNSFW,
|
||||
onlyNSFW,
|
||||
hideNoAccess,
|
||||
themeOptions = {}
|
||||
) => {
|
||||
return {
|
||||
type: "channelsSelect",
|
||||
function: (client, guildid, userid) => {
|
||||
let listCount = {}
|
||||
let list = {
|
||||
"-": "",
|
||||
}
|
||||
const guild = client.guilds.cache.get(guildid)
|
||||
const user = guild.members.cache.get(userid)
|
||||
const bot = guild.members.cache.get(client.user.id)
|
||||
client.guilds.cache
|
||||
.get(guildid)
|
||||
.channels.cache.forEach((channel) => {
|
||||
if (!channelTypes.includes(channel.type)) return
|
||||
if (hideNSFW && channel.nsfw) return
|
||||
if (onlyNSFW && !channel.nsfw) return
|
||||
if (hideNoAccess) {
|
||||
if (!user.permissionsIn(channel).has('0x800') || !user.permissionsIn(channel).has('0x400')) return
|
||||
if (!bot.permissionsIn(channel).has('0x800') || !bot.permissionsIn(channel).has('0x800')) return
|
||||
}
|
||||
listCount[channel.name]
|
||||
? (listCount[channel.name] =
|
||||
listCount[channel.name] + 1)
|
||||
: (listCount[channel.name] = 1)
|
||||
if (list[channel.name])
|
||||
list[
|
||||
`${channel.name} (${listCount[channel.name]})`
|
||||
] = channel.id
|
||||
else list[channel.name] = channel.id
|
||||
})
|
||||
|
||||
let myObj = list
|
||||
let keys = Object.keys(myObj),
|
||||
i = null,
|
||||
len = keys.length
|
||||
|
||||
keys.sort()
|
||||
list = {}
|
||||
|
||||
for (i = 0; i < len; i++) {
|
||||
k = keys[i]
|
||||
list[k] = myObj[k]
|
||||
}
|
||||
|
||||
return {
|
||||
values: Object.values(list),
|
||||
keys: Object.keys(list),
|
||||
}
|
||||
},
|
||||
disabled,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
channelsMultiSelect: (
|
||||
disabled,
|
||||
required,
|
||||
channelTypes = ["GUILD_TEXT"],
|
||||
hideNSFW,
|
||||
onlyNSFW,
|
||||
hideNoAccess,
|
||||
themeOptions = {}
|
||||
) => {
|
||||
return {
|
||||
type: "channelsMultiSelect",
|
||||
function: (client, guildid, userid) => {
|
||||
let listCount = {}
|
||||
let list = {}
|
||||
const guild = client.guilds.cache.get(guildid)
|
||||
const user = guild.members.cache.get(userid)
|
||||
const bot = guild.members.cache.get(client.user.id)
|
||||
client.guilds.cache
|
||||
.get(guildid)
|
||||
.channels.cache.forEach((channel) => {
|
||||
if (!channelTypes.includes(channel.type)) return
|
||||
if (hideNSFW && channel.nsfw) return
|
||||
if (onlyNSFW && !channel.nsfw) return
|
||||
if (hideNoAccess) {
|
||||
if (!user.permissionsIn(channel).has('0x800') || !user.permissionsIn(channel).has('0x400')) return
|
||||
if (!bot.permissionsIn(channel).has('0x800') || !bot.permissionsIn(channel).has('0x800')) return
|
||||
}
|
||||
listCount[channel.name]
|
||||
? (listCount[channel.name] =
|
||||
listCount[channel.name] + 1)
|
||||
: (listCount[channel.name] = 1)
|
||||
if (list[channel.name])
|
||||
list[
|
||||
`${channel.name} (${listCount[channel.name]})`
|
||||
] = channel.id
|
||||
else list[channel.name] = channel.id
|
||||
})
|
||||
|
||||
let myObj = list
|
||||
let keys = Object.keys(myObj),
|
||||
i = null,
|
||||
len = keys.length
|
||||
|
||||
keys.sort()
|
||||
list = {}
|
||||
|
||||
for (i = 0; i < len; i++) {
|
||||
k = keys[i]
|
||||
list[k] = myObj[k]
|
||||
}
|
||||
|
||||
return {
|
||||
values: Object.values(list),
|
||||
keys: Object.keys(list),
|
||||
}
|
||||
},
|
||||
disabled,
|
||||
required,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
rolesMultiSelect: (disabled, required, includeBots, hideHigherRoles, themeOptions = {}) => {
|
||||
return {
|
||||
type: "rolesMultiSelect",
|
||||
function: (client, guildid, userid) => {
|
||||
let listCount = {}
|
||||
const list = []
|
||||
const guild = client.guilds.cache.get(guildid)
|
||||
const user = guild.members.cache.get(userid)
|
||||
const bot = guild.members.cache.get(client.user.id)
|
||||
|
||||
client.guilds.cache.get(guildid).roles.cache.forEach((role) => {
|
||||
if (role.managed && !includeBots) return
|
||||
if (role.id === guildid) return // @everyone role
|
||||
if (hideHigherRoles) {
|
||||
if (role.position >= user.roles.highest.position) return
|
||||
if (role.position >= bot.roles.highest.position) return
|
||||
}
|
||||
listCount[role.name]
|
||||
? (listCount[role.name] = listCount[role.name] + 1)
|
||||
: (listCount[role.name] = 1)
|
||||
if (listCount[role.name] > 1)
|
||||
list.push({ key: `${role.name} (${listCount[role.name]})`, value: role.id, position: role.position })
|
||||
else list.push({ key: role.name, value: role.id, position: role.position })
|
||||
})
|
||||
|
||||
list.sort((a, b) => b.position - a.position)
|
||||
|
||||
const sortedList = {}
|
||||
list.forEach(({ key, value }) => (sortedList[key] = value))
|
||||
|
||||
return {
|
||||
values: Object.values(sortedList),
|
||||
keys: Object.keys(sortedList),
|
||||
}
|
||||
},
|
||||
disabled,
|
||||
required,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
rolesSelect: (disabled, includeBots, hideHigherRoles, themeOptions = {}) => {
|
||||
return {
|
||||
type: "rolesSelect",
|
||||
function: (client, guildid, userid) => {
|
||||
let listCount = {}
|
||||
const list = [{ key: '-', value: '' }]
|
||||
const guild = client.guilds.cache.get(guildid)
|
||||
const user = guild.members.cache.get(userid)
|
||||
const bot = guild.members.cache.get(client.user.id)
|
||||
client.guilds.cache.get(guildid).roles.cache.forEach((role) => {
|
||||
if (role.managed && !includeBots) return
|
||||
if (role.id === guildid) return // @everyone role
|
||||
if (hideHigherRoles) {
|
||||
if (role.position >= user.roles.highest.position) return
|
||||
if (role.position >= bot.roles.highest.position) return
|
||||
}
|
||||
listCount[role.name]
|
||||
? (listCount[role.name] = listCount[role.name] + 1)
|
||||
: (listCount[role.name] = 1)
|
||||
if (listCount[role.name] > 1)
|
||||
list.push({ key: `${role.name} (${listCount[role.name]})`, value: role.id, position: role.position })
|
||||
else list.push({ key: role.name, value: role.id, position: role.position })
|
||||
})
|
||||
|
||||
list.sort((a, b) => b.position - a.position)
|
||||
|
||||
const sortedList = {}
|
||||
list.forEach(({ key, value }) => (sortedList[key] = value))
|
||||
|
||||
return {
|
||||
values: Object.values(sortedList),
|
||||
keys: Object.keys(sortedList),
|
||||
}
|
||||
},
|
||||
disabled,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
colorSelect: (defaultState, disabled, themeOptions = {}) => {
|
||||
return {
|
||||
type: "colorSelect",
|
||||
data: defaultState,
|
||||
disabled,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
embedBuilder: (defaultSettings, disabled, themeOptions = {}) => {
|
||||
return {
|
||||
type: "embedBuilder",
|
||||
data: defaultSettings,
|
||||
disabled,
|
||||
themeOptions,
|
||||
}
|
||||
},
|
||||
}
|
74
README.MD
Normal file
74
README.MD
Normal file
|
@ -0,0 +1,74 @@
|
|||
<img src="https://cdn.assistantscenter.com/kxlrosdn">
|
||||
|
||||
<p align="center"><img src="https://nodei.co/npm/discord-dashboard.png"></p>
|
||||
<p align="center"><img src="https://img.shields.io/npm/v/discord-dashboard"> <img src="https://img.shields.io/github/repo-size/breftejk/Discord.js-Web-Dashboard"> <img src="https://img.shields.io/badge/License-CC%20BY--NC--SA%204.0-green.svg"> <img src="https://img.shields.io/github/contributors/breftejk/Discord.js-Web-Dashboard?color=green"> <img src="https://img.shields.io/badge/node.js-16.6.0+-blue"> <img src="https://img.shields.io/github/package-json/dependency-version/breftejk/Discord.js-Web-Dashboard/discord.js"> <img src="https://img.shields.io/github/package-json/dependency-version/breftejk/Discord.js-Web-Dashboard/express"> </p>
|
||||
<p align="center"><a href="https://discord.gg/Nkc8MWxHRD"> <img src="https://discordapp.com/api/guilds/803034737261936670/widget.png" alt="Discord server"/></a></p>
|
||||
|
||||
# Discord-Dashboard
|
||||
|
||||
- Easier creation of a Dashboard for Bot using Node.js
|
||||
- No need to know how the API works, no need to write it
|
||||
- Easy configuration
|
||||
- Community support
|
||||
- Free for no-profit usage
|
||||
- Paid for profit usage
|
||||
|
||||
# Install
|
||||
|
||||
```js
|
||||
npm i discord-dashboard
|
||||
//or
|
||||
yarn add discord-dashboard
|
||||
```
|
||||
|
||||
# Get help
|
||||
|
||||
Join our Discord Support Server
|
||||
|
||||
<a href="https://discord.gg/Nkc8MWxHRD"> <img src="https://discordapp.com/api/guilds/803034737261936670/widget.png" alt="Discord server"/></a>
|
||||
|
||||
# Documentation
|
||||
|
||||
**Tutorial:**
|
||||
- **#1** [Introduction](https://learnit.assistantscenter.com/discord-dashboard-tutorial-1-introduction/)
|
||||
- **#2** [Setting up a project](https://learnit.assistantscenter.com/discord-dashboard-tutorial-2-setting-up-a-project/)
|
||||
- ***#2.1.*** [Let's make it work](https://learnit.assistantscenter.com/discord-dashboard-tutorial-2-1-lets-make-it-work/)
|
||||
- ***#2.2.*** [Creating custom options](https://learnit.assistantscenter.com/discord-dashboard-tutorial-2-2-creating-custom-options/)
|
||||
- ***#2.3.*** [FormTypes](https://learnit.assistantscenter.com/discord-dashboard-tutorial-2-3-formtypes/)
|
||||
- ***#2.4.*** [Theme customization](https://learnit.assistantscenter.com/discord-dashboard-tutorial-2-3-theme-customization/)
|
||||
- **#3** [How to...? ](https://learnit.assistantscenter.com/discord-dashboard-tutorial-3-how-to/)
|
||||
- ***#3.1.*** [Connect domain with the dashboard](https://learnit.assistantscenter.com/discord-dashboard-tutorial-3-1-connect-domain-with-the-dashboard/)
|
||||
- ***#2.2.*** [Make dashboard HTTPS (SSL)](https://learnit.assistantscenter.com/discord-dashboard-tutorial-3-2-make-dashboard-https-ssl/)
|
||||
|
||||
Straight-written DOCS can be found at the URL: https://dbd-docs.assistantscenter.com/
|
||||
|
||||
We'd prefer using Tutorial 😅
|
||||
|
||||
# Contact
|
||||
|
||||
- **E-mail**: contact@assistantscenter.com, support@assistantscenter.com, billing@assistantscenter.com
|
||||
- **Discord**: https://discord.gg/Nkc8MWxHRD
|
||||
|
||||
# Licenses
|
||||
|
||||
The project is covered by a **Shareware (CC BY-NC-SA 4.0) licence**. This means that the project is theoretically free and limited, but it has a paid versions.
|
||||
|
||||
### What types of shareware license does the discord-dashboard project fall under?
|
||||
|
||||
- **OpenSource License**: Free. Personal usage only. Non-profit.
|
||||
|
||||
**For profit usage**: Please contact use us to resell rights. Email: *contact@assistantscenter.com*
|
||||
|
||||
#### You can generate OpenSource License there: https://licenses.assistantscenter.com/
|
||||
|
||||
# v3 Info
|
||||
|
||||
The v3 version of the project designed using solutions like Fastify+Next.js is coming!
|
||||
|
||||
### **Stay tuned:**
|
||||
|
||||
**Discord v3 Updates**: https://discord.gg/Nkc8MWxHRD
|
||||
|
||||
<hr>
|
||||
|
||||
© 2020-2023 Assistants Center, All rights reserved
|
888
Routes/dashboard.js
Normal file
888
Routes/dashboard.js
Normal file
|
@ -0,0 +1,888 @@
|
|||
const { PermissionFlagsBits } = require("discord.js")
|
||||
const Discord = require("discord.js")
|
||||
const router = require("express").Router()
|
||||
|
||||
module.exports = (app, config, themeConfig) => {
|
||||
const RL = require("express-rate-limit")
|
||||
const RateLimits = config.rateLimits || {}
|
||||
let RateFunctions = {}
|
||||
|
||||
const NoRL = (req, res, next) => next()
|
||||
|
||||
if (RateLimits.manage) {
|
||||
RateFunctions.manage = RL.rateLimit({
|
||||
windowMs: RateLimits.manage.windowMs,
|
||||
max: RateLimits.manage.max,
|
||||
message: RateLimits.manage.message,
|
||||
store: RateLimits.manage.store || new RL.MemoryStore(),
|
||||
})
|
||||
}
|
||||
|
||||
if (RateLimits.guildPage) {
|
||||
RateFunctions.guildPage = RL.rateLimit({
|
||||
windowMs: RateLimits.guildPage.windowMs,
|
||||
max: RateLimits.guildPage.max,
|
||||
message: RateLimits.guildPage.message,
|
||||
store: RateLimits.guildPage.store || new RL.MemoryStore(),
|
||||
})
|
||||
}
|
||||
|
||||
if (RateLimits.settingsUpdatePostAPI) {
|
||||
RateFunctions.settingsUpdatePostAPI = RL.rateLimit({
|
||||
windowMs: RateLimits.settingsUpdatePostAPI.windowMs,
|
||||
max: RateLimits.settingsUpdatePostAPI.max,
|
||||
message: RateLimits.settingsUpdatePostAPI.message,
|
||||
store:
|
||||
RateLimits.settingsUpdatePostAPI.store || new RL.MemoryStore(),
|
||||
})
|
||||
}
|
||||
|
||||
router.get(
|
||||
"/manage",
|
||||
RateFunctions.manage ? RateFunctions.manage : NoRL,
|
||||
async (req, res) => {
|
||||
if (!req.session.user) return res.redirect("/discord?r=/manage")
|
||||
let customThemeOptions
|
||||
if (themeConfig?.customThemeOptions?.manage) {
|
||||
customThemeOptions =
|
||||
await themeConfig.customThemeOptions.manage({
|
||||
req: req,
|
||||
res: res,
|
||||
config: config,
|
||||
})
|
||||
}
|
||||
res.render("guilds", {
|
||||
req: req,
|
||||
bot: config.bot,
|
||||
themeConfig: req.themeConfig,
|
||||
customThemeOptions: customThemeOptions || {},
|
||||
config,
|
||||
})
|
||||
}
|
||||
)
|
||||
|
||||
router.get(
|
||||
"/guild/:id",
|
||||
RateFunctions.guildPage ? RateFunctions.guildPage : NoRL,
|
||||
async (req, res) => {
|
||||
res.redirect("/settings/" + req.params.id)
|
||||
|
||||
if (!req.session.user)
|
||||
return res.redirect("/discord?r=/guild/" + req.params.id)
|
||||
let customThemeOptions
|
||||
if (themeConfig?.customThemeOptions?.getGuild) {
|
||||
customThemeOptions =
|
||||
await themeConfig.customThemeOptions.getGuild({
|
||||
req: req,
|
||||
res: res,
|
||||
config: config,
|
||||
guildId: req.params.id,
|
||||
})
|
||||
}
|
||||
let bot = config.bot
|
||||
if (!bot.guilds.cache.get(req.params.id)) {
|
||||
try {
|
||||
await bot.guilds.fetch(req.params.id)
|
||||
} catch (err) { }
|
||||
}
|
||||
|
||||
if (!bot.guilds.cache.get(req.params.id))
|
||||
return res.redirect("/manage?error=noPermsToManageGuild")
|
||||
if (
|
||||
!bot.guilds.cache
|
||||
.get(req.params.id)
|
||||
.members.cache.get(req.session.user.id)
|
||||
) {
|
||||
try {
|
||||
await bot.guilds.cache
|
||||
.get(req.params.id)
|
||||
.members.fetch(req.session.user.id)
|
||||
} catch (err) { }
|
||||
}
|
||||
for (let PermissionRequired of req.requiredPermissions) {
|
||||
let converted = PermissionRequired[0]
|
||||
const DiscordJsVersion = Discord.version.split(".")[0]
|
||||
|
||||
if (DiscordJsVersion === "14")
|
||||
converted = convert14(PermissionRequired[0])
|
||||
|
||||
if (
|
||||
!bot.guilds.cache
|
||||
.get(req.params.id)
|
||||
.members.cache.get(req.session.user.id)
|
||||
.permissions.has(converted)
|
||||
)
|
||||
return res.redirect("/manage?error=noPermsToManageGuild")
|
||||
}
|
||||
|
||||
if (bot.guilds.cache.get(req.params.id).channels.cache.size < 1) {
|
||||
try {
|
||||
await bot.guilds.cache.get(req.params.id).channels.fetch()
|
||||
} catch (err) { }
|
||||
}
|
||||
|
||||
if (bot.guilds.cache.get(req.params.id).roles.cache.size < 2) {
|
||||
try {
|
||||
await bot.guilds.cache.get(req.params.id).roles.fetch()
|
||||
} catch (err) { }
|
||||
}
|
||||
|
||||
let actual = {}
|
||||
|
||||
let canUseList = {}
|
||||
if (!config.useCategorySet)
|
||||
for (const s of config.settings) {
|
||||
if (!canUseList[s.categoryId]) canUseList[s.categoryId] = {}
|
||||
for (const c of s.categoryOptionsList) {
|
||||
if (c.allowedCheck) {
|
||||
const canUse = await c.allowedCheck({
|
||||
guild: { id: req.params.id },
|
||||
user: { id: req.session.user.id },
|
||||
})
|
||||
if (typeof canUse != "object")
|
||||
throw new TypeError(
|
||||
`${s.categoryId} category option with id ${c.optionId} allowedCheck function need to return {allowed: Boolean, errorMessage: String | null}`
|
||||
)
|
||||
canUseList[s.categoryId][c.optionId] = canUse
|
||||
} else {
|
||||
canUseList[s.categoryId][c.optionId] = {
|
||||
allowed: true,
|
||||
errorMessage: null,
|
||||
}
|
||||
}
|
||||
|
||||
if (c.optionType == "spacer") {
|
||||
} else {
|
||||
if (!actual[s.categoryId]) {
|
||||
actual[s.categoryId] = {}
|
||||
}
|
||||
if (!actual[s.categoryId][c.optionId]) {
|
||||
actual[s.categoryId][c.optionId] =
|
||||
await c.getActualSet({
|
||||
guild: {
|
||||
id: req.params.id,
|
||||
object: bot.guilds.cache.get(
|
||||
req.params.id
|
||||
),
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: bot.guilds.cache
|
||||
.get(req.params.id)
|
||||
.members.cache.get(
|
||||
req.session.user.id
|
||||
),
|
||||
},
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
for (const category of config.settings) {
|
||||
if (!canUseList[category.categoryId])
|
||||
canUseList[category.categoryId] = {}
|
||||
const catGAS = await category.getActualSet({
|
||||
guild: {
|
||||
id: req.params.id,
|
||||
object: bot.guilds.cache.get(req.params.id),
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: bot.guilds.cache
|
||||
.get(req.params.id)
|
||||
.members.cache.get(req.session.user.id),
|
||||
},
|
||||
})
|
||||
|
||||
for (const o of catGAS) {
|
||||
if (!o || !o?.optionId)
|
||||
console.log(
|
||||
"WARNING: You haven't set the optionId for a category option in your config. This is required for the category option to work."
|
||||
)
|
||||
else {
|
||||
const option = category.categoryOptionsList.find(
|
||||
(c) => c.optionId == o.optionId
|
||||
)
|
||||
if (option) {
|
||||
if (option.allowedCheck) {
|
||||
const canUse = await option.allowedCheck({
|
||||
guild: {
|
||||
id: req.params.id,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
},
|
||||
})
|
||||
if (typeof canUse != "object")
|
||||
throw new TypeError(
|
||||
`${category.categoryId} category option with id ${option.optionId} allowedCheck function need to return {allowed: Boolean, errorMessage: String | null}`
|
||||
)
|
||||
canUseList[category.categoryId][
|
||||
option.optionId
|
||||
] = canUse
|
||||
} else {
|
||||
canUseList[category.categoryId][
|
||||
option.optionId
|
||||
] = {
|
||||
allowed: true,
|
||||
errorMessage: null,
|
||||
}
|
||||
}
|
||||
|
||||
if (option.optionType !== "spacer") {
|
||||
if (!actual[category.categoryId]) {
|
||||
actual[category.categoryId] = {}
|
||||
}
|
||||
if (
|
||||
!actual[category.categoryId][
|
||||
option.optionId
|
||||
]
|
||||
) {
|
||||
actual[category.categoryId][
|
||||
option.optionId
|
||||
] = o.data
|
||||
}
|
||||
}
|
||||
} else
|
||||
console.log(
|
||||
`WARNING: Option ${o.optionId} in category ${category.categoryId} doesn't exist in your config.`
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
let errors
|
||||
let success
|
||||
|
||||
if (req.session.errors) {
|
||||
if (String(req.session.errors).includes("%is%")) {
|
||||
errors = req.session.errors.split("%and%")
|
||||
}
|
||||
}
|
||||
|
||||
if (req.session.success) {
|
||||
if (typeof req.session.success == "boolean") {
|
||||
success = true
|
||||
} else {
|
||||
if (String(req.session.success).includes("%is%")) {
|
||||
success = req.session.success.split("%and%")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
req.session.errors = null
|
||||
req.session.success = null
|
||||
|
||||
res.render("guild", {
|
||||
successes: success,
|
||||
errors: errors,
|
||||
settings: config.settings,
|
||||
actual: actual,
|
||||
canUseList,
|
||||
bot: config.bot,
|
||||
req: req,
|
||||
guildid: req.params.id,
|
||||
themeConfig: req.themeConfig,
|
||||
customThemeOptions: customThemeOptions || {},
|
||||
config,
|
||||
})
|
||||
}
|
||||
)
|
||||
|
||||
router.post(
|
||||
"/settings/update/:guildId/:categoryId",
|
||||
RateFunctions.settingsUpdatePostAPI
|
||||
? RateFunctions.settingsUpdatePostAPI
|
||||
: NoRL,
|
||||
async (req, res) => {
|
||||
if (!req.session.user)
|
||||
return res.redirect("/discord?r=/guild/" + req.params.guildId)
|
||||
|
||||
let customThemeOptions
|
||||
if (themeConfig?.customThemeOptions?.settingsUpdate) {
|
||||
customThemeOptions =
|
||||
await themeConfig.customThemeOptions.settingsUpdate({
|
||||
req: req,
|
||||
config: config,
|
||||
guildId: req.params.id,
|
||||
categoryId: req.params.categoryId,
|
||||
})
|
||||
}
|
||||
let bot = config.bot
|
||||
|
||||
if (!bot.guilds.cache.get(req.params.guildId))
|
||||
return res.redirect("/manage?error=noPermsToManageGuild")
|
||||
await bot.guilds.cache
|
||||
.get(req.params.guildId)
|
||||
.members.fetch(req.session.user.id)
|
||||
for (let PermissionRequired of req.requiredPermissions) {
|
||||
let converted2 = PermissionRequired[0]
|
||||
const DiscordJsVersion2 = Discord.version.split(".")[0]
|
||||
|
||||
if (DiscordJsVersion2 === "14")
|
||||
converted2 = await convert14(PermissionRequired[0])
|
||||
|
||||
if (
|
||||
!bot.guilds.cache
|
||||
.get(req.params.guildId)
|
||||
.members.cache.get(req.session.user.id)
|
||||
.permissions.has(converted2)
|
||||
)
|
||||
return res.redirect("/manage?error=noPermsToManageGuild")
|
||||
}
|
||||
|
||||
let cid = req.params.categoryId
|
||||
let settings = config.settings
|
||||
|
||||
let category = settings.find((c) => c.categoryId == cid)
|
||||
|
||||
if (!category)
|
||||
return res.send({
|
||||
error: true,
|
||||
message: "No category found",
|
||||
})
|
||||
|
||||
let setNewRes
|
||||
let errors = []
|
||||
let successes = []
|
||||
let catO = []
|
||||
|
||||
const userGuildMemberObject = bot.guilds.cache
|
||||
.get(req.params.guildId)
|
||||
.members.cache.get(req.session.user.id)
|
||||
const guildObject = bot.guilds.cache.get(req.params.guildId)
|
||||
|
||||
for (let option of category.categoryOptionsList) {
|
||||
let canUse = {}
|
||||
|
||||
if (option.allowedCheck) {
|
||||
canUse = await option.allowedCheck({
|
||||
guild: { id: req.params.guildId },
|
||||
user: { id: req.session.user.id },
|
||||
})
|
||||
} else {
|
||||
canUse = { allowed: true, errorMessage: null }
|
||||
}
|
||||
|
||||
if (canUse.allowed == false) {
|
||||
setNewRes = { error: canUse.errorMessage }
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else if (option.optionType != "spacer") {
|
||||
if (config.useCategorySet) {
|
||||
if (
|
||||
option.optionType.type == "rolesMultiSelect" ||
|
||||
option.optionType.type == "channelsMultiSelect" ||
|
||||
option.optionType.type == "multiSelect"
|
||||
) {
|
||||
if (
|
||||
!req.body[option.optionId] ||
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
)
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: [],
|
||||
})
|
||||
else if (
|
||||
typeof req.body[option.optionId] != "object"
|
||||
)
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: [req.body[option.optionId]],
|
||||
})
|
||||
else
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: req.body[option.optionId],
|
||||
})
|
||||
} else if (option.optionType.type == "switch") {
|
||||
if (
|
||||
req.body[option.optionId] ||
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
) {
|
||||
if (
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
)
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: false,
|
||||
})
|
||||
else
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: true,
|
||||
})
|
||||
}
|
||||
} else if (option.optionType.type == "embedBuilder") {
|
||||
if (
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
)
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: option.optionType.data,
|
||||
})
|
||||
else {
|
||||
try {
|
||||
const parsedResponse = JSON.parse(
|
||||
req.body[option.optionId]
|
||||
)
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: parsedResponse,
|
||||
})
|
||||
} catch (err) {
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: option.optionType.data,
|
||||
})
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (
|
||||
req.body[option.optionId] == undefined ||
|
||||
req.body[option.optionId] == null
|
||||
)
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: null,
|
||||
})
|
||||
else
|
||||
catO.push({
|
||||
optionId: option.optionId,
|
||||
data: req.body[option.optionId],
|
||||
})
|
||||
}
|
||||
} else {
|
||||
if (
|
||||
option.optionType.type == "rolesMultiSelect" ||
|
||||
option.optionType.type == "channelsMultiSelect" ||
|
||||
option.optionType.type == "multiSelect"
|
||||
) {
|
||||
if (
|
||||
!req.body[option.optionId] ||
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
) {
|
||||
setNewRes = await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: [],
|
||||
})
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
} else if (
|
||||
typeof req.body[option.optionId] != "object"
|
||||
) {
|
||||
setNewRes = await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: [req.body[option.optionId]],
|
||||
})
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
} else {
|
||||
setNewRes = await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: req.body[option.optionId],
|
||||
})
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
}
|
||||
} else if (option.optionType.type == "switch") {
|
||||
if (
|
||||
req.body[option.optionId] ||
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
) {
|
||||
if (
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
) {
|
||||
setNewRes =
|
||||
(await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: false,
|
||||
})) || {}
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
} else {
|
||||
setNewRes =
|
||||
(await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: true,
|
||||
})) || {}
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (option.optionType.type == "embedBuilder") {
|
||||
if (
|
||||
req.body[option.optionId] == null ||
|
||||
req.body[option.optionId] == undefined
|
||||
) {
|
||||
setNewRes =
|
||||
(await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: option.optionType.data,
|
||||
})) || {}
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
} else {
|
||||
try {
|
||||
const parsedResponse = JSON.parse(
|
||||
req.body[option.optionId]
|
||||
)
|
||||
setNewRes =
|
||||
(await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: parsedResponse,
|
||||
})) || {}
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
} catch (err) {
|
||||
setNewRes =
|
||||
(await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: option.optionType.data,
|
||||
})) || {}
|
||||
setNewRes = {
|
||||
error: "JSON parse for embed builder went wrong, your settings have been reset.",
|
||||
}
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (
|
||||
req.body[option.optionId] == undefined ||
|
||||
req.body[option.optionId] == null
|
||||
) {
|
||||
setNewRes =
|
||||
(await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: null,
|
||||
})) || {}
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
} else {
|
||||
setNewRes =
|
||||
(await option.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
newData: req.body[option.optionId],
|
||||
})) || {}
|
||||
setNewRes ? null : (setNewRes = {})
|
||||
if (setNewRes.error) {
|
||||
errors.push(
|
||||
option.optionName +
|
||||
"%is%" +
|
||||
setNewRes.error +
|
||||
"%is%" +
|
||||
option.optionId
|
||||
)
|
||||
} else {
|
||||
successes.push(option.optionName)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (config.useCategorySet && catO.length) {
|
||||
let sNR = await category.setNew({
|
||||
guild: {
|
||||
id: req.params.guildId,
|
||||
object: guildObject,
|
||||
},
|
||||
user: {
|
||||
id: req.session.user.id,
|
||||
object: userGuildMemberObject,
|
||||
},
|
||||
data: catO,
|
||||
})
|
||||
sNR ? null : (sNR = {})
|
||||
if (sNR.error) {
|
||||
errors.push(category.categoryId + "%is%" + sNR.error)
|
||||
} else {
|
||||
successes.push(category.categoryId)
|
||||
}
|
||||
}
|
||||
|
||||
let successesForDBDEvent = []
|
||||
let errorsForDBDEvent = []
|
||||
|
||||
successes.forEach((item) => {
|
||||
if (typeof item == "string") {
|
||||
successesForDBDEvent.push(item.split("%is%"))
|
||||
}
|
||||
})
|
||||
|
||||
errors.forEach((item) => {
|
||||
if (typeof item == "string") {
|
||||
errorsForDBDEvent.push(item.split("%is%"))
|
||||
}
|
||||
})
|
||||
|
||||
req.DBDEvents.emit("guildSettingsUpdated", {
|
||||
user: req.session.user,
|
||||
changes: { successesForDBDEvent, errorsForDBDEvent },
|
||||
guildId: req.params.guildId,
|
||||
})
|
||||
|
||||
if (errors[0]) {
|
||||
if (!successes) successes = []
|
||||
req.session.success = successes.join("%and%")
|
||||
req.session.errors = errors.join("%and%")
|
||||
return res.redirect("/guild/" + req.params.guildId)
|
||||
} else {
|
||||
req.session.success = true
|
||||
req.session.errors = false
|
||||
return res.redirect("/guild/" + req.params.guildId)
|
||||
}
|
||||
}
|
||||
)
|
||||
|
||||
return router
|
||||
}
|
||||
|
||||
function convert14(perm) {
|
||||
let final = "NULL"
|
||||
|
||||
switch (perm) {
|
||||
case "CREATE_INSTANT_INVITE":
|
||||
final = "CreateInstantInvite"
|
||||
break
|
||||
case "KICK_MEMBERS":
|
||||
final = "KickMembers"
|
||||
break
|
||||
case "BAN_MEMBERS":
|
||||
final = "BanMembers"
|
||||
break
|
||||
case "ADMINISTRATOR":
|
||||
final = "Administrator"
|
||||
break
|
||||
case "MANAGE_CHANNELS":
|
||||
final = "ManageChannels"
|
||||
break
|
||||
case "MANAGE_GUILD":
|
||||
final = "ManageGuild"
|
||||
break
|
||||
case "ADD_REACTIONS":
|
||||
final = "AddReactions"
|
||||
break
|
||||
case "VIEW_AUDIT_LOG":
|
||||
final = "ViewAuditLog"
|
||||
break
|
||||
case "PRIORITY_SPEAKER":
|
||||
final = "PrioritySpeaker"
|
||||
break
|
||||
case "STREAM":
|
||||
final = "Stream"
|
||||
break
|
||||
case "VIEW_CHANNEL":
|
||||
final = "ViewChannel"
|
||||
break
|
||||
case "SEND_MESSAGES":
|
||||
final = "SendMessages"
|
||||
break
|
||||
case "SEND_TTS_MESSAGES":
|
||||
final = "SendTTSMessages"
|
||||
break
|
||||
case "MANAGE_MESSAGES":
|
||||
final = "ManageMessages"
|
||||
break
|
||||
case "EMBED_LINKS":
|
||||
final = "ManageMessages"
|
||||
break
|
||||
case "ATTACH_FILES":
|
||||
final = "AttachFiles"
|
||||
break
|
||||
case "READ_MESSAGE_HISTORY":
|
||||
final = "ReadMessageHistory"
|
||||
break
|
||||
case "MENTION_EVERYONE":
|
||||
final = "MentionEveryone"
|
||||
break
|
||||
case "USE_EXTERNAL_EMOJIS":
|
||||
final = "UseExternalEmojis"
|
||||
break
|
||||
case "VIEW_GUILD_INSIGHTS":
|
||||
final = "ViewGuildInsughts"
|
||||
break
|
||||
case "CONNECT":
|
||||
final = "Connect"
|
||||
break
|
||||
case "SPEAK":
|
||||
final = "Speak"
|
||||
break
|
||||
}
|
||||
|
||||
return final
|
||||
}
|
421
Routes/discord.js
Normal file
421
Routes/discord.js
Normal file
|
@ -0,0 +1,421 @@
|
|||
const router = require("express").Router()
|
||||
const fetch = require("node-fetch")
|
||||
const DBDStats = require("../ExternalStatistics/index")
|
||||
|
||||
const DiscordOauth2 = require("discord-oauth2")
|
||||
const oauth = new DiscordOauth2()
|
||||
|
||||
module.exports = (app, config, themeConfig) => {
|
||||
const scopes = config.guildAfterAuthorization?.use
|
||||
? ["identify", "guilds", "guilds.join"]
|
||||
: ["identify", "guilds"]
|
||||
const RL = require("express-rate-limit")
|
||||
const RateLimits = config.rateLimits || {}
|
||||
let RateFunctions = {}
|
||||
|
||||
const NoRL = (req, res, next) => next()
|
||||
|
||||
if (RateLimits.discordOAuth2) {
|
||||
RateFunctions.discordOAuth2 = RL.rateLimit({
|
||||
windowMs: RateLimits.discordOAuth2.windowMs,
|
||||
max: RateLimits.discordOAuth2.max,
|
||||
message: RateLimits.discordOAuth2.message,
|
||||
store: RateLimits.discordOAuth2.store || new RL.MemoryStore(),
|
||||
})
|
||||
}
|
||||
|
||||
router.get("/", (req, res) => {
|
||||
const clientId = req.client.id
|
||||
const redirectUri = req.redirectUri
|
||||
|
||||
let newPage = "/"
|
||||
if (themeConfig.landingPage?.enabled) newPage = "/dash"
|
||||
req.session.r = req.query.r || newPage
|
||||
|
||||
const authorizeUrl = `https://discord.com/api/oauth2/authorize?client_id=${clientId}&redirect_uri=${encodeURIComponent(
|
||||
redirectUri
|
||||
)}&response_type=code&scope=${scopes.join("%20")}`
|
||||
res.redirect(authorizeUrl)
|
||||
})
|
||||
|
||||
router.get("/status", async (req, res) => {
|
||||
res.send(req.session?.discordAuthStatus)
|
||||
})
|
||||
|
||||
router.get(
|
||||
"/callback",
|
||||
RateFunctions.discordOAuth2 ? RateFunctions.discordOAuth2 : NoRL,
|
||||
async (req, res) => {
|
||||
req.session.discordAuthStatus = {
|
||||
loading: true,
|
||||
success: null,
|
||||
state: {
|
||||
error: null,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
const clientId = req.client.id
|
||||
const clientSecret = req.client.secret
|
||||
const redirectUri = req.redirectUri
|
||||
|
||||
const accessCode = req.query.code
|
||||
if (!accessCode)
|
||||
return res.redirect("/?error=NoAccessCodeReturnedFromDiscord")
|
||||
|
||||
res.redirect("/loading")
|
||||
|
||||
let OAuth2Response
|
||||
let OAuth2UserResponse
|
||||
let OAuth2GuildsResponse
|
||||
|
||||
/*
|
||||
Get Discord OAuth2 API Response with Access Code gained
|
||||
*/
|
||||
try {
|
||||
req.session.discordAuthStatus = {
|
||||
loading: true,
|
||||
success: null,
|
||||
state: {
|
||||
error: null,
|
||||
data: "Requesting token...",
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
OAuth2Response = await oauth.tokenRequest({
|
||||
clientId,
|
||||
clientSecret,
|
||||
|
||||
code: accessCode,
|
||||
scope: scopes.join(" "),
|
||||
grantType: "authorization_code",
|
||||
|
||||
redirectUri,
|
||||
})
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - OAuth2Response (line 86)",
|
||||
err
|
||||
)
|
||||
req.session.discordAuthStatus = {
|
||||
loading: false,
|
||||
success: false,
|
||||
state: {
|
||||
error: err,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
return
|
||||
}
|
||||
|
||||
/*
|
||||
Get User from Discord OAuth2 API using gained access_token and update its values with tag and avatarURL
|
||||
*/
|
||||
|
||||
try {
|
||||
req.session.discordAuthStatus = {
|
||||
loading: true,
|
||||
success: null,
|
||||
state: {
|
||||
error: null,
|
||||
data: "Getting User...",
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
OAuth2UserResponse = await oauth.getUser(
|
||||
OAuth2Response.access_token
|
||||
)
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - OAuth2UserResponse (line 118)",
|
||||
err
|
||||
)
|
||||
req.session.discordAuthStatus = {
|
||||
loading: false,
|
||||
success: false,
|
||||
state: {
|
||||
error: err,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
return
|
||||
}
|
||||
OAuth2UserResponse.tag = `${OAuth2UserResponse.username}#${OAuth2UserResponse.discriminator}`
|
||||
OAuth2UserResponse.avatarURL = OAuth2UserResponse.avatar
|
||||
? `https://cdn.discordapp.com/avatars/${OAuth2UserResponse.id}/${OAuth2UserResponse.avatar}.png?size=1024`
|
||||
: null
|
||||
|
||||
/*
|
||||
Save user token in Assistants Secure Storage
|
||||
*/
|
||||
|
||||
try {
|
||||
req.AssistantsSecureStorage.SaveUser(
|
||||
OAuth2UserResponse.id,
|
||||
OAuth2Response.access_token
|
||||
)
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - Assistants Secure Storage (line 141)",
|
||||
err
|
||||
)
|
||||
req.session.discordAuthStatus = {
|
||||
loading: false,
|
||||
success: false,
|
||||
state: {
|
||||
error: err,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
return
|
||||
}
|
||||
|
||||
/*
|
||||
Save user in session
|
||||
*/
|
||||
|
||||
req.session.user = OAuth2UserResponse
|
||||
req.session.loggedInLastTime = true
|
||||
|
||||
/*
|
||||
Register user to DBD Stats and emit userLoggedIn event
|
||||
*/
|
||||
|
||||
try {
|
||||
DBDStats.registerUser(OAuth2UserResponse.id)
|
||||
req.DBDEvents.emit("userLoggedIn", OAuth2UserResponse)
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - DBDStats register and DBDEvent emit userLoggedIn (line 170)",
|
||||
err
|
||||
)
|
||||
req.session.discordAuthStatus = {
|
||||
loading: false,
|
||||
success: false,
|
||||
state: {
|
||||
error: err,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
return
|
||||
}
|
||||
|
||||
/*
|
||||
Gain and update session with user guilds
|
||||
*/
|
||||
|
||||
try {
|
||||
req.session.discordAuthStatus = {
|
||||
loading: true,
|
||||
success: null,
|
||||
state: {
|
||||
error: null,
|
||||
data: "Getting List of User Guilds...",
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
OAuth2GuildsResponse = await oauth.getUserGuilds(
|
||||
OAuth2Response.access_token
|
||||
)
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - OAuth2GuildsResponse (line 201)",
|
||||
err
|
||||
)
|
||||
req.session.discordAuthStatus = {
|
||||
loading: false,
|
||||
success: false,
|
||||
state: {
|
||||
error: err,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
return
|
||||
}
|
||||
req.session.guilds = OAuth2GuildsResponse || []
|
||||
|
||||
/*
|
||||
Loop and fetch each guild into bots cache
|
||||
*/
|
||||
|
||||
if (!req.config.disableResolvingGuildCache) {
|
||||
try {
|
||||
req.session.discordAuthStatus = {
|
||||
loading: true,
|
||||
success: null,
|
||||
state: {
|
||||
error: null,
|
||||
data: "Resolving guilds cache...",
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
for (let g of OAuth2GuildsResponse) {
|
||||
try {
|
||||
await req.bot.guilds.fetch(g.id)
|
||||
} catch (err) {}
|
||||
}
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - OAuth2GuildsResponse Whole Loop (line 239)",
|
||||
err
|
||||
)
|
||||
req.session.discordAuthStatus = {
|
||||
loading: false,
|
||||
success: false,
|
||||
state: {
|
||||
error: err,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
If joining specific guild after authorization is enabled, do it
|
||||
*/
|
||||
|
||||
if (req.guildAfterAuthorization.use == true) {
|
||||
req.session.discordAuthStatus = {
|
||||
loading: true,
|
||||
success: null,
|
||||
state: {
|
||||
error: null,
|
||||
data: "Authorizing user with guild...",
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
try {
|
||||
await oauth.addMember({
|
||||
accessToken: OAuth2Response.access_token,
|
||||
botToken: req.botToken,
|
||||
guildId: req.guildAfterAuthorization.guildId,
|
||||
userId: OAuth2UserResponse.id,
|
||||
|
||||
...(req.guildAfterAuthorization.options || {}),
|
||||
/*
|
||||
options?: {
|
||||
nickname?: string,
|
||||
roles?: [string],
|
||||
mute?: boolean,
|
||||
deaf?: boolean,
|
||||
}
|
||||
*/
|
||||
})
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - guildAfterAuthorization (line 287)",
|
||||
err
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
req.session.discordAuthStatus = {
|
||||
loading: false,
|
||||
success: true,
|
||||
state: {
|
||||
error: null,
|
||||
data: null,
|
||||
},
|
||||
}
|
||||
req.session.save(function (err) {})
|
||||
|
||||
return
|
||||
}
|
||||
)
|
||||
|
||||
router.get("/logout", (req, res) => {
|
||||
let r = req.query.r || "/"
|
||||
req.session.destroy()
|
||||
res.redirect(r)
|
||||
})
|
||||
|
||||
router.get("/guilds/reload", async (req, res) => {
|
||||
if (!req.session.user) return res.redirect("/discord")
|
||||
|
||||
/*
|
||||
Fetch user token
|
||||
*/
|
||||
|
||||
const access_token = req.AssistantsSecureStorage.GetUser(
|
||||
req.session.user.id
|
||||
)
|
||||
if (!access_token)
|
||||
return res.send({
|
||||
error: true,
|
||||
message: "You don't have any access_token saved.",
|
||||
login_again_text: true,
|
||||
})
|
||||
|
||||
/*
|
||||
Gain and update session with user guilds
|
||||
*/
|
||||
|
||||
let OAuth2GuildsResponse
|
||||
|
||||
try {
|
||||
OAuth2GuildsResponse = await oauth.getUserGuilds(access_token)
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - OAuth2GuildsResponse for ReloadGuilds (line 335)",
|
||||
err
|
||||
)
|
||||
return res.send({
|
||||
error: true,
|
||||
message:
|
||||
"An error occured. Access_token is wrong or you're being rate limited.",
|
||||
login_again_text: true,
|
||||
})
|
||||
}
|
||||
req.session.guilds = OAuth2GuildsResponse || []
|
||||
|
||||
/*
|
||||
Loop and fetch each guild into bots cache
|
||||
*/
|
||||
|
||||
try {
|
||||
const Promises = []
|
||||
for (let g of OAuth2GuildsResponse) {
|
||||
Promises.push(
|
||||
new Promise(async (resolve, reject) => {
|
||||
try {
|
||||
await req.bot.guilds.fetch(g.id)
|
||||
} catch (err) {}
|
||||
resolve(1)
|
||||
})
|
||||
)
|
||||
try {
|
||||
await Promises.all(Promises)
|
||||
} catch (err) {}
|
||||
}
|
||||
} catch (err) {
|
||||
req.config.reportError(
|
||||
"Discord.js Route - OAuth2GuildsResponse Whole Loop for ReloadGuilds (line 363)",
|
||||
err
|
||||
)
|
||||
return res.send({
|
||||
error: true,
|
||||
message:
|
||||
"An error occured. Access_token is wrong or you're being rate limited.",
|
||||
login_again_text: true,
|
||||
})
|
||||
}
|
||||
|
||||
/*
|
||||
Success
|
||||
*/
|
||||
|
||||
return res.send({
|
||||
error: false,
|
||||
message: null,
|
||||
login_again_text: false,
|
||||
})
|
||||
})
|
||||
|
||||
return router
|
||||
}
|
122
Routes/main.js
Normal file
122
Routes/main.js
Normal file
|
@ -0,0 +1,122 @@
|
|||
const Discord = require("discord.js")
|
||||
const router = require("express").Router()
|
||||
|
||||
module.exports = (app, config, themeConfig, modules) => {
|
||||
router.get(
|
||||
themeConfig.landingPage?.enabled ? "/dash" : "/",
|
||||
async (req, res) => {
|
||||
let customThemeOptions
|
||||
if (themeConfig?.customThemeOptions?.index) {
|
||||
customThemeOptions = await themeConfig.customThemeOptions.index(
|
||||
{ req: req, res: res, config: config }
|
||||
)
|
||||
}
|
||||
res.render("index", {
|
||||
req: req,
|
||||
themeConfig: req.themeConfig,
|
||||
bot: config.bot,
|
||||
customThemeOptions: customThemeOptions || {},
|
||||
config,
|
||||
require,
|
||||
})
|
||||
}
|
||||
)
|
||||
|
||||
if (themeConfig.landingPage?.enabled)
|
||||
router.get("/", async (req, res) => {
|
||||
res.setHeader("Content-Type", "text/html")
|
||||
res.send(await themeConfig.landingPage.getLandingPage(req, res))
|
||||
})
|
||||
|
||||
router.get("/loading", async (req, res) => {
|
||||
if (!req.session?.discordAuthStatus?.loading)
|
||||
return res.redirect("/manage")
|
||||
|
||||
res.render("loading", { req, themeConfig, bot: config.bot })
|
||||
})
|
||||
|
||||
router.get("/invite", (req, res) => {
|
||||
let config = req.config
|
||||
config.invite ? null : (config.invite = {})
|
||||
const scopes = config.invite.scopes || ["bot"]
|
||||
|
||||
if (req.query.redirect && !req.query.g)
|
||||
return res.redirect(
|
||||
`https://discord.com/oauth2/authorize?client_id=${
|
||||
config.invite.clientId || config.bot.user.id
|
||||
}&scope=${scopes.join("%20")}&permissions=${
|
||||
config.invite.permissions || "0"
|
||||
}&response_type=code&redirect_uri=${req.query.redirect}${
|
||||
config.invite.otherParams || ""
|
||||
}`
|
||||
)
|
||||
if (req.query.redirect && req.query.g)
|
||||
return res.redirect(
|
||||
`https://discord.com/oauth2/authorize?client_id=${
|
||||
config.invite.clientId || config.bot.user.id
|
||||
}&scope=${scopes.join("%20")}&permissions=${
|
||||
config.invite.permissions || "0"
|
||||
}&response_type=code&redirect_uri=${
|
||||
req.query.redirect
|
||||
}&guild_id=${req.query.g}${config.invite.otherParams || ""}`
|
||||
)
|
||||
|
||||
if (req.query.g) {
|
||||
let thingymabob = config.invite.redirectUri
|
||||
? `&response_type=code&redirect_uri=${config.invite.redirectUri}`
|
||||
: null;
|
||||
if(!thingymabob) thingymabob = config.invite.specialredirectUri
|
||||
? `&response_type=code&redirect_uri=${config.invite.specialRedirectUri.replace("{SERVER}", req.query.g)}`
|
||||
: "";
|
||||
|
||||
return res.redirect(
|
||||
`https://discord.com/oauth2/authorize?client_id=${
|
||||
config.invite.clientId || config.bot.user.id
|
||||
}&scope=${scopes.join("%20")}&permissions=${
|
||||
config.invite.permissions || "0"
|
||||
}${thingymabob}&guild_id=${req.query.g}${config.invite.otherParams || ""}`
|
||||
)
|
||||
}
|
||||
|
||||
res.redirect(
|
||||
`https://discord.com/oauth2/authorize?client_id=${
|
||||
config.invite.clientId || config.bot.user.id
|
||||
}&scope=${scopes.join("%20")}&permissions=${
|
||||
config.invite.permissions || "0"
|
||||
}${
|
||||
config.invite.redirectUri
|
||||
? `&response_type=code&redirect_uri=${config.invite.redirectUri}`
|
||||
: ""
|
||||
}${config.invite.otherParams || ""}`
|
||||
)
|
||||
})
|
||||
|
||||
if (!config.supportServer) config.supportServer = {}
|
||||
|
||||
router.get(config.supportServer.slash || "/support-server", (req, res) => {
|
||||
let config = req.config
|
||||
config.supportServer ? null : (config.supportServer = {})
|
||||
if (!config.supportServer.inviteUrl)
|
||||
return res.send({
|
||||
error: true,
|
||||
message:
|
||||
"No inviteUrl defined (discord-dashboard config.supportServer).",
|
||||
})
|
||||
if (
|
||||
!config.supportServer.inviteUrl
|
||||
.toLowerCase()
|
||||
.startsWith("https://discord.gg/") &&
|
||||
!config.supportServer.inviteUrl
|
||||
.toLowerCase()
|
||||
.startsWith("https://discord.com/")
|
||||
)
|
||||
return res.send({
|
||||
error: true,
|
||||
message:
|
||||
"Invite url should start with 'https://discord.gg/' or 'https://discord.com/'.",
|
||||
})
|
||||
res.redirect(config.supportServer.inviteUrl)
|
||||
})
|
||||
|
||||
return router
|
||||
}
|
10
SECURITY.md
Normal file
10
SECURITY.md
Normal file
|
@ -0,0 +1,10 @@
|
|||
# Security Policy
|
||||
|
||||
## Supported Versions
|
||||
|
||||
Currently supported versions:
|
||||
|
||||
| Version | Supported |
|
||||
| ------- | ------------------ |
|
||||
| > 2.\* | :white_check_mark: |
|
||||
| < 2.\* | :x: |
|
13
__ModuleExample/index.js
Normal file
13
__ModuleExample/index.js
Normal file
|
@ -0,0 +1,13 @@
|
|||
module.exports = {
|
||||
app: ({ app, config, themeConfig }) => {
|
||||
app.get("/moduleexample", (req, res) => {
|
||||
res.send("ModuleExample: Hello World!")
|
||||
})
|
||||
},
|
||||
server: ({ io, server, config, themeConfig }) => {
|
||||
const test = io.of("/moduleexample")
|
||||
test.on("connection", () => {
|
||||
console.log("ModuleExample socket.io connected.")
|
||||
})
|
||||
},
|
||||
}
|
9
index.js
Normal file
9
index.js
Normal file
File diff suppressed because one or more lines are too long
0
licensedDashboardClass.js
Normal file
0
licensedDashboardClass.js
Normal file
4537
package-lock.json
generated
Normal file
4537
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
55
package.json
Normal file
55
package.json
Normal file
|
@ -0,0 +1,55 @@
|
|||
{
|
||||
"name": "discord-dashboard",
|
||||
"version": "2.3.61",
|
||||
"description": "Create an Dashboard for your bot in 10 minutes without APIs knowledge and slapping code from scratch!",
|
||||
"main": "index.js",
|
||||
"directories": {
|
||||
"test": "test"
|
||||
},
|
||||
"typings": ".d.ts",
|
||||
"dependencies": {
|
||||
"assistants-safe-storage": "^1.0.0",
|
||||
"body-parser": "^1.20.0",
|
||||
"colors": "^1.4.0",
|
||||
"cookie-parser": "^1.4.6",
|
||||
"discord-dashboard": "^2.3.39",
|
||||
"discord-dashboard-pp-system": "^1.0.2",
|
||||
"discord-oauth2": "^2.10.0",
|
||||
"discord.js": "*",
|
||||
"ejs": "^3.1.8",
|
||||
"express": "^4.17.3",
|
||||
"express-partials": "^0.3.0",
|
||||
"express-rate-limit": "^6.4.0",
|
||||
"express-session": "^1.17.3",
|
||||
"https": "^1.0.0",
|
||||
"keyv": "^4.5.2",
|
||||
"node-fetch": "^2.6.7",
|
||||
"readline-sync": "^1.4.10",
|
||||
"socket.io": "^4.5.1",
|
||||
"uuid": "^8.3.2"
|
||||
},
|
||||
"scripts": {
|
||||
"test": "echo \"Error: no test specified\" && exit 1"
|
||||
},
|
||||
"author": "breftejk",
|
||||
"license": "CC BY-NC-SA 4.0",
|
||||
"homepage": "https://dbd-docs.assistantscenter.com/",
|
||||
"keywords": [
|
||||
"discord",
|
||||
"discord.js",
|
||||
"discordjs",
|
||||
"discord dashboard",
|
||||
"discord web dashboard",
|
||||
"web dashboard",
|
||||
"dashboard"
|
||||
],
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/Assistants-Center/Discord.js-Web-Dashboard.git"
|
||||
},
|
||||
"devDependencies": {
|
||||
"dbd-dark-dashboard": "^1.6.58",
|
||||
"javascript-obfuscator": "^4.0.0",
|
||||
"prettier": "2.7.1"
|
||||
}
|
||||
}
|
173
router.js
Normal file
173
router.js
Normal file
|
@ -0,0 +1,173 @@
|
|||
module.exports = (app, config, themeConfig, modules) => {
|
||||
app.use(require("cookie-parser")())
|
||||
|
||||
app.use((req, res, next) => {
|
||||
req.bot = config.bot
|
||||
next()
|
||||
})
|
||||
|
||||
if (themeConfig.defaultLocales) {
|
||||
app.use((req, res, next) => {
|
||||
if (req.cookies?.lang) req.lang = req.cookies.lang
|
||||
else req.lang = req.acceptsLanguages()[0].replace("-", "")
|
||||
|
||||
if (themeConfig.locales) {
|
||||
if (Object.keys(themeConfig.locales).includes(req.lang)) {
|
||||
req.locales = themeConfig.locales[req.lang]
|
||||
} else {
|
||||
req.locales =
|
||||
themeConfig.locales[Object.keys(themeConfig.locales)[0]]
|
||||
}
|
||||
} else {
|
||||
req.locales =
|
||||
themeConfig.defaultLocales[
|
||||
Object.keys(themeConfig.defaultLocales).includes(
|
||||
req.lang
|
||||
)
|
||||
? req.lang
|
||||
: "enUS"
|
||||
]
|
||||
}
|
||||
|
||||
next()
|
||||
})
|
||||
}
|
||||
|
||||
app.use(
|
||||
"/discord",
|
||||
require("./Routes/discord")(app, config, themeConfig, modules)
|
||||
)
|
||||
|
||||
if (config.useUnderMaintenance) {
|
||||
app.get(
|
||||
config.underMaintenanceAccessPage || "/total-secret-get-access",
|
||||
(req, res) => {
|
||||
res.send(`
|
||||
<form action="${config.domain}${
|
||||
config.underMaintenanceAccessPage ||
|
||||
"/total-secret-get-access"
|
||||
}" method="POST" >
|
||||
<input id="accessKey" name="accessKey"/>
|
||||
<button role="submit">Submit</button>
|
||||
</form>
|
||||
`)
|
||||
}
|
||||
)
|
||||
|
||||
app.post(
|
||||
config.underMaintenanceAccessPage || "/total-secret-get-access",
|
||||
(req, res) => {
|
||||
if (!req.body) req.body = {}
|
||||
const accessKey = req.body.accessKey
|
||||
if (accessKey != config.underMaintenanceAccessKey)
|
||||
return res.send("Wrong key.")
|
||||
req.session.umaccess = true
|
||||
res.redirect("/")
|
||||
}
|
||||
)
|
||||
|
||||
app.use((req, res, next) => {
|
||||
if (req.originalUrl.startsWith("/loading")) return next()
|
||||
if (!req.session.umaccess && !req.session.user) {
|
||||
if (!config.useThemeMaintenance)
|
||||
return res.send(
|
||||
config.underMaintenanceCustomHtml ||
|
||||
require("./underMaintenancePageDefault")(
|
||||
config.underMaintenance,
|
||||
false
|
||||
)
|
||||
)
|
||||
else
|
||||
res.render("maintenance", {
|
||||
req: req,
|
||||
bot: config.bot,
|
||||
themeConfig: req.themeConfig,
|
||||
loggedIn: false,
|
||||
defaultMaintenanceConfig: config.underMaintenance || {},
|
||||
})
|
||||
} else if (
|
||||
!req.session.umaccess &&
|
||||
config.ownerIDs &&
|
||||
!config.ownerIDs.includes(req.session.user.id)
|
||||
) {
|
||||
if (!config.useThemeMaintenance)
|
||||
return res.send(
|
||||
config.underMaintenanceCustomHtml ||
|
||||
require("./underMaintenancePageDefault")(
|
||||
config.underMaintenance,
|
||||
true
|
||||
)
|
||||
)
|
||||
else
|
||||
res.render("maintenance", {
|
||||
req: req,
|
||||
bot: config.bot,
|
||||
themeConfig: req.themeConfig,
|
||||
loggedIn: true,
|
||||
defaultMaintenanceConfig: config.underMaintenance || {},
|
||||
})
|
||||
} else next()
|
||||
})
|
||||
}
|
||||
|
||||
app.use("/", require("./Routes/main")(app, config, themeConfig, modules))
|
||||
app.use(
|
||||
"/",
|
||||
require("./Routes/dashboard")(app, config, themeConfig, modules)
|
||||
)
|
||||
|
||||
config.theme.init(app, config)
|
||||
|
||||
let customPages = config.customPages || []
|
||||
customPages.forEach((p) => {
|
||||
if (p.type == "redirect") {
|
||||
app.get(p.endpoint, async (req, res) => {
|
||||
let endpoint = await p.getEndpoint({
|
||||
user: req.session.user || {},
|
||||
req,
|
||||
})
|
||||
res.redirect(endpoint)
|
||||
})
|
||||
} else if (p.type == "html") {
|
||||
app.get(p.endpoint, async (req, res) => {
|
||||
let html = await p.getHtml({
|
||||
user: req.session.user || {},
|
||||
req,
|
||||
})
|
||||
res.send(html)
|
||||
})
|
||||
} else if (p.type == "json") {
|
||||
app.get(p.endpoint, async (req, res) => {
|
||||
let json = await p.getJson({
|
||||
user: req.session.user || {},
|
||||
req,
|
||||
})
|
||||
res.send(json)
|
||||
})
|
||||
}
|
||||
})
|
||||
|
||||
modules.forEach((module) => {
|
||||
module.app({
|
||||
app: app,
|
||||
config: this.config,
|
||||
themeConfig: themeConfig,
|
||||
})
|
||||
})
|
||||
|
||||
if (!config.useTheme404) {
|
||||
app.get("*", (req, res) => {
|
||||
let text =
|
||||
config.html404 ||
|
||||
require("./404pagedefault")(
|
||||
config.websiteTitle || themeConfig.websiteName
|
||||
)
|
||||
res.send(
|
||||
text.replace(
|
||||
"{{websiteTitle}}",
|
||||
config.websiteTitle || themeConfig.websiteName
|
||||
)
|
||||
)
|
||||
})
|
||||
}
|
||||
}
|
689
underMaintenancePageDefault.js
Normal file
689
underMaintenancePageDefault.js
Normal file
|
@ -0,0 +1,689 @@
|
|||
module.exports = (
|
||||
{
|
||||
title,
|
||||
contentTitle,
|
||||
texts = [],
|
||||
bodyBackgroundColors = ["#ffa191", "#ffc247"],
|
||||
buildingsColor = "#ff6347",
|
||||
craneDivBorderColor = "#ff6347",
|
||||
craneArmColor = "#f88f7c",
|
||||
craneWeightColor = "#f88f7c",
|
||||
outerCraneColor = "#ff6347",
|
||||
craneLineColor = "#ff6347",
|
||||
craneCabinColor = "#f88f7c",
|
||||
craneStandColors = ["#ff6347", "#f29b8b"],
|
||||
},
|
||||
loggedIn
|
||||
) => {
|
||||
let logged = "<a class='text' href='/discord'>Login</a>"
|
||||
if (loggedIn) logged = "<a class='text' href='/discord/logout'>Logout</a>"
|
||||
return `
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<title>${title}</title>
|
||||
<style>
|
||||
@import url("https://fonts.googleapis.com/css?family=Ubuntu");
|
||||
#content-title {
|
||||
text-align: center !important;
|
||||
}
|
||||
#outerCraneContainer {
|
||||
position: absolute;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
bottom: 0;
|
||||
overflow: hidden;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
box-shadow: inset 0 -60px 0 -30px ${outerCraneColor};
|
||||
}
|
||||
.buildings {
|
||||
height: 84.9673202614379px;
|
||||
width: 100%;
|
||||
left: 0;
|
||||
}
|
||||
.buildings div {
|
||||
height: inherit;
|
||||
width: 42.48366013071895px;
|
||||
background: ${buildingsColor};
|
||||
position: absolute;
|
||||
bottom: 10%;
|
||||
}
|
||||
.buildings div:after {
|
||||
content: '';
|
||||
width: 80%;
|
||||
height: 60%;
|
||||
left: 10%;
|
||||
bottom: 30%;
|
||||
background: url("") repeat;
|
||||
position: absolute;
|
||||
}
|
||||
.buildings div:nth-of-type(1) {
|
||||
width: 42.48366013071895px;
|
||||
height: 21.241830065359476px;
|
||||
right: 37%;
|
||||
bottom: 18%;
|
||||
}
|
||||
.buildings div:nth-of-type(1):after {
|
||||
bottom: 11%;
|
||||
}
|
||||
.buildings div:nth-of-type(2) {
|
||||
width: 48.552754435107374px;
|
||||
height: 28.322440087145967px;
|
||||
right: 30%;
|
||||
bottom: 35%;
|
||||
transform: rotate(180deg);
|
||||
}
|
||||
.buildings div:nth-of-type(2):after {
|
||||
width: 60%;
|
||||
left: 11%;
|
||||
}
|
||||
.buildings div:nth-of-type(3) {
|
||||
width: 24.276377217553687px;
|
||||
height: 37.76325344952796px;
|
||||
left: 40%;
|
||||
bottom: 35%;
|
||||
}
|
||||
.buildings div:nth-of-type(3):after {
|
||||
bottom: 0;
|
||||
width: 20%;
|
||||
height: 85%;
|
||||
left: 70%;
|
||||
}
|
||||
.buildings div:nth-of-type(4) {
|
||||
width: 84.9673202614379px;
|
||||
height: 21.241830065359476px;
|
||||
left: 24%;
|
||||
bottom: 20%;
|
||||
}
|
||||
.buildings div:nth-of-type(4):after {
|
||||
background: none;
|
||||
}
|
||||
.buildings div:nth-of-type(5) {
|
||||
width: 61.794414735591204px;
|
||||
height: 67.97385620915033px;
|
||||
left: 47%;
|
||||
bottom: 10%;
|
||||
}
|
||||
.buildings div:nth-of-type(5):after {
|
||||
bottom: 0;
|
||||
width: 40%;
|
||||
height: 85%;
|
||||
left: 20%;
|
||||
}
|
||||
.crane,
|
||||
.buildings {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
}
|
||||
.crane div {
|
||||
border-radius: 2px;
|
||||
position: absolute;
|
||||
}
|
||||
.crane .line {
|
||||
border: none;
|
||||
background: ${craneLineColor};
|
||||
outline: 1px solid transparent;
|
||||
z-index: 0;
|
||||
}
|
||||
.crane .lineOne {
|
||||
width: 60%;
|
||||
left: 11%;
|
||||
top: 0;
|
||||
}
|
||||
.crane .lineTwo {
|
||||
width: 19%;
|
||||
right: 8%;
|
||||
top: 0;
|
||||
}
|
||||
.crane .line.lineThree {
|
||||
height: 30%;
|
||||
top: 22%;
|
||||
left: 9%;
|
||||
}
|
||||
.crane .line.lineThree:after {
|
||||
content: '';
|
||||
position: absolute;
|
||||
height: 0.2em;
|
||||
width: 9000%;
|
||||
background: ${craneLineColor};
|
||||
display: block;
|
||||
bottom: 0;
|
||||
left: -4500%;
|
||||
border: solid 1px ${craneLineColor};
|
||||
}
|
||||
.craneTwo .line.lineThree:after {
|
||||
height: 0.1em;
|
||||
}
|
||||
.craneThree .line.lineThree:after {
|
||||
height: 0.05em;
|
||||
}
|
||||
.stand {
|
||||
height: 100%;
|
||||
width: 5%;
|
||||
right: 25%;
|
||||
z-index: 1;
|
||||
background: linear-gradient(to top, ${
|
||||
craneStandColors[0]
|
||||
}, ${craneStandColors[1]});
|
||||
}
|
||||
.craneTwo .stand {
|
||||
background: linear-gradient(to top, ${
|
||||
craneStandColors[0]
|
||||
}, ${craneStandColors[1]});
|
||||
}
|
||||
.craneThree .stand {
|
||||
background: linear-gradient(to top, ${
|
||||
craneStandColors[0]
|
||||
}, ${craneStandColors[1]});
|
||||
}
|
||||
.weight {
|
||||
height: 20%;
|
||||
width: 8%;
|
||||
right: 4%;
|
||||
top: 12%;
|
||||
z-index: 2;
|
||||
background: ${craneWeightColor};
|
||||
}
|
||||
.craneTwo .weight {
|
||||
background: ${craneWeightColor};
|
||||
}
|
||||
.craneThree .weight {
|
||||
background: ${craneWeightColor};
|
||||
}
|
||||
.cabin {
|
||||
height: 9%;
|
||||
width: 12%;
|
||||
right: 24%;
|
||||
top: 20%;
|
||||
z-index: 2;
|
||||
background: ${craneCabinColor};
|
||||
}
|
||||
.cabin:after {
|
||||
content: '';
|
||||
position: absolute;
|
||||
height: 10%;
|
||||
width: 100%;
|
||||
background: ${craneCabinColor};
|
||||
display: block;
|
||||
top: 60%;
|
||||
left: 0;
|
||||
}
|
||||
.craneTwo .cabin {
|
||||
background: ${craneCabinColor};
|
||||
}
|
||||
.craneThree .cabin {
|
||||
background: ${craneCabinColor};
|
||||
}
|
||||
.arm {
|
||||
height: 7%;
|
||||
width: 100%;
|
||||
top: 15%;
|
||||
z-index: 3;
|
||||
background: ${craneArmColor};
|
||||
}
|
||||
.craneTwo .arm {
|
||||
background: ${craneArmColor};
|
||||
}
|
||||
.craneThree .arm {
|
||||
background: ${craneArmColor};
|
||||
}
|
||||
.crane div.arm {
|
||||
border-top-left-radius: 10px;
|
||||
}
|
||||
.brick {
|
||||
height: 6%;
|
||||
width: 9%;
|
||||
bottom: 0;
|
||||
left: 40%;
|
||||
background: ${craneArmColor};
|
||||
}
|
||||
.brickTwo {
|
||||
left: 48%;
|
||||
}
|
||||
.brickThree {
|
||||
bottom: 5.5%;
|
||||
left: 44%;
|
||||
}
|
||||
.craneOne {
|
||||
width: 260px;
|
||||
height: 169.9346405228758px;
|
||||
left: 20%;
|
||||
}
|
||||
.craneOne div {
|
||||
border: solid 1px ${craneDivBorderColor};
|
||||
}
|
||||
.craneOne .line {
|
||||
height: 1px;
|
||||
}
|
||||
.craneOne .lineThree {
|
||||
width: 1px;
|
||||
}
|
||||
.craneTwo {
|
||||
width: 200px;
|
||||
height: 130.718954248366px;
|
||||
transform: scaleX(-1);
|
||||
left: 40%;
|
||||
z-index: -1;
|
||||
}
|
||||
.craneTwo div {
|
||||
border: solid 1px ${craneDivBorderColor};
|
||||
}
|
||||
.craneTwo .line {
|
||||
height: 0.769230769230769px;
|
||||
}
|
||||
.craneTwo .lineThree {
|
||||
width: 0.714285714285714px;
|
||||
animation-delay: 3s;
|
||||
}
|
||||
.craneTwo .cabin,
|
||||
.craneTwo .arm,
|
||||
.craneTwo .picker,
|
||||
.craneTwo .weight {
|
||||
animation-delay: 3s;
|
||||
}
|
||||
.craneThree {
|
||||
width: 130px;
|
||||
height: 84.9673202614379px;
|
||||
left: 60%;
|
||||
z-index: -1;
|
||||
}
|
||||
.craneThree div {
|
||||
border: solid 1px ${craneDivBorderColor};
|
||||
}
|
||||
.craneThree .line {
|
||||
height: 0.5px;
|
||||
}
|
||||
.craneThree .lineThree {
|
||||
width: 0.5px;
|
||||
animation-delay: 1.5s;
|
||||
}
|
||||
.craneThree .brickThree {
|
||||
bottom: 5%;
|
||||
}
|
||||
.craneThree .brickOne,
|
||||
.craneThree .brickTwo {
|
||||
bottom: 0;
|
||||
}
|
||||
.craneThree .cabin,
|
||||
.craneThree .arm,
|
||||
.craneThree .picker,
|
||||
.craneThree .weight {
|
||||
animation-delay: 1.5s;
|
||||
}
|
||||
.crane {
|
||||
perspective: 600px;
|
||||
}
|
||||
.lineOne {
|
||||
transform-origin: right 0;
|
||||
animation: moveLineOne 12s infinite alternate;
|
||||
}
|
||||
.lineTwo {
|
||||
transform-origin: top left;
|
||||
animation: moveLineTwo 12s infinite alternate;
|
||||
}
|
||||
.lineThree {
|
||||
transform-origin: right center;
|
||||
animation: moveLineThree 12s ease-in-out infinite alternate;
|
||||
}
|
||||
.cabin,
|
||||
.arm,
|
||||
.picker {
|
||||
transform-origin: 80% center;
|
||||
animation: moveCrane 12s infinite alternate;
|
||||
}
|
||||
.weight {
|
||||
transform-origin: 0 center;
|
||||
animation: moveWeight 12s infinite alternate;
|
||||
}
|
||||
html {
|
||||
-ms-text-size-adjust: 100%;
|
||||
-webkit-text-size-adjust: 100%;
|
||||
-webkit-tap-highlight-color: transparent;
|
||||
}
|
||||
html,
|
||||
body {
|
||||
height: 100%;
|
||||
}
|
||||
* {
|
||||
box-sizing: border-box;
|
||||
}
|
||||
body {
|
||||
background: linear-gradient(to top, ${
|
||||
bodyBackgroundColors[0]
|
||||
}, ${bodyBackgroundColors[1]});
|
||||
-webkit-font-smoothing: antialiased;
|
||||
-moz-osx-font-smoothing: grayscale;
|
||||
text-shadow: 1px 1px 1px rgba(0,0,0,0.004);
|
||||
}
|
||||
#content {
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
font-family: Ubuntu;
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
</style>
|
||||
<style>
|
||||
#content h1,
|
||||
#content p {
|
||||
margin: -8rem 0 0 1rem;
|
||||
}
|
||||
#content p {
|
||||
margin-top: 0.5rem;
|
||||
}
|
||||
#license {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
right: 5px;
|
||||
}
|
||||
@-moz-keyframes moveCrane {
|
||||
0%, 20% {
|
||||
transform: rotateY(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg);
|
||||
}
|
||||
}
|
||||
@-webkit-keyframes moveCrane {
|
||||
0%, 20% {
|
||||
transform: rotateY(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg);
|
||||
}
|
||||
}
|
||||
@-o-keyframes moveCrane {
|
||||
0%, 20% {
|
||||
transform: rotateY(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg);
|
||||
}
|
||||
}
|
||||
@keyframes moveCrane {
|
||||
0%, 20% {
|
||||
transform: rotateY(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg);
|
||||
}
|
||||
}
|
||||
@-moz-keyframes moveWeight {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) translateX(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) translateX(-50%);
|
||||
}
|
||||
}
|
||||
@-webkit-keyframes moveWeight {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) translateX(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) translateX(-50%);
|
||||
}
|
||||
}
|
||||
@-o-keyframes moveWeight {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) translateX(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) translateX(-50%);
|
||||
}
|
||||
}
|
||||
@keyframes moveWeight {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) translateX(0);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) translateX(-50%);
|
||||
}
|
||||
}
|
||||
@-moz-keyframes moveLineOne {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(-10deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) rotateZ(-10deg);
|
||||
}
|
||||
}
|
||||
@-webkit-keyframes moveLineOne {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(-10deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) rotateZ(-10deg);
|
||||
}
|
||||
}
|
||||
@-o-keyframes moveLineOne {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(-10deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) rotateZ(-10deg);
|
||||
}
|
||||
}
|
||||
@keyframes moveLineOne {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(-10deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(45deg) rotateZ(-10deg);
|
||||
}
|
||||
}
|
||||
@-moz-keyframes moveLineTwo {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(29deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(15deg) rotateZ(29deg);
|
||||
}
|
||||
}
|
||||
@-webkit-keyframes moveLineTwo {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(29deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(15deg) rotateZ(29deg);
|
||||
}
|
||||
}
|
||||
@-o-keyframes moveLineTwo {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(29deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(15deg) rotateZ(29deg);
|
||||
}
|
||||
}
|
||||
@keyframes moveLineTwo {
|
||||
0%, 20% {
|
||||
transform: rotateY(0) rotateZ(29deg);
|
||||
}
|
||||
70%, 100% {
|
||||
transform: rotateY(15deg) rotateZ(29deg);
|
||||
}
|
||||
}
|
||||
@-moz-keyframes moveLineThree {
|
||||
0% {
|
||||
transform: translate(0, 0);
|
||||
}
|
||||
20% {
|
||||
transform: translate(2500%, -18%);
|
||||
}
|
||||
60% {
|
||||
transform: translate(11000%, -25%);
|
||||
}
|
||||
70% {
|
||||
transform: translate(9100%, -25%);
|
||||
height: 30%;
|
||||
}
|
||||
90%, 100% {
|
||||
transform: translate(9100%, -15%);
|
||||
height: 80%;
|
||||
}
|
||||
}
|
||||
@-webkit-keyframes moveLineThree {
|
||||
0% {
|
||||
transform: translate(0, 0);
|
||||
}
|
||||
20% {
|
||||
transform: translate(2500%, -18%);
|
||||
}
|
||||
60% {
|
||||
transform: translate(11000%, -25%);
|
||||
}
|
||||
70% {
|
||||
transform: translate(9100%, -25%);
|
||||
height: 30%;
|
||||
}
|
||||
90%, 100% {
|
||||
transform: translate(9100%, -15%);
|
||||
height: 80%;
|
||||
}
|
||||
}
|
||||
@-o-keyframes moveLineThree {
|
||||
0% {
|
||||
transform: translate(0, 0);
|
||||
}
|
||||
20% {
|
||||
transform: translate(2500%, -18%);
|
||||
}
|
||||
60% {
|
||||
transform: translate(11000%, -25%);
|
||||
}
|
||||
70% {
|
||||
transform: translate(9100%, -25%);
|
||||
height: 30%;
|
||||
}
|
||||
90%, 100% {
|
||||
transform: translate(9100%, -15%);
|
||||
height: 80%;
|
||||
}
|
||||
}
|
||||
@keyframes moveLineThree {
|
||||
0% {
|
||||
transform: translate(0, 0);
|
||||
}
|
||||
20% {
|
||||
transform: translate(2500%, -18%);
|
||||
}
|
||||
60% {
|
||||
transform: translate(11000%, -25%);
|
||||
}
|
||||
70% {
|
||||
transform: translate(9100%, -25%);
|
||||
height: 30%;
|
||||
}
|
||||
90%, 100% {
|
||||
transform: translate(9100%, -15%);
|
||||
height: 80%;
|
||||
}
|
||||
}
|
||||
|
||||
body {
|
||||
background-color: lightblue;
|
||||
}
|
||||
|
||||
.text {
|
||||
font-size: 30px;
|
||||
text-align: center;
|
||||
z-index:9999999;
|
||||
}
|
||||
#content-title {
|
||||
font-size: 40px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
@media only screen and (min-device-width: 380px) {
|
||||
.text {
|
||||
font-size: 28px;
|
||||
text-align: center;
|
||||
z-index:9999999;
|
||||
}
|
||||
#content-title {
|
||||
font-size: 58px;
|
||||
text-align: center;
|
||||
}
|
||||
}
|
||||
|
||||
@media only screen and (max-device-width: 430px) {
|
||||
.text {
|
||||
font-size: 50px;
|
||||
text-align: center;
|
||||
z-index:9999999;
|
||||
}
|
||||
|
||||
#content-title {
|
||||
font-size: 80px;
|
||||
text-align: center;
|
||||
}
|
||||
}
|
||||
|
||||
@media only screen and (min-device-width: 750px) {
|
||||
.text {
|
||||
font-size: 20px;
|
||||
text-align: center;
|
||||
z-index:9999999;
|
||||
}
|
||||
#content-title {
|
||||
font-size: 40px;
|
||||
text-align: center;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body style="margin: 0px !important;">
|
||||
<div id="content">
|
||||
<h1 id="content-title">${contentTitle}</h1>
|
||||
${texts
|
||||
.map((text) => '<p class="text">' + text + "</p>")
|
||||
.join("")}
|
||||
<br><br><br>
|
||||
${logged}
|
||||
</div>
|
||||
<div id="outerCraneContainer">
|
||||
<div class="buildings">
|
||||
<div></div>
|
||||
<div class="1"></div>
|
||||
<div class="2"></div>
|
||||
<div class="3"></div>
|
||||
<div class="4"></div>
|
||||
</div>
|
||||
<div class="crane craneThree">
|
||||
<div class="line lineOne"></div>
|
||||
<div class="line lineTwo"></div>
|
||||
<div class="line lineThree"></div>
|
||||
<div class="stand"></div>
|
||||
<div class="weight"></div>
|
||||
<div class="cabin"></div>
|
||||
<div class="arm"></div>
|
||||
</div>
|
||||
<div class="crane craneTwo">
|
||||
<div class="line lineOne"></div>
|
||||
<div class="line lineTwo"></div>
|
||||
<div class="line lineThree"></div>
|
||||
<div class="stand"></div>
|
||||
<div class="weight"></div>
|
||||
<div class="cabin"></div>
|
||||
<div class="arm"></div>
|
||||
</div>
|
||||
<div class="crane craneOne">
|
||||
<div class="line lineOne"></div>
|
||||
<div class="line lineTwo"></div>
|
||||
<div class="line lineThree"></div>
|
||||
<div class="stand"></div>
|
||||
<div class="weight"></div>
|
||||
<div class="cabin"></div>
|
||||
<div class="arm"></div>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
`
|
||||
}
|
Loading…
Reference in a new issue