UX: Show loading spinner while loading dependencies for ace-editor (#26099)

Why this change?

On a slow network, using the `AceEditor` component will result in a blob
of text being shown first before being swapped out with the `ace.js`
editor after it has completed loading.

There is also a problem when setting the theme for the editor which
would result in a "flash" as reported in
https://github.com/ajaxorg/ace/issues/3286. To avoid this, we need to
load the theme js file before displaying the editor.

What does this change do?

1. Adds a loading spinner and set the `div.ace` with a `.hidden` class.
2. Once all the relevant scripts and initialization is done, we will
   then remove the loading spinner and remove `div.ace`.
This commit is contained in:
Alan Guo Xiang Tan 2024-03-11 06:56:17 +08:00 committed by GitHub
parent f8964f8f8f
commit 7d8dd0d8e3
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 106 additions and 82 deletions

View File

@ -1 +1,5 @@
<div class="ace">{{this.content}}</div> {{#if this.isLoading}}
{{loading-spinner size="small"}}
{{else}}
<div class="ace">{{this.content}}</div>
{{/if}}

View File

@ -1,5 +1,6 @@
import Component from "@ember/component"; import Component from "@ember/component";
import { action } from "@ember/object"; import { action } from "@ember/object";
import { next } from "@ember/runloop";
import { classNames } from "@ember-decorators/component"; import { classNames } from "@ember-decorators/component";
import { observes, on } from "@ember-decorators/object"; import { observes, on } from "@ember-decorators/object";
import $ from "jquery"; import $ from "jquery";
@ -13,6 +14,7 @@ const COLOR_VARS_REGEX =
@classNames("ace-wrapper") @classNames("ace-wrapper")
export default class AceEditor extends Component { export default class AceEditor extends Component {
isLoading = true;
mode = "css"; mode = "css";
disabled = false; disabled = false;
htmlPlaceholder = false; htmlPlaceholder = false;
@ -95,67 +97,78 @@ export default class AceEditor extends Component {
didInsertElement() { didInsertElement() {
super.didInsertElement(...arguments); super.didInsertElement(...arguments);
loadScript("/javascripts/ace/ace.js").then(() => { loadScript("/javascripts/ace/ace.js").then(() => {
window.ace.require(["ace/ace"], (loadedAce) => { loadScript(`/javascripts/ace/theme-${this.aceTheme}.js`).then(() => {
loadedAce.config.set("loadWorkerFromBlob", false); this.set("isLoading", false);
loadedAce.config.set("workerPath", getURL("/javascripts/ace")); // Do not use CDN for workers
if (this.htmlPlaceholder) { next(() => {
this._overridePlaceholder(loadedAce); window.ace.require(["ace/ace"], (loadedAce) => {
} loadedAce.config.set("loadWorkerFromBlob", false);
loadedAce.config.set("workerPath", getURL("/javascripts/ace")); // Do not use CDN for workers
if (!this.element || this.isDestroying || this.isDestroyed) { if (this.htmlPlaceholder) {
return; this._overridePlaceholder(loadedAce);
} }
const editor = loadedAce.edit(this.element.querySelector(".ace"));
editor.setShowPrintMargin(false); if (!this.element || this.isDestroying || this.isDestroyed) {
editor.setOptions({ fontSize: "14px", placeholder: this.placeholder }); return;
editor.getSession().setMode("ace/mode/" + this.mode); }
editor.on("change", () => { const aceElement = this.element.querySelector(".ace");
this._skipContentChangeEvent = true; const editor = loadedAce.edit(aceElement);
this.set("content", editor.getSession().getValue()); editor.setShowPrintMargin(false);
}); editor.setOptions({
if (this.save) { fontSize: "14px",
editor.commands.addCommand({ placeholder: this.placeholder,
name: "save", });
exec: () => { editor.getSession().setMode("ace/mode/" + this.mode);
this.save(); editor.on("change", () => {
}, this._skipContentChangeEvent = true;
bindKey: { mac: "cmd-s", win: "ctrl-s" }, this.set("content", editor.getSession().getValue());
});
if (this.save) {
editor.commands.addCommand({
name: "save",
exec: () => {
this.save();
},
bindKey: { mac: "cmd-s", win: "ctrl-s" },
});
}
editor.on("blur", () => {
this.warnSCSSDeprecations();
});
editor.$blockScrolling = Infinity;
editor.renderer.setScrollMargin(10, 10);
this.element.setAttribute("data-editor", editor);
this._editor = editor;
this.changeDisabledState();
this.warnSCSSDeprecations();
$(window)
.off("ace:resize")
.on("ace:resize", () => this.appEvents.trigger("ace:resize"));
if (this.appEvents) {
// xxx: don't run during qunit tests
this.appEvents.on("ace:resize", this, "resize");
}
if (this.autofocus) {
this.send("focus");
}
this.setAceTheme();
this._darkModeListener = window.matchMedia(
"(prefers-color-scheme: dark)"
);
this._darkModeListener.addListener(this.setAceTheme);
}); });
}
editor.on("blur", () => {
this.warnSCSSDeprecations();
}); });
editor.$blockScrolling = Infinity;
editor.renderer.setScrollMargin(10, 10);
this.element.setAttribute("data-editor", editor);
this._editor = editor;
this.changeDisabledState();
this.warnSCSSDeprecations();
$(window)
.off("ace:resize")
.on("ace:resize", () => this.appEvents.trigger("ace:resize"));
if (this.appEvents) {
// xxx: don't run during qunit tests
this.appEvents.on("ace:resize", this, "resize");
}
if (this.autofocus) {
this.send("focus");
}
this.setAceTheme();
this._darkModeListener = window.matchMedia(
"(prefers-color-scheme: dark)"
);
this._darkModeListener.addListener(this.setAceTheme);
}); });
}); });
} }
@ -165,15 +178,17 @@ export default class AceEditor extends Component {
this._darkModeListener?.removeListener(this.setAceTheme); this._darkModeListener?.removeListener(this.setAceTheme);
} }
@bind get aceTheme() {
setAceTheme() {
const schemeType = getComputedStyle(document.body) const schemeType = getComputedStyle(document.body)
.getPropertyValue("--scheme-type") .getPropertyValue("--scheme-type")
.trim(); .trim();
this._editor.setTheme( return schemeType === "dark" ? "chaos" : "chrome";
`ace/theme/${schemeType === "dark" ? "chaos" : "chrome"}` }
);
@bind
setAceTheme() {
this._editor.setTheme(`ace/theme/${this.aceTheme}`);
} }
warnSCSSDeprecations() { warnSCSSDeprecations() {

View File

@ -3,6 +3,8 @@
export const PUBLIC_JS_VERSIONS = { export const PUBLIC_JS_VERSIONS = {
"ace/ace.js": "ace.js/1.4.13/ace.js", "ace/ace.js": "ace.js/1.4.13/ace.js",
"ace/theme-chrome.js": "ace.js/1.4.13/theme-chrome.js",
"ace/theme-chaos.js": "ace.js/1.4.13/theme-chaos.js",
"jsoneditor.js": "@json-editor/json-editor/2.10.0/jsoneditor.js", "jsoneditor.js": "@json-editor/json-editor/2.10.0/jsoneditor.js",
"chart.min.js": "chart.js/3.5.1/chart.min.js", "chart.min.js": "chart.js/3.5.1/chart.min.js",
"chartjs-plugin-datalabels.min.js": "chartjs-plugin-datalabels.min.js":

View File

@ -1,4 +1,4 @@
import { render } from "@ember/test-helpers"; import { render, waitUntil } from "@ember/test-helpers";
import { hbs } from "ember-cli-htmlbars"; import { hbs } from "ember-cli-htmlbars";
import { module, test } from "qunit"; import { module, test } from "qunit";
import { setupRenderingTest } from "discourse/tests/helpers/component-test"; import { setupRenderingTest } from "discourse/tests/helpers/component-test";
@ -56,9 +56,10 @@ module(
) )
) )
}} />`); }} />`);
const lines = document.querySelectorAll(".ace_line");
const indexOf = lines[0].innerHTML.indexOf("["); await waitUntil(() => document.querySelectorAll(".ace_line")[0]);
assert.ok(indexOf >= 0);
assert.dom(".ace_line").hasText("[");
}); });
test("input is valid json", async function (assert) { test("input is valid json", async function (assert) {

View File

@ -232,28 +232,30 @@ task "javascript:update" => "clean_up" do
dest = "#{path}/#{filename}" dest = "#{path}/#{filename}"
FileUtils.mkdir_p(path) unless File.exist?(path) FileUtils.mkdir_p(path) unless File.exist?(path)
if src.include? "ace.js"
versions["ace/ace.js"] = versions.delete("ace.js")
themes = %w[theme-chrome theme-chaos]
themes.each do |file|
versions["ace/#{file}.js"] = "#{package_dir_name}/#{package_version}/#{file}.js"
end
ace_root = "#{library_src}/ace-builds/src-min-noconflict/"
addtl_files = %w[ext-searchbox mode-html mode-scss mode-sql mode-yaml worker-html].concat(
themes,
)
dest_path = dest.split("/")[0..-2].join("/")
addtl_files.each { |file| FileUtils.cp_r("#{ace_root}#{file}.js", dest_path) }
end
end end
else else
dest = "#{vendor_js}/#{filename}" dest = "#{vendor_js}/#{filename}"
end end
if src.include? "ace.js"
versions["ace/ace.js"] = versions.delete("ace.js")
ace_root = "#{library_src}/ace-builds/src-min-noconflict/"
addtl_files = %w[
ext-searchbox
mode-html
mode-scss
mode-sql
mode-yaml
theme-chrome
theme-chaos
worker-html
]
dest_path = dest.split("/")[0..-2].join("/")
addtl_files.each { |file| FileUtils.cp_r("#{ace_root}#{file}.js", dest_path) }
end
STDERR.puts "New dependency added: #{dest}" unless File.exist?(dest) STDERR.puts "New dependency added: #{dest}" unless File.exist?(dest)
FileUtils.cp_r(src, dest) FileUtils.cp_r(src, dest)