mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-06-09 09:34:57 +09:00
LibWebView: Add language settings to about:settings
This implements a setting to change the languages provided to websites from `navigator.language(s)` and the `Accept-Language` header. Whereas the existing Qt settings dialog allows users to type their language of choice, this setting allows users to select from a predefined list of languages. They may choose any number of languages and their preferred order. This patch only implements the persisted settings and their UI. It does not integrate the choses languages into the WebContent process.
This commit is contained in:
parent
26ec01068f
commit
f242920cc9
Notes:
github-actions[bot]
2025-04-04 08:17:38 +00:00
Author: https://github.com/trflynn89
Commit: f242920cc9
Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/4215
Reviewed-by: https://github.com/gmta ✅
8 changed files with 466 additions and 13 deletions
|
@ -215,6 +215,17 @@
|
|||
opacity: 1;
|
||||
}
|
||||
|
||||
dialog .dialog-button:disabled {
|
||||
opacity: 0.3;
|
||||
}
|
||||
|
||||
dialog .dialog-description {
|
||||
margin-bottom: 10px;
|
||||
|
||||
font-size: 14px;
|
||||
opacity: 0.7;
|
||||
}
|
||||
|
||||
dialog .dialog-list {
|
||||
outline: 1px solid var(--border-color);
|
||||
border-radius: 4px;
|
||||
|
@ -256,7 +267,13 @@
|
|||
gap: 10px;
|
||||
}
|
||||
|
||||
dialog .dialog-controls input {
|
||||
dialog .dialog-controls button svg {
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
}
|
||||
|
||||
dialog .dialog-controls input,
|
||||
dialog .dialog-controls select {
|
||||
flex-grow: 1;
|
||||
}
|
||||
|
||||
|
@ -288,6 +305,13 @@
|
|||
<label for="new-tab-page-url">New Tab Page URL</label>
|
||||
<input id="new-tab-page-url" type="url" placeholder="about:newtab" />
|
||||
</div>
|
||||
|
||||
<hr />
|
||||
|
||||
<div class="card-group permission-container">
|
||||
<span>Languages</span>
|
||||
<button id="languages-settings" class="secondary-button">Settings...</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
@ -349,6 +373,25 @@
|
|||
<button id="restore-defaults" class="primary-button">Restore Defaults</button>
|
||||
</div>
|
||||
|
||||
<dialog id="languages-dialog">
|
||||
<div class="dialog-header">
|
||||
<h3 class="dialog-title">Languages</h3>
|
||||
<button id="languages-close" class="close-button dialog-button">×</button>
|
||||
</div>
|
||||
<div class="dialog-body">
|
||||
<p class="dialog-description">
|
||||
Choose languages for websites in order of preference
|
||||
</p>
|
||||
<div id="languages-list" class="dialog-list"></div>
|
||||
<div class="dialog-controls">
|
||||
<select id="languages-select">
|
||||
<option value="">Select a language to add...</option>
|
||||
</select>
|
||||
<button id="languages-add" class="primary-button" disabled>Add</button>
|
||||
</div>
|
||||
</div>
|
||||
</dialog>
|
||||
|
||||
<dialog id="site-settings">
|
||||
<div class="dialog-header">
|
||||
<h3 id="site-settings-title" class="dialog-title"></h3>
|
||||
|
@ -374,8 +417,16 @@
|
|||
</div>
|
||||
</dialog>
|
||||
|
||||
<script src="resource://ladybird/about-pages/settings/languages.js"></script>
|
||||
|
||||
<script>
|
||||
const newTabPageURL = document.querySelector("#new-tab-page-url");
|
||||
const languagesAdd = document.querySelector("#languages-add");
|
||||
const languagesClose = document.querySelector("#languages-close");
|
||||
const languagesDialog = document.querySelector("#languages-dialog");
|
||||
const languagesList = document.querySelector("#languages-list");
|
||||
const languagesSelect = document.querySelector("#languages-select");
|
||||
const languagesSettings = document.querySelector("#languages-settings");
|
||||
const searchToggle = document.querySelector("#search-toggle");
|
||||
const searchEngine = document.querySelector("#search-engine");
|
||||
const autocompleteToggle = document.querySelector("#autocomplete-toggle");
|
||||
|
@ -392,6 +443,13 @@
|
|||
const doNotTrackToggle = document.querySelector("#do-not-track-toggle");
|
||||
const restoreDefaults = document.querySelector("#restore-defaults");
|
||||
|
||||
// FIXME: When we support per-glyph font fallbacks, replace these SVGs with analogous code points.
|
||||
// https://github.com/LadybirdBrowser/ladybird/issues/864
|
||||
const upwardArrowSVG =
|
||||
'<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="18 15 12 9 6 15"></polyline></svg>';
|
||||
const downwardArrowSVG =
|
||||
'<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="6 9 12 15 18 9"></polyline></svg>';
|
||||
|
||||
window.settings = {};
|
||||
|
||||
const loadSettings = settings => {
|
||||
|
@ -403,6 +461,10 @@
|
|||
newTabPageURL.classList.remove("error");
|
||||
newTabPageURL.value = window.settings.newTabPageURL;
|
||||
|
||||
if (languagesDialog.open) {
|
||||
showLanguages();
|
||||
}
|
||||
|
||||
const renderEngineSettings = (type, setting) => {
|
||||
const [name, toggle, engine] = engineForType(type);
|
||||
|
||||
|
@ -445,6 +507,137 @@
|
|||
}, 1000);
|
||||
});
|
||||
|
||||
const languageDisplayName = language => {
|
||||
const item = window.languages.find(item => item.language === language);
|
||||
return item.displayName;
|
||||
};
|
||||
|
||||
const saveLanguages = () => {
|
||||
ladybird.sendMessage("setLanguages", window.settings.languages);
|
||||
};
|
||||
|
||||
const moveLanguage = (from, to) => {
|
||||
[window.settings.languages[from], window.settings.languages[to]] = [
|
||||
window.settings.languages[to],
|
||||
window.settings.languages[from],
|
||||
];
|
||||
|
||||
saveLanguages();
|
||||
};
|
||||
|
||||
const removeLanguage = index => {
|
||||
window.settings.languages.splice(index, 1);
|
||||
saveLanguages();
|
||||
};
|
||||
|
||||
const loadLanguages = () => {
|
||||
for (const language of window.languages) {
|
||||
const option = document.createElement("option");
|
||||
option.text = language.displayName;
|
||||
option.value = language.language;
|
||||
|
||||
languagesSelect.add(option);
|
||||
}
|
||||
};
|
||||
|
||||
const showLanguages = () => {
|
||||
languagesList.innerHTML = "";
|
||||
|
||||
window.settings.languages.forEach((language, index) => {
|
||||
const name = document.createElement("span");
|
||||
name.className = "dialog-list-item-label";
|
||||
name.textContent = languageDisplayName(language);
|
||||
|
||||
const moveUp = document.createElement("button");
|
||||
moveUp.className = "dialog-button";
|
||||
moveUp.innerHTML = upwardArrowSVG;
|
||||
moveUp.title = "Move up";
|
||||
|
||||
if (index === 0) {
|
||||
moveUp.disabled = true;
|
||||
} else {
|
||||
moveUp.addEventListener("click", () => {
|
||||
moveLanguage(index, index - 1);
|
||||
});
|
||||
}
|
||||
|
||||
const moveDown = document.createElement("button");
|
||||
moveDown.className = "dialog-button";
|
||||
moveDown.innerHTML = downwardArrowSVG;
|
||||
moveDown.title = "Move down";
|
||||
|
||||
if (index === window.settings.languages.length - 1) {
|
||||
moveDown.disabled = true;
|
||||
} else {
|
||||
moveDown.addEventListener("click", () => {
|
||||
moveLanguage(index, index + 1);
|
||||
});
|
||||
}
|
||||
|
||||
const remove = document.createElement("button");
|
||||
remove.className = "dialog-button";
|
||||
remove.innerHTML = "×";
|
||||
remove.title = "Remove";
|
||||
|
||||
if (window.settings.languages.length <= 1) {
|
||||
remove.disabled = true;
|
||||
} else {
|
||||
remove.addEventListener("click", () => {
|
||||
removeLanguage(index);
|
||||
});
|
||||
}
|
||||
|
||||
const controls = document.createElement("div");
|
||||
controls.className = "dialog-controls";
|
||||
controls.appendChild(moveUp);
|
||||
controls.appendChild(moveDown);
|
||||
controls.appendChild(remove);
|
||||
|
||||
const item = document.createElement("div");
|
||||
item.className = "dialog-list-item";
|
||||
item.appendChild(name);
|
||||
item.appendChild(controls);
|
||||
|
||||
languagesList.appendChild(item);
|
||||
});
|
||||
|
||||
for (const language of languagesSelect.options) {
|
||||
language.disabled = window.settings.languages.includes(language.value);
|
||||
}
|
||||
|
||||
if (!languagesDialog.open) {
|
||||
setTimeout(() => languagesSelect.focus());
|
||||
languagesDialog.showModal();
|
||||
}
|
||||
};
|
||||
|
||||
languagesAdd.addEventListener("click", () => {
|
||||
const language = languagesSelect.value;
|
||||
|
||||
languagesAdd.disabled = true;
|
||||
languagesSelect.selectedIndex = 0;
|
||||
|
||||
if (!language || window.settings.languages.includes(language)) {
|
||||
return;
|
||||
}
|
||||
|
||||
window.settings.languages.push(language);
|
||||
saveLanguages();
|
||||
});
|
||||
|
||||
languagesClose.addEventListener("click", () => {
|
||||
languagesDialog.close();
|
||||
});
|
||||
|
||||
languagesSelect.addEventListener("change", () => {
|
||||
languagesAdd.disabled = !languagesSelect.value;
|
||||
});
|
||||
|
||||
languagesSettings.addEventListener("click", event => {
|
||||
showLanguages();
|
||||
event.stopPropagation();
|
||||
});
|
||||
|
||||
const Engine = Object.freeze({
|
||||
search: 1,
|
||||
autocomplete: 2,
|
||||
|
@ -633,26 +826,32 @@
|
|||
|
||||
// FIXME: Once we support `dialog::backdrop`, this event listener should be on `siteSettings`.
|
||||
document.addEventListener("click", event => {
|
||||
if (!siteSettings.open) {
|
||||
return;
|
||||
}
|
||||
const close = dialog => {
|
||||
if (!dialog.open) {
|
||||
return;
|
||||
}
|
||||
|
||||
const rect = siteSettings.getBoundingClientRect();
|
||||
const rect = dialog.getBoundingClientRect();
|
||||
|
||||
if (
|
||||
event.clientX < rect.left ||
|
||||
event.clientX > rect.right ||
|
||||
event.clientY < rect.top ||
|
||||
event.clientY > rect.bottom
|
||||
) {
|
||||
siteSettings.close();
|
||||
}
|
||||
if (
|
||||
event.clientX < rect.left ||
|
||||
event.clientX > rect.right ||
|
||||
event.clientY < rect.top ||
|
||||
event.clientY > rect.bottom
|
||||
) {
|
||||
dialog.close();
|
||||
}
|
||||
};
|
||||
|
||||
close(languagesDialog);
|
||||
close(siteSettings);
|
||||
});
|
||||
|
||||
document.addEventListener("WebUILoaded", () => {
|
||||
ladybird.sendMessage("loadAvailableEngines");
|
||||
ladybird.sendMessage("loadCurrentSettings");
|
||||
ladybird.sendMessage("loadForciblyEnabledSiteSettings");
|
||||
loadLanguages();
|
||||
});
|
||||
|
||||
document.addEventListener("WebUIMessage", event => {
|
||||
|
|
177
Base/res/ladybird/about-pages/settings/languages.js
Normal file
177
Base/res/ladybird/about-pages/settings/languages.js
Normal file
|
@ -0,0 +1,177 @@
|
|||
// Rather than creating a list of all languages supported by ICU (of which there are on the order of a thousand), we
|
||||
// create a list of languages that are supported by both Chrome and Firefox. We can extend this list as needed.
|
||||
//
|
||||
// https://github.com/chromium/chromium/blob/main/ui/base/l10n/l10n_util.cc (see kAcceptLanguageList)
|
||||
// https://github.com/mozilla/gecko-dev/blob/master/intl/locale/language.properties
|
||||
window.languages = (() => {
|
||||
const display = new Intl.DisplayNames([], { type: "language", languageDisplay: "standard" });
|
||||
|
||||
const language = languageID => {
|
||||
return {
|
||||
language: languageID,
|
||||
displayName: display.of(languageID),
|
||||
};
|
||||
};
|
||||
|
||||
const languages = [
|
||||
language("af"),
|
||||
language("ak"),
|
||||
language("am"),
|
||||
language("ar"),
|
||||
language("as"),
|
||||
language("ast"),
|
||||
language("az"),
|
||||
language("be"),
|
||||
language("bg"),
|
||||
language("bm"),
|
||||
language("bn"),
|
||||
language("br"),
|
||||
language("bs"),
|
||||
language("ca"),
|
||||
language("cs"),
|
||||
language("cy"),
|
||||
language("da"),
|
||||
language("de"),
|
||||
language("de-AT"),
|
||||
language("de-CH"),
|
||||
language("de-DE"),
|
||||
language("de-LI"),
|
||||
language("ee"),
|
||||
language("el"),
|
||||
language("en"),
|
||||
language("en-AU"),
|
||||
language("en-CA"),
|
||||
language("en-GB"),
|
||||
language("en-IE"),
|
||||
language("en-NZ"),
|
||||
language("en-US"),
|
||||
language("en-ZA"),
|
||||
language("eo"),
|
||||
language("es"),
|
||||
language("es-AR"),
|
||||
language("es-CL"),
|
||||
language("es-CO"),
|
||||
language("es-CR"),
|
||||
language("es-ES"),
|
||||
language("es-HN"),
|
||||
language("es-MX"),
|
||||
language("es-PE"),
|
||||
language("es-UY"),
|
||||
language("es-VE"),
|
||||
language("et"),
|
||||
language("eu"),
|
||||
language("fa"),
|
||||
language("fi"),
|
||||
language("fo"),
|
||||
language("fr"),
|
||||
language("fr-CA"),
|
||||
language("fr-CH"),
|
||||
language("fr-FR"),
|
||||
language("fy"),
|
||||
language("ga"),
|
||||
language("gd"),
|
||||
language("gl"),
|
||||
language("gu"),
|
||||
language("ha"),
|
||||
language("haw"),
|
||||
language("he"),
|
||||
language("hi"),
|
||||
language("hr"),
|
||||
language("hu"),
|
||||
language("hy"),
|
||||
language("ia"),
|
||||
language("id"),
|
||||
language("ig"),
|
||||
language("is"),
|
||||
language("it"),
|
||||
language("it-CH"),
|
||||
language("ja"),
|
||||
language("jv"),
|
||||
language("ka"),
|
||||
language("kk"),
|
||||
language("km"),
|
||||
language("kn"),
|
||||
language("ko"),
|
||||
language("kok"),
|
||||
language("ku"),
|
||||
language("ky"),
|
||||
language("lb"),
|
||||
language("lg"),
|
||||
language("ln"),
|
||||
language("lo"),
|
||||
language("lt"),
|
||||
language("lv"),
|
||||
language("mai"),
|
||||
language("mg"),
|
||||
language("mi"),
|
||||
language("mk"),
|
||||
language("ml"),
|
||||
language("mn"),
|
||||
language("mr"),
|
||||
language("ms"),
|
||||
language("mt"),
|
||||
language("my"),
|
||||
language("nb"),
|
||||
language("ne"),
|
||||
language("nl"),
|
||||
language("nn"),
|
||||
language("no"),
|
||||
language("nso"),
|
||||
language("oc"),
|
||||
language("om"),
|
||||
language("or"),
|
||||
language("pa"),
|
||||
language("pl"),
|
||||
language("ps"),
|
||||
language("pt"),
|
||||
language("pt-BR"),
|
||||
language("pt-PT"),
|
||||
language("qu"),
|
||||
language("rm"),
|
||||
language("ro"),
|
||||
language("ru"),
|
||||
language("rw"),
|
||||
language("sa"),
|
||||
language("sd"),
|
||||
language("si"),
|
||||
language("sk"),
|
||||
language("sl"),
|
||||
language("so"),
|
||||
language("sq"),
|
||||
language("sr"),
|
||||
language("st"),
|
||||
language("su"),
|
||||
language("sv"),
|
||||
language("sw"),
|
||||
language("ta"),
|
||||
language("te"),
|
||||
language("tg"),
|
||||
language("th"),
|
||||
language("ti"),
|
||||
language("tk"),
|
||||
language("tn"),
|
||||
language("to"),
|
||||
language("tr"),
|
||||
language("tt"),
|
||||
language("ug"),
|
||||
language("uk"),
|
||||
language("ur"),
|
||||
language("uz"),
|
||||
language("vi"),
|
||||
language("wo"),
|
||||
language("xh"),
|
||||
language("yi"),
|
||||
language("yo"),
|
||||
language("zh"),
|
||||
language("zh-CN"),
|
||||
language("zh-HK"),
|
||||
language("zh-TW"),
|
||||
language("zu"),
|
||||
];
|
||||
|
||||
languages.sort((lhs, rhs) => {
|
||||
return lhs.displayName.localeCompare(rhs.displayName);
|
||||
});
|
||||
|
||||
return languages;
|
||||
})();
|
|
@ -114,6 +114,10 @@ button.primary-button:active {
|
|||
background-color: var(--violet-900);
|
||||
}
|
||||
|
||||
button.primary-button:disabled {
|
||||
background-color: var(--violet-60);
|
||||
}
|
||||
|
||||
button.secondary-button {
|
||||
background-color: var(--secondary-button-color);
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue