1
0
mirror of https://github.com/hrfee/jfa-go.git synced 2025-01-01 14:00:12 +00:00

Compare commits

..

2 Commits

Author SHA1 Message Date
a3f5396211
parse and render settings menus 2021-01-02 21:39:36 +00:00
d629cae71e
change config-base format to allow easier parsing
*note*: only GetConfig has been changed for now.
the config now has "order" and "sections", and each section now has
"meta", "order" and "settings". This allows for the use of a struct for
loading in Go and less janky web code. Also now appears in swagger.
2021-01-02 21:24:26 +00:00
11 changed files with 1252 additions and 737 deletions

1
.gitignore vendored
View File

@ -5,4 +5,5 @@ build/
version.go version.go
notes notes
docs/* docs/*
config-payload.json
!docs/go.mod !docs/go.mod

49
api.go
View File

@ -1072,13 +1072,14 @@ func (app *appContext) ApplySettings(gc *gin.Context) {
// @Summary Get jfa-go configuration. // @Summary Get jfa-go configuration.
// @Produce json // @Produce json
// @Success 200 {object} configDTO "Uses the same format as config-base.json" // @Success 200 {object} settings "Uses the same format as config-base.json"
// @Router /config [get] // @Router /config [get]
// @Security Bearer // @Security Bearer
// @tags Configuration // @tags Configuration
func (app *appContext) GetConfig(gc *gin.Context) { func (app *appContext) GetConfig(gc *gin.Context) {
app.info.Println("Config requested") app.info.Println("Config requested")
resp := map[string]interface{}{} resp := app.configBase
// Load language options
langPath := filepath.Join(app.localPath, "lang", "form") langPath := filepath.Join(app.localPath, "lang", "form")
app.lang.langFiles, _ = ioutil.ReadDir(langPath) app.lang.langFiles, _ = ioutil.ReadDir(langPath)
app.lang.langOptions = make([]string, len(app.lang.langFiles)) app.lang.langOptions = make([]string, len(app.lang.langFiles))
@ -1095,37 +1096,25 @@ func (app *appContext) GetConfig(gc *gin.Context) {
app.lang.langOptions[i] = meta.(map[string]interface{})["name"].(string) app.lang.langOptions[i] = meta.(map[string]interface{})["name"].(string)
} }
} }
for section, settings := range app.configBase { s := resp.Sections["ui"].Settings["language"]
if section == "order" { s.Options = app.lang.langOptions
resp[section] = settings.([]interface{}) s.Value = app.lang.langOptions[app.lang.chosenIndex]
} else { resp.Sections["ui"].Settings["language"] = s
resp[section] = make(map[string]interface{}) for sectName, section := range resp.Sections {
for key, values := range settings.(map[string]interface{}) { for settingName, setting := range section.Settings {
if key == "order" { val := app.config.Section(sectName).Key(settingName)
resp[section].(map[string]interface{})[key] = values.([]interface{}) s := resp.Sections[sectName].Settings[settingName]
} else { switch setting.Type {
resp[section].(map[string]interface{})[key] = values.(map[string]interface{}) case "text", "email", "select":
if key != "meta" { s.Value = val.MustString("")
dataType := resp[section].(map[string]interface{})[key].(map[string]interface{})["type"].(string) case "number":
configKey := app.config.Section(section).Key(key) s.Value = val.MustInt(0)
if dataType == "number" { case "bool":
if val, err := configKey.Int(); err == nil { s.Value = val.MustBool(false)
resp[section].(map[string]interface{})[key].(map[string]interface{})["value"] = val
} }
} else if dataType == "bool" { resp.Sections[sectName].Settings[settingName] = s
resp[section].(map[string]interface{})[key].(map[string]interface{})["value"] = configKey.MustBool(false)
} else if dataType == "select" && key == "language" {
resp[section].(map[string]interface{})[key].(map[string]interface{})["options"] = app.lang.langOptions
resp[section].(map[string]interface{})[key].(map[string]interface{})["value"] = app.lang.langOptions[app.lang.chosenIndex]
} else {
resp[section].(map[string]interface{})[key].(map[string]interface{})["value"] = configKey.String()
} }
} }
}
}
}
}
// resp["jellyfin"].(map[string]interface{})["language"].(map[string]interface{})["options"].([]string)
gc.JSON(200, resp) gc.JSON(200, resp)
} }

View File

@ -1,9 +1,13 @@
{ {
"order": [],
"sections": {
"jellyfin": { "jellyfin": {
"order": [],
"meta": { "meta": {
"name": "Jellyfin", "name": "Jellyfin",
"description": "Settings for connecting to Jellyfin" "description": "Settings for connecting to Jellyfin"
}, },
"settings": {
"username": { "username": {
"name": "Jellyfin Username", "name": "Jellyfin Username",
"required": true, "required": true,
@ -51,12 +55,15 @@
"value": 30, "value": 30,
"description": "Timeout of user cache in minutes. Set to 0 to disable." "description": "Timeout of user cache in minutes. Set to 0 to disable."
} }
}
}, },
"ui": { "ui": {
"order": [],
"meta": { "meta": {
"name": "General", "name": "General",
"description": "Settings related to the UI and program functionality." "description": "Settings related to the UI and program functionality."
}, },
"settings": {
"language": { "language": {
"name": "Language", "name": "Language",
"required": false, "required": false,
@ -188,12 +195,15 @@
"value": "", "value": "",
"description": "URL base for when running jfa-go with a reverse proxy in a subfolder." "description": "URL base for when running jfa-go with a reverse proxy in a subfolder."
} }
}
}, },
"password_validation": { "password_validation": {
"order": [],
"meta": { "meta": {
"name": "Password Validation", "name": "Password Validation",
"description": "Password validation (minimum length, etc.)" "description": "Password validation (minimum length, etc.)"
}, },
"settings": {
"enabled": { "enabled": {
"name": "Enabled", "name": "Enabled",
"required": false, "required": false,
@ -236,12 +246,15 @@
"type": "text", "type": "text",
"value": "0" "value": "0"
} }
}
}, },
"email": { "email": {
"order": [],
"meta": { "meta": {
"name": "Email", "name": "Email",
"description": "General email settings. Ignore if not using email features." "description": "General email settings. Ignore if not using email features."
}, },
"settings": {
"no_username": { "no_username": {
"name": "Use email addresses as username", "name": "Use email addresses as username",
"required": false, "required": false,
@ -307,12 +320,15 @@
"value": "Jellyfin", "value": "Jellyfin",
"description": "The name of the sender" "description": "The name of the sender"
} }
}
}, },
"password_resets": { "password_resets": {
"order": [],
"meta": { "meta": {
"name": "Password Resets", "name": "Password Resets",
"description": "Settings for the password reset handler." "description": "Settings for the password reset handler."
}, },
"settings": {
"enabled": { "enabled": {
"name": "Enabled", "name": "Enabled",
"required": false, "required": false,
@ -357,12 +373,15 @@
"value": "Password Reset - Jellyfin", "value": "Password Reset - Jellyfin",
"description": "Subject of password reset emails." "description": "Subject of password reset emails."
} }
}
}, },
"invite_emails": { "invite_emails": {
"order": [],
"meta": { "meta": {
"name": "Invite emails", "name": "Invite emails",
"description": "Settings for sending invites directly to users." "description": "Settings for sending invites directly to users."
}, },
"settings": {
"enabled": { "enabled": {
"name": "Enabled", "name": "Enabled",
"required": false, "required": false,
@ -406,12 +425,15 @@
"value": "http://accounts.jellyf.in:8056/invite", "value": "http://accounts.jellyf.in:8056/invite",
"description": "Base URL for jfa-go. This is necessary because using a reverse proxy means the program has no way of knowing the URL itself." "description": "Base URL for jfa-go. This is necessary because using a reverse proxy means the program has no way of knowing the URL itself."
} }
}
}, },
"notifications": { "notifications": {
"order": [],
"meta": { "meta": {
"name": "Notifications", "name": "Notifications",
"description": "Notification related settings." "description": "Notification related settings."
}, },
"settings": {
"enabled": { "enabled": {
"name": "Enabled", "name": "Enabled",
"required": "false", "required": "false",
@ -456,12 +478,15 @@
"value": "", "value": "",
"description": "Path to user creation notification email in plaintext." "description": "Path to user creation notification email in plaintext."
} }
}
}, },
"mailgun": { "mailgun": {
"order": [],
"meta": { "meta": {
"name": "Mailgun (Email)", "name": "Mailgun (Email)",
"description": "Mailgun API connection settings" "description": "Mailgun API connection settings"
}, },
"settings": {
"api_url": { "api_url": {
"name": "API URL", "name": "API URL",
"required": false, "required": false,
@ -476,12 +501,15 @@
"type": "text", "type": "text",
"value": "your api key" "value": "your api key"
} }
}
}, },
"smtp": { "smtp": {
"order": [],
"meta": { "meta": {
"name": "SMTP (Email)", "name": "SMTP (Email)",
"description": "SMTP Server connection settings." "description": "SMTP Server connection settings."
}, },
"settings": {
"username": { "username": {
"name": "Username", "name": "Username",
"required": false, "required": false,
@ -524,12 +552,15 @@
"type": "password", "type": "password",
"value": "smtp password" "value": "smtp password"
} }
}
}, },
"ombi": { "ombi": {
"order": [],
"meta": { "meta": {
"name": "Ombi Integration", "name": "Ombi Integration",
"description": "Connect to Ombi to automatically create both Ombi and Jellyfin accounts for new users. You'll need to create a user template for this to work. Once enabled, refresh to see an option in settings for this." "description": "Connect to Ombi to automatically create both Ombi and Jellyfin accounts for new users. You'll need to create a user template for this to work. Once enabled, refresh to see an option in settings for this."
}, },
"settings": {
"enabled": { "enabled": {
"name": "Enabled", "name": "Enabled",
"required": false, "required": false,
@ -556,12 +587,15 @@
"depends_true": "enabled", "depends_true": "enabled",
"description": "API Key. Get this from the first tab in Ombi settings." "description": "API Key. Get this from the first tab in Ombi settings."
} }
}
}, },
"deletion": { "deletion": {
"order": [],
"meta": { "meta": {
"name": "Account Deletion", "name": "Account Deletion",
"description": "Subject/email files for account deletion emails." "description": "Subject/email files for account deletion emails."
}, },
"settings": {
"subject": { "subject": {
"name": "Email subject", "name": "Email subject",
"required": false, "required": false,
@ -586,12 +620,15 @@
"value": "", "value": "",
"description": "Path to custom email in plain text" "description": "Path to custom email in plain text"
} }
}
}, },
"files": { "files": {
"order": [],
"meta": { "meta": {
"name": "File Storage", "name": "File Storage",
"description": "Optional settings for changing storage locations." "description": "Optional settings for changing storage locations."
}, },
"settings": {
"invites": { "invites": {
"name": "Invite Storage", "name": "Invite Storage",
"required": false, "required": false,
@ -666,3 +703,5 @@
} }
} }
} }
}
}

View File

@ -9,17 +9,17 @@ args = parser.parse_args()
with open(args.input, 'r') as f: with open(args.input, 'r') as f:
config = json.load(f) config = json.load(f)
newconfig = {"order": []} newconfig = {"sections": {}, "order": []}
for sect in config: for sect in config["sections"]:
newconfig["order"].append(sect) newconfig["order"].append(sect)
newconfig[sect] = {} newconfig["sections"][sect] = {}
newconfig[sect]["order"] = [] newconfig["sections"][sect]["order"] = []
newconfig[sect]["meta"] = config[sect]["meta"] newconfig["sections"][sect]["meta"] = config["sections"][sect]["meta"]
for setting in config[sect]: newconfig["sections"][sect]["settings"] = {}
if setting != "meta": for setting in config["sections"][sect]["settings"]:
newconfig[sect]["order"].append(setting) newconfig["sections"][sect]["order"].append(setting)
newconfig[sect][setting] = config[sect][setting] newconfig["sections"][sect]["settings"][setting] = config["sections"][sect]["settings"][setting]
with open(args.output, 'w') as f: with open(args.output, 'w') as f:
f.write(json.dumps(newconfig, indent=4)) f.write(json.dumps(newconfig, indent=4))

View File

@ -14,13 +14,14 @@ def generate_ini(base_file, ini_file):
ini = configparser.RawConfigParser(allow_no_value=True) ini = configparser.RawConfigParser(allow_no_value=True)
for section in config_base: for section in config_base["sections"]:
ini.add_section(section) ini.add_section(section)
for entry in config_base[section]: if "meta" in config_base["sections"][section]:
if "description" in config_base[section][entry]: ini.set(section, "; " + config_base["sections"][section]["meta"]["description"])
ini.set(section, "; " + config_base[section][entry]["description"]) for entry in config_base["sections"][section]["settings"]:
if entry != "meta": if "description" in config_base["sections"][section]["settings"][entry]:
value = config_base[section][entry]["value"] ini.set(section, "; " + config_base["sections"][section]["settings"][entry]["description"])
value = config_base["sections"][section]["settings"][entry]["value"]
if isinstance(value, bool): if isinstance(value, bool):
value = str(value).lower() value = str(value).lower()
else: else:

View File

@ -196,6 +196,11 @@ sup.\~critical, .text-critical {
align-items: center; align-items: center;
} }
.inv-empty .inv-codearea {
justify-content: start;
}
.invite-link { .invite-link {
text-overflow: ellipsis; text-overflow: ellipsis;
overflow: hidden; overflow: hidden;

View File

@ -236,40 +236,11 @@
<span class="button ~neutral !normal" id="accounts-add-user">Save</span> <span class="button ~neutral !normal" id="accounts-add-user">Save</span>
</div> </div>
<div class="row"> <div class="row">
<div class="card ~neutral !normal col"> <div class="card ~neutral !normal col" id="settings-sidebar">
<aside class="aside sm ~info mb-half">Note: <span class="badge ~critical">*</span> indicates a required field, <span class="badge ~critical">R</span> indicates changes require a restart.</aside> <aside class="aside sm ~info mb-half">Note: <span class="badge ~critical">*</span> indicates a required field, <span class="badge ~critical">R</span> indicates changes require a restart.</aside>
<span class="button ~neutral !low settings-section-button mb-half" id="setting-about">About</span> <span class="button ~neutral !low settings-section-button mb-half" id="setting-about">About</span>
<span class="button ~neutral !low settings-section-button mb-half selected">User Profiles</span>
</div>
<div class="card ~neutral !normal col">
<div class="settings-section">
<p class="support lg mb-half">Settings section description.</p>
<div class="setting">
<label class="label" for="settings-select">Select <span class="badge ~critical">R</span></label>
<div class="select ~neutral !normal mt-half">
<select id="settings-select">
<option>Option 1</option>
</select>
</div>
</div>
<div class="setting">
<label class="label" for="settings-input">
Input <span class="badge ~critical">*</span>
<div class="tooltip right">
<i class="icon ri-information-line"></i>
<span class="content sm">An example tooltip.</span>
</div>
</label>
<input type="text" class="input ~neutral !normal mt-half" placeholder="Value">
</div>
<div class="setting">
<label class="switch settings">
<input type="checkbox" id="settings-check">
<span>Checkbox <span class="badge ~critical">R</span></span>
</label>
</div>
</div>
</div> </div>
<div class="card ~neutral !normal col" id="settings-panel"></div>
</div> </div>
</div> </div>
</div> </div>

View File

@ -48,7 +48,7 @@ type appContext struct {
config *ini.File config *ini.File
configPath string configPath string
configBasePath string configBasePath string
configBase map[string]interface{} configBase settings
dataPath string dataPath string
localPath string localPath string
cssFile string cssFile string

View File

@ -127,3 +127,32 @@ type errorListDTO map[string]map[string]string
type configDTO map[string]interface{} type configDTO map[string]interface{}
// Below are for sending config
type meta struct {
Name string `json:"name"`
Description string `json:"description"`
}
type setting struct {
Name string `json:"name"`
Description string `json:"description"`
Required bool `json:"required"`
RequiresRestart bool `json:"requires_restart"`
Type string `json:"type"` // Type (string, number, bool, etc.)
Value interface{} `json:"value"`
Options []string `json:"options,omitempty"`
DependsTrue string `json:"depends_true,omitempty"` // If specified, this field is enabled when the specified bool setting is enabled.
DependsFalse string `json:"depends_false,omitempty"` // If specified, opposite behaviour of DependsTrue.
}
type section struct {
Meta meta `json:"meta"`
Order []string `json:"order"`
Settings map[string]setting `json:"settings"`
}
type settings struct {
Order []string `json:"order"`
Sections map[string]section `json:"sections"`
}

View File

@ -3,6 +3,7 @@ import { Modal } from "./modules/modal.js";
import { Tabs } from "./modules/tabs.js"; import { Tabs } from "./modules/tabs.js";
import { inviteList, createInvite } from "./modules/invites.js"; import { inviteList, createInvite } from "./modules/invites.js";
import { accountsList } from "./modules/accounts.js"; import { accountsList } from "./modules/accounts.js";
import { settingsList } from "./modules/settings.js";
import { _post, notificationBox, whichAnimationEvent, toggleLoader } from "./modules/common.js"; import { _post, notificationBox, whichAnimationEvent, toggleLoader } from "./modules/common.js";
loadTheme(); loadTheme();
@ -42,6 +43,8 @@ var accounts = new accountsList();
window.invites = new inviteList(); window.invites = new inviteList();
var settings = new settingsList();
window.notifications = new notificationBox(document.getElementById('notification-box') as HTMLDivElement, 5); window.notifications = new notificationBox(document.getElementById('notification-box') as HTMLDivElement, 5);
/*const modifySettingsSource = function () { /*const modifySettingsSource = function () {
@ -59,10 +62,9 @@ window.notifications = new notificationBox(document.getElementById('notification
// load tabs // load tabs
window.tabs = new Tabs(); window.tabs = new Tabs();
for (let tabID of ["invitesTab", "settingsTab"]) { window.tabs.addTab("invitesTab");
window.tabs.addTab(tabID);
}
window.tabs.addTab("accountsTab", null, accounts.reload); window.tabs.addTab("accountsTab", null, accounts.reload);
window.tabs.addTab("settingsTab", null, settings.reload);
function login(username: string, password: string, run?: (state?: number) => void) { function login(username: string, password: string, run?: (state?: number) => void) {
const req = new XMLHttpRequest(); const req = new XMLHttpRequest();

478
ts/modules/settings.ts Normal file
View File

@ -0,0 +1,478 @@
import { _get } from "../modules/common.js";
interface settingsBoolEvent extends Event {
detail: boolean;
}
interface Profile {
Admin: boolean;
LibraryAccess: string;
FromUser: string;
}
interface Meta {
name: string;
description: string;
}
interface Setting {
name: string;
description: string;
required: boolean;
requires_restart: boolean;
type: string;
value: string | boolean | number;
depends_true?: Setting;
depends_false?: Setting;
asElement: () => HTMLElement;
update: (s: Setting) => void;
}
class DOMInput {
protected _input: HTMLInputElement;
private _container: HTMLDivElement;
private _tooltip: HTMLDivElement;
private _required: HTMLSpanElement;
private _restart: HTMLSpanElement;
get name(): string { return this._container.querySelector("span.setting-label").textContent; }
set name(n: string) { this._container.querySelector("span.setting-label").textContent = n; }
get description(): string { return this._tooltip.querySelector("span.content").textContent; }
set description(d: string) {
const content = this._tooltip.querySelector("span.content") as HTMLSpanElement;
content.textContent = d;
if (d == "") {
this._tooltip.classList.add("unfocused");
} else {
this._tooltip.classList.remove("unfocused");
}
}
get required(): boolean { return this._required.classList.contains("badge"); }
set required(state: boolean) {
if (state) {
this._required.classList.add("badge", "~critical");
this._required.textContent = "*";
} else {
this._required.classList.remove("badge", "~critical");
this._required.textContent = "";
}
}
get requires_restart(): boolean { return this._restart.classList.contains("badge"); }
set requires_restart(state: boolean) {
if (state) {
this._restart.classList.add("badge", "~critical");
this._restart.textContent = "R";
} else {
this._restart.classList.remove("badge", "~critical");
this._restart.textContent = "";
}
}
constructor(inputType: string, setting: Setting, section: string) {
this._container = document.createElement("div");
this._container.classList.add("setting");
this._container.innerHTML = `
<label class="label">
<span class="setting-label"></span> <span class="setting-required"></span> <span class="setting-restart"></span>
<div class="setting-tooltip tooltip right unfocused">
<i class="icon ri-information-line"></i>
<span class="content sm"></span>
</div>
<input type="${inputType}" class="input ~neutral !normal mt-half">
</label>
`;
this._tooltip = this._container.querySelector("div.setting-tooltip") as HTMLDivElement;
this._required = this._container.querySelector("span.setting-required") as HTMLSpanElement;
this._restart = this._container.querySelector("span.setting-restart") as HTMLSpanElement;
this._input = this._container.querySelector("input[type=" + inputType + "]") as HTMLInputElement;
if (setting.depends_false || setting.depends_true) {
let dependant = setting.depends_true || setting.depends_false;
let state = true;
if (setting.depends_false) { state = false; }
document.addEventListener(`settings-${section}-${dependant}`, (event: settingsBoolEvent) => {
this._input.disabled = (event.detail !== state);
});
}
this.update(setting);
}
get value(): any { return this._input.value; }
set value(v: any) { this._input.value = v; }
update = (s: Setting) => {
this.name = s.name;
this.description = s.description;
this.required = s.required;
this.requires_restart = s.requires_restart;
this.value = s.value;
}
asElement = (): HTMLDivElement => { return this._container; }
}
interface SText extends Setting {
value: string;
}
class DOMText extends DOMInput implements SText {
constructor(setting: Setting, section: string) { super("text", setting, section); }
type: string = "text";
get value(): string { return this._input.value }
set value(v: string) { this._input.value = v; }
}
interface SPassword extends Setting {
value: string;
}
class DOMPassword extends DOMInput implements SPassword {
constructor(setting: Setting, section: string) { super("password", setting, section); }
type: string = "password";
get value(): string { return this._input.value }
set value(v: string) { this._input.value = v; }
}
interface SEmail extends Setting {
value: string;
}
class DOMEmail extends DOMInput implements SEmail {
constructor(setting: Setting, section: string) { super("email", setting, section); }
type: string = "email";
get value(): string { return this._input.value }
set value(v: string) { this._input.value = v; }
}
interface SNumber extends Setting {
value: number;
}
class DOMNumber extends DOMInput implements SNumber {
constructor(setting: Setting, section: string) { super("number", setting, section); }
type: string = "number";
get value(): number { return +this._input.value; }
set value(v: number) { this._input.value = ""+v; }
}
interface SBool extends Setting {
value: boolean;
}
class DOMBool implements SBool {
protected _input: HTMLInputElement;
private _container: HTMLDivElement;
private _tooltip: HTMLDivElement;
private _required: HTMLSpanElement;
private _restart: HTMLSpanElement;
type: string = "bool";
get name(): string { return this._container.querySelector("span.setting-label").textContent; }
set name(n: string) { this._container.querySelector("span.setting-label").textContent = n; }
get description(): string { return this._tooltip.querySelector("span.content").textContent; }
set description(d: string) {
const content = this._tooltip.querySelector("span.content") as HTMLSpanElement;
content.textContent = d;
if (d == "") {
this._tooltip.classList.add("unfocused");
} else {
this._tooltip.classList.remove("unfocused");
}
}
get required(): boolean { return this._required.classList.contains("badge"); }
set required(state: boolean) {
if (state) {
this._required.classList.add("badge", "~critical");
this._required.textContent = "*";
} else {
this._required.classList.remove("badge", "~critical");
this._required.textContent = "";
}
}
get requires_restart(): boolean { return this._restart.classList.contains("badge"); }
set requires_restart(state: boolean) {
if (state) {
this._restart.classList.add("badge", "~critical");
this._restart.textContent = "R";
} else {
this._restart.classList.remove("badge", "~critical");
this._restart.textContent = "";
}
}
get value(): boolean { return this._input.checked; }
set value(state: boolean) { this._input.checked = state; }
constructor(setting: SBool, section: string, name: string) {
this._container = document.createElement("div");
this._container.classList.add("setting");
this._container.innerHTML = `
<label class="switch">
<input type="checkbox">
<span class="setting-label"></span> <span class="setting-required"></span> <span class="setting-restart"></span>
<div class="setting-tooltip tooltip right unfocused">
<i class="icon ri-information-line"></i>
<span class="content sm"></span>
</div>
</label>
`;
this._tooltip = this._container.querySelector("div.setting-tooltip") as HTMLDivElement;
this._required = this._container.querySelector("span.setting-required") as HTMLSpanElement;
this._restart = this._container.querySelector("span.setting-restart") as HTMLSpanElement;
this._input = this._container.querySelector("input[type=checkbox]") as HTMLInputElement;
const onValueChange = () => {
const event = new CustomEvent(`settings-${section}-${name}`, { "detail": this._input.checked })
document.dispatchEvent(event);
};
this._input.onchange = onValueChange;
document.addEventListener(`settings-loaded`, onValueChange);
if (setting.depends_false || setting.depends_true) {
let dependant = setting.depends_true || setting.depends_false;
let state = true;
if (setting.depends_false) { state = false; }
document.addEventListener(`settings-${section}-${dependant}`, (event: settingsBoolEvent) => {
this._input.disabled = (event.detail !== state);
});
}
this.update(setting);
}
update = (s: SBool) => {
this.name = s.name;
this.description = s.description;
this.required = s.required;
this.requires_restart = s.requires_restart;
this.value = s.value;
}
asElement = (): HTMLDivElement => { return this._container; }
}
interface SSelect extends Setting {
options: string[];
value: string;
}
class DOMSelect implements SSelect {
protected _select: HTMLSelectElement;
private _container: HTMLDivElement;
private _tooltip: HTMLDivElement;
private _required: HTMLSpanElement;
private _restart: HTMLSpanElement;
private _options: string[];
type: string = "bool";
get name(): string { return this._container.querySelector("span.setting-label").textContent; }
set name(n: string) { this._container.querySelector("span.setting-label").textContent = n; }
get description(): string { return this._tooltip.querySelector("span.content").textContent; }
set description(d: string) {
const content = this._tooltip.querySelector("span.content") as HTMLSpanElement;
content.textContent = d;
if (d == "") {
this._tooltip.classList.add("unfocused");
} else {
this._tooltip.classList.remove("unfocused");
}
}
get required(): boolean { return this._required.classList.contains("badge"); }
set required(state: boolean) {
if (state) {
this._required.classList.add("badge", "~critical");
this._required.textContent = "*";
} else {
this._required.classList.remove("badge", "~critical");
this._required.textContent = "";
}
}
get requires_restart(): boolean { return this._restart.classList.contains("badge"); }
set requires_restart(state: boolean) {
if (state) {
this._restart.classList.add("badge", "~critical");
this._restart.textContent = "R";
} else {
this._restart.classList.remove("badge", "~critical");
this._restart.textContent = "";
}
}
get value(): string { return this._select.value; }
set value(v: string) { this._select.value = v; }
get options(): string[] { return this._options; }
set options(opt: string[]) {
this._options = opt;
let innerHTML = "";
for (let option of this._options) {
innerHTML += `<option value="${option}">${option}</option>`;
}
this._select.innerHTML = innerHTML;
}
constructor(setting: SSelect, section: string) {
this._options = [];
this._container = document.createElement("div");
this._container.classList.add("setting");
this._container.innerHTML = `
<label class="label">
<span class="setting-label"></span> <span class="setting-required"></span> <span class="setting-restart"></span>
<div class="setting-tooltip tooltip right unfocused">
<i class="icon ri-information-line"></i>
<span class="content sm"></span>
</div>
<div class="select ~neutral !normal mt-half">
<select class="settings-select"></select>
</div>
</label>
`;
this._tooltip = this._container.querySelector("div.setting-tooltip") as HTMLDivElement;
this._required = this._container.querySelector("span.setting-required") as HTMLSpanElement;
this._restart = this._container.querySelector("span.setting-restart") as HTMLSpanElement;
this._select = this._container.querySelector("select.settings-select") as HTMLSelectElement;
if (setting.depends_false || setting.depends_true) {
let dependant = setting.depends_true || setting.depends_false;
let state = true;
if (setting.depends_false) { state = false; }
document.addEventListener(`settings-${section}-${dependant}`, (event: settingsBoolEvent) => {
this._input.disabled = (event.detail !== state);
});
}
this.update(setting);
}
update = (s: SSelect) => {
this.name = s.name;
this.description = s.description;
this.required = s.required;
this.requires_restart = s.requires_restart;
this.options = s.options;
this.value = s.value;
}
asElement = (): HTMLDivElement => { return this._container; }
}
interface Section {
meta: Meta;
order: string[];
settings: { [settingName: string]: Setting };
}
class sectionPanel {
private _section: HTMLDivElement;
private _settings: { [name: string]: Setting };
private _sectionName: string;
constructor(s: Section, sectionName: string) {
this._sectionName = sectionName;
this._settings = {};
this._section = document.createElement("div") as HTMLDivElement;
this._section.classList.add("settings-section", "unfocused");
this._section.innerHTML = `<p class="support lg mb-half">${s.meta.description}</p>`;
this.update(s);
}
update = (s: Section) => {
for (let name of s.order) {
let setting: Setting = s.settings[name];
if (name in this._settings) {
this._settings[name].update(setting);
} else {
switch (setting.type) {
case "text":
setting = new DOMText(setting, this._sectionName);
break;
case "password":
setting = new DOMPassword(setting, this._sectionName);
break;
case "email":
setting = new DOMEmail(setting, this._sectionName);
break;
case "number":
setting = new DOMNumber(setting, this._sectionName);
break;
case "bool":
setting = new DOMBool(setting as SBool, this._sectionName, name);
break;
case "select":
setting = new DOMSelect(setting as SSelect, this._sectionName);
break;
}
this._section.appendChild(setting.asElement());
this._settings[name] = setting;
}
}
}
get visible(): boolean { return !this._section.classList.contains("unfocused"); }
set visible(s: boolean) {
if (s) {
this._section.classList.remove("unfocused");
} else {
this._section.classList.add("unfocused");
}
}
asElement = (): HTMLDivElement => { return this._section; }
}
interface Settings {
order: string[];
sections: { [sectionName: string]: Section };
}
export class settingsList {
private _panel = document.getElementById("settings-panel") as HTMLDivElement;
private _sidebar = document.getElementById("settings-sidebar") as HTMLDivElement;
private _sections: { [name: string]: sectionPanel }
private _buttons: { [name: string]: HTMLSpanElement }
addSection = (name: string, s: Section) => {
const section = new sectionPanel(s, name);
this._sections[name] = section;
this._panel.appendChild(this._sections[name].asElement());
const button = document.createElement("span") as HTMLSpanElement;
button.classList.add("button", "~neutral", "!low", "settings-section-button", "mb-half");
button.textContent = s.meta.name;
button.onclick = () => { this._showPanel(name); };
this._buttons[name] = button;
this._sidebar.appendChild(this._buttons[name]);
}
private _showPanel = (name: string) => {
for (let n in this._sections) {
if (n == name) {
console.log("found", n);
this._sections[name].visible = true;
this._buttons[name].classList.add("selected");
} else {
this._sections[n].visible = false;
this._buttons[n].classList.remove("selected");
}
}
}
constructor() {
this._sections = {};
this._buttons = {};
}
reload = () => _get("/config", null, (req: XMLHttpRequest) => {
if (req.readyState == 4) {
if (req.status != 200) {
window.notifications.customError("settingsLoadError", "Failed to load settings.");
return;
}
let settings = req.response as Settings;
for (let name of settings.order) {
if (name in this._sections) {
this._sections[name].update(settings.sections[name]);
} else {
this.addSection(name, settings.sections[name]);
}
}
document.dispatchEvent(new CustomEvent("settings-loaded"));
}
})
}