2019-10-31 04:28:29 +08:00
|
|
|
import { equal, match, alias } from "@ember/object/computed";
|
2019-10-30 01:31:44 +08:00
|
|
|
import { inject } from "@ember/controller";
|
2019-10-24 01:06:54 +08:00
|
|
|
import Controller from "@ember/controller";
|
2019-02-21 03:58:31 +08:00
|
|
|
import ModalFunctionality from "discourse/mixins/modal-functionality";
|
|
|
|
import { ajax } from "discourse/lib/ajax";
|
|
|
|
import { popupAjaxError } from "discourse/lib/ajax-error";
|
|
|
|
import {
|
2019-11-08 05:38:28 +08:00
|
|
|
default as discourseComputed,
|
2019-02-21 03:58:31 +08:00
|
|
|
observes
|
2019-11-08 05:38:28 +08:00
|
|
|
} from "discourse-common/utils/decorators";
|
2019-02-21 03:58:31 +08:00
|
|
|
import { THEMES, COMPONENTS } from "admin/models/theme";
|
2019-06-03 22:47:17 +08:00
|
|
|
import { POPULAR_THEMES } from "discourse-common/helpers/popular-themes";
|
2019-11-06 02:43:49 +08:00
|
|
|
import { set } from "@ember/object";
|
2019-02-21 03:58:31 +08:00
|
|
|
|
|
|
|
const MIN_NAME_LENGTH = 4;
|
|
|
|
|
2019-10-24 01:06:54 +08:00
|
|
|
export default Controller.extend(ModalFunctionality, {
|
2019-10-31 04:28:29 +08:00
|
|
|
popular: equal("selection", "popular"),
|
|
|
|
local: equal("selection", "local"),
|
|
|
|
remote: equal("selection", "remote"),
|
|
|
|
create: equal("selection", "create"),
|
2019-02-21 03:58:31 +08:00
|
|
|
selection: "popular",
|
2019-10-30 01:13:31 +08:00
|
|
|
adminCustomizeThemes: inject(),
|
2019-02-21 03:58:31 +08:00
|
|
|
loading: false,
|
|
|
|
keyGenUrl: "/admin/themes/generate_key_pair",
|
|
|
|
importUrl: "/admin/themes/import",
|
2019-02-28 18:48:30 +08:00
|
|
|
recordType: "theme",
|
2019-10-31 04:28:29 +08:00
|
|
|
checkPrivate: match("uploadUrl", /^git/),
|
2019-02-21 03:58:31 +08:00
|
|
|
localFile: null,
|
|
|
|
uploadUrl: null,
|
|
|
|
urlPlaceholder: "https://github.com/discourse/sample_theme",
|
|
|
|
advancedVisible: false,
|
2019-10-30 01:13:31 +08:00
|
|
|
themesController: inject("adminCustomizeThemes"),
|
2019-10-31 04:28:29 +08:00
|
|
|
selectedType: alias("themesController.currentTab"),
|
|
|
|
component: equal("selectedType", COMPONENTS),
|
2019-02-21 03:58:31 +08:00
|
|
|
|
2019-05-28 18:15:12 +08:00
|
|
|
init() {
|
|
|
|
this._super(...arguments);
|
|
|
|
|
|
|
|
this.createTypes = [
|
|
|
|
{ name: I18n.t("admin.customize.theme.theme"), value: THEMES },
|
|
|
|
{ name: I18n.t("admin.customize.theme.component"), value: COMPONENTS }
|
|
|
|
];
|
|
|
|
},
|
|
|
|
|
2019-11-08 05:38:28 +08:00
|
|
|
@discourseComputed("themesController.installedThemes")
|
2019-02-21 03:58:31 +08:00
|
|
|
themes(installedThemes) {
|
|
|
|
return POPULAR_THEMES.map(t => {
|
|
|
|
if (installedThemes.includes(t.name)) {
|
2019-11-06 02:43:49 +08:00
|
|
|
set(t, "installed", true);
|
2019-02-21 03:58:31 +08:00
|
|
|
}
|
|
|
|
return t;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2019-11-08 05:38:28 +08:00
|
|
|
@discourseComputed(
|
2019-02-21 03:58:31 +08:00
|
|
|
"loading",
|
|
|
|
"remote",
|
|
|
|
"uploadUrl",
|
|
|
|
"local",
|
|
|
|
"localFile",
|
|
|
|
"create",
|
|
|
|
"nameTooShort"
|
|
|
|
)
|
|
|
|
installDisabled(
|
|
|
|
isLoading,
|
|
|
|
isRemote,
|
|
|
|
uploadUrl,
|
|
|
|
isLocal,
|
|
|
|
localFile,
|
|
|
|
isCreate,
|
|
|
|
nameTooShort
|
|
|
|
) {
|
|
|
|
return (
|
|
|
|
isLoading ||
|
|
|
|
(isRemote && !uploadUrl) ||
|
|
|
|
(isLocal && !localFile) ||
|
|
|
|
(isCreate && nameTooShort)
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
@observes("privateChecked")
|
|
|
|
privateWasChecked() {
|
2019-05-27 16:15:39 +08:00
|
|
|
this.privateChecked
|
2019-02-21 03:58:31 +08:00
|
|
|
? this.set("urlPlaceholder", "git@github.com:discourse/sample_theme.git")
|
|
|
|
: this.set("urlPlaceholder", "https://github.com/discourse/sample_theme");
|
|
|
|
|
2019-05-27 16:15:39 +08:00
|
|
|
const checked = this.privateChecked;
|
2019-02-21 03:58:31 +08:00
|
|
|
if (checked && !this._keyLoading) {
|
|
|
|
this._keyLoading = true;
|
2019-05-27 16:15:39 +08:00
|
|
|
ajax(this.keyGenUrl, { method: "POST" })
|
2019-02-21 03:58:31 +08:00
|
|
|
.then(pair => {
|
2019-02-21 05:22:38 +08:00
|
|
|
this.setProperties({
|
|
|
|
privateKey: pair.private_key,
|
|
|
|
publicKey: pair.public_key
|
|
|
|
});
|
2019-02-21 03:58:31 +08:00
|
|
|
})
|
|
|
|
.catch(popupAjaxError)
|
|
|
|
.finally(() => {
|
|
|
|
this._keyLoading = false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-11-08 05:38:28 +08:00
|
|
|
@discourseComputed("name")
|
2019-02-21 03:58:31 +08:00
|
|
|
nameTooShort(name) {
|
|
|
|
return !name || name.length < MIN_NAME_LENGTH;
|
|
|
|
},
|
|
|
|
|
2019-11-08 05:38:28 +08:00
|
|
|
@discourseComputed("component")
|
2019-02-21 03:58:31 +08:00
|
|
|
placeholder(component) {
|
|
|
|
if (component) {
|
|
|
|
return I18n.t("admin.customize.theme.component_name");
|
|
|
|
} else {
|
|
|
|
return I18n.t("admin.customize.theme.theme_name");
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-11-08 05:38:28 +08:00
|
|
|
@discourseComputed("selection")
|
2019-02-21 03:58:31 +08:00
|
|
|
submitLabel(selection) {
|
|
|
|
return `admin.customize.theme.${
|
|
|
|
selection === "create" ? "create" : "install"
|
|
|
|
}`;
|
|
|
|
},
|
|
|
|
|
2019-11-08 05:38:28 +08:00
|
|
|
@discourseComputed("privateChecked", "checkPrivate", "publicKey")
|
2019-02-21 03:58:31 +08:00
|
|
|
showPublicKey(privateChecked, checkPrivate, publicKey) {
|
|
|
|
return privateChecked && checkPrivate && publicKey;
|
|
|
|
},
|
|
|
|
|
|
|
|
actions: {
|
|
|
|
uploadLocaleFile() {
|
|
|
|
this.set("localFile", $("#file-input")[0].files[0]);
|
|
|
|
},
|
|
|
|
|
|
|
|
toggleAdvanced() {
|
|
|
|
this.toggleProperty("advancedVisible");
|
|
|
|
},
|
|
|
|
|
|
|
|
installThemeFromList(url) {
|
|
|
|
this.set("uploadUrl", url);
|
|
|
|
this.send("installTheme");
|
|
|
|
},
|
|
|
|
|
|
|
|
installTheme() {
|
2019-05-27 16:15:39 +08:00
|
|
|
if (this.create) {
|
2019-02-21 03:58:31 +08:00
|
|
|
this.set("loading", true);
|
2019-05-27 16:15:39 +08:00
|
|
|
const theme = this.store.createRecord(this.recordType);
|
2019-02-21 03:58:31 +08:00
|
|
|
theme
|
2019-05-27 16:15:39 +08:00
|
|
|
.save({ name: this.name, component: this.component })
|
2019-02-21 03:58:31 +08:00
|
|
|
.then(() => {
|
2019-05-27 16:15:39 +08:00
|
|
|
this.themesController.send("addTheme", theme);
|
2019-02-21 03:58:31 +08:00
|
|
|
this.send("closeModal");
|
|
|
|
})
|
|
|
|
.catch(popupAjaxError)
|
|
|
|
.finally(() => this.set("loading", false));
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let options = {
|
|
|
|
type: "POST"
|
|
|
|
};
|
|
|
|
|
2019-05-27 16:15:39 +08:00
|
|
|
if (this.local) {
|
2019-02-21 03:58:31 +08:00
|
|
|
options.processData = false;
|
|
|
|
options.contentType = false;
|
|
|
|
options.data = new FormData();
|
2019-05-27 16:15:39 +08:00
|
|
|
options.data.append("theme", this.localFile);
|
2019-02-21 03:58:31 +08:00
|
|
|
}
|
|
|
|
|
2019-05-27 16:15:39 +08:00
|
|
|
if (this.remote || this.popular) {
|
2019-02-21 03:58:31 +08:00
|
|
|
options.data = {
|
2019-05-27 16:15:39 +08:00
|
|
|
remote: this.uploadUrl,
|
|
|
|
branch: this.branch
|
2019-02-21 03:58:31 +08:00
|
|
|
};
|
|
|
|
|
2019-05-27 16:15:39 +08:00
|
|
|
if (this.privateChecked) {
|
|
|
|
options.data.private_key = this.privateKey;
|
2019-02-21 03:58:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.get("model.user_id")) {
|
|
|
|
// Used by theme-creator
|
|
|
|
options.data["user_id"] = this.get("model.user_id");
|
|
|
|
}
|
|
|
|
|
|
|
|
this.set("loading", true);
|
2019-05-27 16:15:39 +08:00
|
|
|
ajax(this.importUrl, options)
|
2019-02-21 03:58:31 +08:00
|
|
|
.then(result => {
|
2019-05-27 16:42:53 +08:00
|
|
|
const theme = this.store.createRecord(this.recordType, result.theme);
|
2019-05-27 16:15:39 +08:00
|
|
|
this.adminCustomizeThemes.send("addTheme", theme);
|
2019-02-21 03:58:31 +08:00
|
|
|
this.send("closeModal");
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
this.setProperties({ privateKey: null, publicKey: null });
|
|
|
|
})
|
|
|
|
.catch(popupAjaxError)
|
|
|
|
.finally(() => this.set("loading", false));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|