zulip/web/src/realm_playground.ts

109 lines
3.9 KiB
TypeScript

import assert from "minimalistic-assert";
import * as typeahead from "../shared/src/typeahead";
import {$t} from "./i18n";
import * as pygments_data from "./pygments_data";
export type RealmPlayground = {
id: number;
name: string;
pygments_language: string;
url_template: string;
};
const map_language_to_playground_info = new Map<string, RealmPlayground[]>();
const map_pygments_pretty_name_to_aliases = new Map<string, string[]>();
export function update_playgrounds(playgrounds_data: RealmPlayground[]): void {
map_language_to_playground_info.clear();
for (const data of playgrounds_data) {
const element_to_push: RealmPlayground = {
id: data.id,
name: data.name,
url_template: data.url_template,
pygments_language: data.pygments_language,
};
if (map_language_to_playground_info.has(data.pygments_language)) {
map_language_to_playground_info.get(data.pygments_language)!.push(element_to_push);
} else {
map_language_to_playground_info.set(data.pygments_language, [element_to_push]);
}
}
}
export function get_playground_info_for_languages(lang: string): RealmPlayground[] | undefined {
return map_language_to_playground_info.get(lang);
}
function sort_pygments_pretty_names_by_priority(
comparator_func: (a: string, b: string) => number,
): void {
const priority_sorted_pygments_data = Object.entries(pygments_data.langs).sort(([a], [b]) =>
comparator_func(a, b),
);
for (const [alias, data] of priority_sorted_pygments_data) {
assert(data !== undefined);
const pretty_name = data.pretty_name;
// JS Map remembers the original order of insertion of keys.
if (map_pygments_pretty_name_to_aliases.has(pretty_name)) {
map_pygments_pretty_name_to_aliases.get(pretty_name)!.push(alias);
} else {
map_pygments_pretty_name_to_aliases.set(pretty_name, [alias]);
}
}
}
// This gets the candidate list for showing autocomplete for a code block in
// the composebox. The candidate list will include pygments data as well as any
// Code Playgrounds.
//
// May return duplicates, since it's common for playground languages
// to also be pygments languages! retain_unique_language_aliases will
// deduplicate them.
export function get_pygments_typeahead_list_for_composebox(): string[] {
const playground_pygment_langs = [...map_language_to_playground_info.keys()];
const pygment_langs = Object.keys(pygments_data.langs);
return [...playground_pygment_langs, ...pygment_langs];
}
// This gets the candidate list for showing autocomplete in settings when
// adding a new Code Playground.
export function get_pygments_typeahead_list_for_settings(query: string): Map<string, string> {
const language_labels = new Map<string, string>();
// Adds a typeahead that allows selecting a custom language, by adding a
// "Custom language" label in the first position of the typeahead list.
const clean_query = typeahead.clean_query_lowercase(query);
if (clean_query !== "") {
language_labels.set(
clean_query,
$t({defaultMessage: "Custom language: {query}"}, {query: clean_query}),
);
}
const playground_pygment_langs = [...map_language_to_playground_info.keys()];
for (const lang of playground_pygment_langs) {
language_labels.set(lang, $t({defaultMessage: "Custom language: {query}"}, {query: lang}));
}
for (const [key, values] of map_pygments_pretty_name_to_aliases) {
language_labels.set(key, key + " (" + values.join(", ") + ")");
}
return language_labels;
}
export function initialize({
playground_data,
pygments_comparator_func,
}: {
playground_data: RealmPlayground[];
pygments_comparator_func: (a: string, b: string) => number;
}): void {
update_playgrounds(playground_data);
sort_pygments_pretty_names_by_priority(pygments_comparator_func);
}