From 36a3392e98cafad383624cd7743b888603d98cc6 Mon Sep 17 00:00:00 2001
From: btangmu
Date: Wed, 17 Apr 2024 12:44:10 -0400
Subject: [PATCH] CLDR-15649 Dashboard using filters
-Add checkbox next to each notification category in Dashboard header
-The category is hidden if the checkbox is unchecked
-Error and Missing notifications still cannot be hidden with pre-existing right-side path-specific checkboxes, but they can be hidden categorically using the new checkboxes
-Show each path with notifications only once in Dashboard
-If a path has more than one notification, combine them on one line
-Perform data conversion on the front end; http response still has multiple notifications per path
-Refactor to avoid http code in Vue; call cldrAjax.mjs from cldrDash.mjs not from DashboardWidget.vue
-New classes DashData and DashEntry in cldrDash.mjs
-Use new method dashData.addEntriesFromJson for both whole-page json and single-path json
-Make hover titles clearer for circled abbreviations like EC for English Changed
-As before, always hide Abstained notifications if user is TC, but implement on the back end instead of the front end (simpler and more efficient)
-Turn off debug-logging in cldrTable (CLDR_TABLE_DEBUG) and cldrVote (CLDR_VOTE_DEBUG)
---
tools/cldr-apps/js/src/esm/cldrDash.mjs | 596 ++++++++++--------
tools/cldr-apps/js/src/esm/cldrTable.mjs | 2 +-
tools/cldr-apps/js/src/esm/cldrVote.mjs | 2 +-
.../js/src/views/DashboardWidget.vue | 299 +++++----
.../java/org/unicode/cldr/web/Dashboard.java | 3 +
5 files changed, 479 insertions(+), 423 deletions(-)
diff --git a/tools/cldr-apps/js/src/esm/cldrDash.mjs b/tools/cldr-apps/js/src/esm/cldrDash.mjs
index cdfde9ed4d6..47ef13404ca 100644
--- a/tools/cldr-apps/js/src/esm/cldrDash.mjs
+++ b/tools/cldr-apps/js/src/esm/cldrDash.mjs
@@ -2,38 +2,287 @@
* cldrDash: encapsulate dashboard data.
*/
import * as cldrAjax from "./cldrAjax.mjs";
+import * as cldrCoverage from "./cldrCoverage.mjs";
import * as cldrNotify from "./cldrNotify.mjs";
import * as cldrProgress from "./cldrProgress.mjs";
import * as cldrStatus from "./cldrStatus.mjs";
import * as cldrSurvey from "./cldrSurvey.mjs";
-import * as cldrXlsx from "./cldrXlsx.mjs";
import * as XLSX from "xlsx";
-/**
- * An object whose keys are xpstrid (xpath hex IDs like "db7b4f2df0427e4"), and whose values are objects whose
- * keys are notification categories such as "Error" or "English_Changed" and values are "entry"
- * objects with code, english, ..., xpstrid, ... elements.
- *
- * Example: pathIndex[xpstrid][category] = entry
- *
- * There can be at most one notification for a path within each category, but the same path may have notifications in
- * multiple categories. For example, the path with xpstrid = "db7b4f2df0427e4" might have both "Error" and "Warning" notifications,
- * but there can't be more than one "Error" notification for the same path, since the server will have combined them
- * into a single "Error" notification with multiple error messages.
- */
-let pathIndex = {};
+class DashData {
+ /**
+ * Construct a new DashData object
+ *
+ * @returns the new DashData object
+ */
+ constructor() {
+ this.entries = []; // array of DashEntry objects
+ this.cats = new Set(); // set of category names, such as "Error"
+ this.catSize = {}; // number of entries in each category
+ this.catFirst = {}; // first entry.xpstrid in each category
+ // An object whose keys are xpstrid (xpath hex IDs like "db7b4f2df0427e4"), and whose values are DashEntry objects
+ this.pathIndex = {};
+ this.hiddenObject = null;
+ }
+
+ addEntriesFromJson(notifications) {
+ for (let catData of notifications) {
+ for (let group of catData.groups) {
+ for (let e of group.entries) {
+ this.addEntry(catData.category, group, e);
+ }
+ }
+ }
+ }
+
+ /**
+ * Create a new DashEntry in this DashData, or update the existing entry if already present based on e.xpstrid
+ *
+ * @param {String} cat the category such as "Error"
+ * @param {Object} group (from json)
+ * @param {Object} e (entry in old format, from json)
+ */
+ addEntry(cat, group, e) {
+ this.addCategory(cat);
+ this.catSize[cat]++;
+ if (!this.catFirst[cat]) {
+ this.catFirst[cat] = e.xpstrid;
+ }
+ if (this.pathIndex[e.xpstrid]) {
+ const dashEntry = this.pathIndex[e.xpstrid];
+ dashEntry.addCategory(cat);
+ dashEntry.setWinning(e.winning);
+ if (e.comment) {
+ dashEntry.setComment(e.comment);
+ }
+ if (e.subtype) {
+ dashEntry.setSubtype(e.subtype);
+ }
+ } else {
+ const dashEntry = new DashEntry(e.xpstrid, e.code, e.english);
+ dashEntry.setSectionPageHeader(group.section, group.page, group.header);
+ dashEntry.addCategory(cat);
+ dashEntry.setWinning(e.winning);
+ dashEntry.setPreviousEnglish(e.previousEnglish);
+ dashEntry.setComment(e.comment);
+ dashEntry.setSubtype(e.subtype);
+ dashEntry.setChecked(this.itemIsChecked(e));
+ this.entries.push(dashEntry);
+ this.pathIndex[e.xpstrid] = dashEntry;
+ }
+ }
+
+ /**
+ * If this category is not already present in the data, add it and
+ * initialize its size to zero. No effect if the category is already present.
+ *
+ * @param {String} cat
+ */
+ addCategory(cat) {
+ if (!this.cats.has(cat)) {
+ this.cats.add(cat);
+ this.catSize[cat] = 0;
+ }
+ }
+
+ /**
+ * Set the object indicating which paths the user wants hidden in the dashboard
+ *
+ * The format matches the json currently returned by the back end. The keys are
+ * "subtype" strings such as "incorrectCasing"; the values are arrays of objects
+ * each of which has an xpstrid and a value string
+ *
+ * @param {Object} hiddenObject
+ * Example of json.hidden:
+ * {
+ "incorrectCasing": [
+ {
+ "value": "région micronésienne",
+ "xpstrid": "fe9015c6c61370"
+ },
+ {
+ "value": "régions éloignées de l’Océanie",
+ "xpstrid": "744c18884a1547be"
+ },
+ {
+ "value": "pseudo-accents",
+ "xpstrid": "4ef00bbec7020af2"
+ }
+ ],
+ "none": [
+ {
+ "value": "iakoute",
+ "xpstrid": "79262aa5d69820da"
+ },
+ {
+ "value": "hanifi",
+ "xpstrid": "e8cecebdded8d76"
+ },
+ {
+ "value": "Monde",
+ "xpstrid": "73c7c09de32184"
+ }
+ ]
+ */
+ setHidden(hiddenObject) {
+ this.hiddenObject = hiddenObject;
+ }
+
+ itemIsChecked(e) {
+ if (!this.hiddenObject[e.subtype]) {
+ return false;
+ }
+ const pathValueArray = this.hiddenObject[e.subtype];
+ return pathValueArray.some(
+ (p) => p.xpstrid === e.xpstrid && p.value === e.winning
+ );
+ }
+
+ /**
+ * After the user has voted, reset attributes for the entry that might have
+ * changed or disappeared as a result of voting. Also update the totals for
+ * this DashData.
+ */
+ cleanEntry(dashEntry) {
+ this.removeEntryCats(dashEntry);
+ dashEntry.clean();
+ }
+
+ removeEntry(dashEntry) {
+ this.removeEntryCats(dashEntry);
+ const index = this.entries.indexOf(dashEntry);
+ this.entries.splice(index, 1);
+ }
+
+ removeEntryCats(dashEntry) {
+ dashEntry.cats.forEach((cat) => {
+ this.catSize[cat]--;
+ if (!this.catSize[cat]) {
+ this.cats.delete(cat);
+ delete this.catSize[cat];
+ }
+ if (this.catFirst[cat] === dashEntry.xpstrid) {
+ this.findCatFirst(cat);
+ }
+ });
+ }
+
+ findCatFirst(cat) {
+ for (let dashEntry of this.entries) {
+ if (dashEntry.cat === cat) {
+ this.catFirst[cat] = dashEntry;
+ return;
+ }
+ }
+ delete this.catFirst[cat];
+ }
+}
+
+class DashEntry {
+ /**
+ * Construct a new DashEntry object
+ *
+ * @param {String} xpstrid the xpath hex string id like "710b6e70773e5764"
+ * @param {String} code the code like "long-one-nominative"
+ * @param {String} english the English value like "{0} metric pint"
+ *
+ * @returns the new DashEntry object
+ */
+ constructor(xpstrid, code, english) {
+ this.xpstrid = xpstrid;
+ this.code = code;
+ this.english = english;
+ this.cats = new Set(); // set of notification category names
+ this.winning = null;
+ this.previousEnglish = null;
+ this.comment = null;
+ this.subtype = null;
+ this.checked = false;
+ }
+
+ setSectionPageHeader(section, page, header) {
+ this.section = section; // e.g., "Units"
+ this.page = page; // e.g., "Volume"
+ this.header = header; // e.g., "pint-metric"
+ }
+
+ setWinning(winning) {
+ this.winning = winning; // like "{0} pinte métrique"
+ }
+
+ setPreviousEnglish(previousEnglish) {
+ this.previousEnglish = previousEnglish; // e.g., "{0} metric pint"
+ }
+
+ setComment(comment) {
+ this.comment = comment; // e.g., "<missing placeholders> Need at least 1 placeholder(s), but only have 0. Placeholders..."
+ }
+
+ setSubtype(subtype) {
+ this.subtype = subtype; // e.g., "missingPlaceholders"
+ }
+
+ setChecked(checked) {
+ this.checked = checked; // boolean; the user added a checkmark for this entry
+ }
+
+ addCategory(category) {
+ this.cats.add(category); // e.g., "Error", "Disputed", "English_Changed
+ }
+
+ /**
+ * After the user has voted, reset attributes that might have changed
+ * or disappeared as a result of voting. Any updated values for these
+ * attributes will be added based on the new server response.
+ */
+ clean() {
+ this.cats = new Set();
+ this.winning = null;
+ this.comment = null;
+ this.subtype = null;
+ }
+}
+
+let fetchErr = "";
+
+let viewSetDataCallback = null;
+
+function doFetch(callback) {
+ viewSetDataCallback = callback;
+ fetchErr = "";
+ const locale = cldrStatus.getCurrentLocale();
+ const level = cldrCoverage.effectiveName(locale);
+ if (!locale || !level) {
+ fetchErr = "Please choose a locale and a coverage level first.";
+ return;
+ }
+ const url = `api/summary/dashboard/${locale}/${level}`;
+ cldrAjax
+ .doFetch(url)
+ .then(cldrAjax.handleFetchErrors)
+ .then((data) => data.json())
+ .then(setData)
+ .catch((err) => {
+ const msg = "Error loading Dashboard data: " + err;
+ console.error(msg);
+ fetchErr = msg;
+ });
+}
+
+function getFetchError() {
+ return fetchErr;
+}
/**
- * Set the data for the Dashboard, add "total" and "checked" fields, and index it.
+ * Set the data for the Dashboard, converting from json to a DashData object
*
* The json data as received from the back end is ordered by category, then by section, page, header, code, ...
- * (but those are not ordered alphabetically). It is presented to the user in that same order.
+ * (but those are not ordered alphabetically).
*
- * @param data - an object with these elements:
+ * @param json - an object with these elements:
* notifications - an array of objects (locally named "catData" meaning "all the data for one category"),
* each having these elements:
* category - a string like "Error" or "English_Changed"
- * total - an integer (number of entries in this category), added by addCounts, not in json
* groups - an array of objects, each having these elements:
* header - a string
* page - a string
@@ -52,246 +301,55 @@ let pathIndex = {};
* Dashboard only uses data.notifications. There are additional fields
* data.* used by cldrProgress for progress meters.
*
- * @return the modified data (with totals, etc., added)
- */
-function setData(data) {
- cldrProgress.updateVoterCompletion(data);
- addCounts(data);
- makePathIndex(data);
- return data;
-}
-
-/**
- * Calculate total counts; modify data by adding "total" for each category
- *
- * @param data
+ * @return the modified/reorganized data as a DashData object
*/
-function addCounts(data) {
- for (let catData of data.notifications) {
- catData.total = 0;
- for (let group of catData.groups) {
- catData.total += group.entries.length;
- }
- }
+function setData(json) {
+ cldrProgress.updateVoterCompletion(json);
+ const newData = convertData(json);
+ viewSetDataCallback(newData);
+ return newData;
}
-/**
- * Create the index; also set checked = true/false for all entries
- *
- * @param data
- */
-function makePathIndex(data) {
- pathIndex = {};
- for (let catData of data.notifications) {
- for (let group of catData.groups) {
- for (let entry of group.entries) {
- entry.checked = itemIsChecked(data, entry);
- if (!pathIndex[entry.xpstrid]) {
- pathIndex[entry.xpstrid] = {};
- } else if (pathIndex[entry.xpstrid][catData.category]) {
- console.error(
- "Duplicate in makePathIndex: " +
- entry.xpstrid +
- ", " +
- catData.category
- );
- }
- pathIndex[entry.xpstrid][catData.category] = entry;
- }
- }
- }
-}
-
-function itemIsChecked(data, entry) {
- if (!data.hidden || !data.hidden[entry.subtype]) {
- return false;
- }
- const pathValueArray = data.hidden[entry.subtype];
- return pathValueArray.some(
- (p) => p.xpstrid === entry.xpstrid && p.value === entry.winning
- );
+function convertData(json) {
+ const newData = new DashData();
+ newData.setHidden(json.hidden);
+ newData.addEntriesFromJson(json.notifications);
+ return newData;
}
/**
* A user has voted. Update the Dashboard data and index as needed.
*
* Even though the json is only for one path, it may have multiple notifications,
- * with different categories such as "Warning" and "English_Changed",
- * affecting multiple Dashboard rows.
+ * with different categories such as "Warning" and "English_Changed".
*
* Ensure that the data gets updated for (1) each new or modified notification,
* and (2) each obsolete notification -- if a notification for this path occurs in
* the (old) data but not in the (new) json, it's obsolete and must be removed.
*
- * @param data - the Dashboard data for all paths, to be updated
+ * @param dashData - the DashData (new format) for all paths, to be updated
* @param json - the response to a request by cldrTable.refreshSingleRow,
- * containing notifications for a single path
+ * containing notifications for a single path (old format)
*/
-function updatePath(data, json) {
+function updatePath(dashData, json) {
try {
- const updater = newPathUpdater(data, json);
- updater.oldCategories.forEach((category) => {
- if (updater.newCategories.includes(category)) {
- updateEntry(updater, category);
- } else {
- removeEntry(updater, category);
- }
- });
- updater.newCategories.forEach((category) => {
- if (!updater.oldCategories.includes(category)) {
- addEntry(updater, category);
- }
- });
- } catch (e) {
- cldrNotify.exception(e, "updating path for Dashboard");
- }
- return data; // for unit test
-}
-
-function newPathUpdater(data, json) {
- if (!json.xpstrid) {
- cldrNotify.error(
- "Invalid server response",
- "Missing path identifier for Dashboard"
- );
- return null;
- }
- const updater = {
- data: data,
- json: json,
- xpstrid: json.xpstrid,
- oldCategories: [],
- newCategories: [],
- group: null,
- };
- for (let catData of json.notifications) {
- updater.newCategories.push(catData.category);
- }
- updater.newCategories = updater.newCategories.sort();
- if (updater.xpstrid in pathIndex) {
- updater.oldCategories = Object.keys(pathIndex[updater.xpstrid]).sort();
- }
- return updater;
-}
-
-function updateEntry(updater, category) {
- try {
- const catData = getDataForCategory(updater.data, category);
- for (let group of catData.groups) {
- const entries = group.entries;
- for (let i in entries) {
- if (entries[i].xpstrid === updater.xpstrid) {
- const newEntry = getNewEntry(updater, category);
- pathIndex[updater.xpstrid][category] = entries[i] = newEntry;
- return;
- }
+ if (json.xpstrid in dashData.pathIndex) {
+ // We already have an entry for this path
+ const dashEntry = dashData.pathIndex[json.xpstrid];
+ if (!json.notifications?.length) {
+ // The path no longer has any notifications, so remove the entry
+ dashData.removeEntry(dashEntry);
+ return;
}
+ // Clear attributes that might have changed or disappeared as a result of voting.
+ // They will be updated/restored from the json.
+ dashData.cleanEntry(dashEntry);
}
+ dashData.addEntriesFromJson(json.notifications);
} catch (e) {
- cldrNotify.exception(e, "updating dashboard entry");
- }
-}
-
-function removeEntry(updater, category) {
- try {
- const catData = getDataForCategory(updater.data, category);
- for (let group of catData.groups) {
- const entries = group.entries;
- for (let i in entries) {
- if (entries[i].xpstrid === updater.xpstrid) {
- entries.splice(i, 1);
- --catData.total;
- delete pathIndex[updater.xpstrid][category];
- return;
- }
- }
- }
- } catch (e) {
- cldrNotify.exception(e, "removing dashboard entry");
- }
-}
-
-function addEntry(updater, category) {
- try {
- const newEntry = getNewEntry(updater, category); // sets updater.group
- const catData = getDataForCategory(updater.data, category);
- const group = getMatchingGroup(catData, updater.group);
- // TODO: insert in a particular order; see https://unicode-org.atlassian.net/browse/CLDR-15202
- group.entries.push(newEntry);
- catData.total++;
- if (!(updater.xpstrid in pathIndex)) {
- pathIndex[updater.xpstrid] = {};
- }
- pathIndex[updater.xpstrid][category] = newEntry;
- } catch (e) {
- cldrNotify.exception(e, "adding dashboard entry");
- }
-}
-
-function getNewEntry(updater, category) {
- for (let catData of updater.json.notifications) {
- if (catData.category === category) {
- for (let group of catData.groups) {
- for (let entry of group.entries) {
- if (entry.xpstrid === updater.xpstrid) {
- updater.group = group;
- entry.checked = itemIsChecked(updater.data, entry);
- return entry;
- }
- }
- }
- }
- }
- throw new Error("New entry not found");
-}
-
-function getMatchingGroup(catData, groupToMatch) {
- if (!groupToMatch) {
- throw new Error("Matching dashboard group not found");
- }
- for (let group of catData.groups) {
- if (groupsMatch(group, groupToMatch)) {
- return group;
- }
- }
- const newGroup = cloneGroup(groupToMatch);
- // TODO: insert in a particular order; see https://unicode-org.atlassian.net/browse/CLDR-15202
- catData.groups.push(newGroup);
- return newGroup;
-}
-
-function groupsMatch(groupA, groupB) {
- return (
- groupA.header === groupB.header &&
- groupA.page === groupB.page &&
- groupA.section === groupB.section
- );
-}
-
-function cloneGroup(group) {
- const newGroup = {
- header: group.header,
- page: group.page,
- section: group.section,
- entries: [],
- };
- return newGroup;
-}
-
-function getDataForCategory(data, category) {
- for (let catData of data.notifications) {
- if (catData.category === category) {
- return catData;
- }
+ cldrNotify.exception(e, "updating path for Dashboard");
}
- const newCatData = {
- category: category,
- total: 0,
- groups: [],
- };
- // TODO: insert in a particular order; see https://unicode-org.atlassian.net/browse/CLDR-15202
- data.notifications.push(newCatData);
- return newCatData;
+ return dashData; // for unit test
}
/**
@@ -322,6 +380,7 @@ function getCheckmarkUrl(entry, locale) {
}
/**
+ * Download as XLSX spreadsheet
*
* @param {Object} data processed data to download
* @param {String} locale locale id
@@ -329,20 +388,16 @@ function getCheckmarkUrl(entry, locale) {
*/
async function downloadXlsx(data, locale, cb) {
const xpathMap = cldrSurvey.getXpathMap();
- const { coverageLevel, notifications } = data;
+ const { coverageLevel, entries } = data;
// Fetch all XPaths in parallel since it'll take a while
cb(`Loading…`);
const allXpaths = [];
- for (const { groups } of notifications) {
- for (const { section, entries } of groups) {
- if (section === "Reports") {
- continue; // skip this
- }
- for (const { xpstrid } of entries) {
- allXpaths.push(xpstrid);
- }
+ for (let dashEntry of entries) {
+ if (dashEntry.section === "Reports") {
+ continue; // skip this
}
+ allXpaths.push(dashEntry.xpstrid);
}
await Promise.all(allXpaths.map((x) => xpathMap.get(x)));
cb(`Calculating…`);
@@ -366,29 +421,25 @@ async function downloadXlsx(data, locale, cb) {
],
];
- for (const { category, groups } of notifications) {
- for (const { header, page, section, entries } of groups) {
- for (const { code, english, old, subtype, winning, xpstrid } of entries) {
- const xpath =
- section === "Reports" ? "-" : (await xpathMap.get(xpstrid)).path;
- ws_data.push([
- category,
- header,
- page,
- section,
- code,
- english,
- old,
- subtype,
- winning,
- xpstrid,
- xpath,
- `https://st.unicode.org/cldr-apps/v#/${locale}/${page}/${xpstrid}`,
- ]);
- }
- }
+ for (let e of entries) {
+ const xpath =
+ section === "Reports" ? "-" : (await xpathMap.get(xpstrid)).path;
+ const url = `https://st.unicode.org/cldr-apps/v#/${e.locale}/${e.page}/${e.xpstrid}`;
+ ws_data.push([
+ e.cat,
+ e.header,
+ e.page,
+ e.section,
+ e.code,
+ e.english,
+ e.old,
+ e.subtype,
+ e.winning,
+ e.xpstrid,
+ xpath,
+ url,
+ ]);
}
-
const ws = XLSX.utils.aoa_to_sheet(ws_data);
// cldrXlsx.pushComment(ws, "C1", `As of ${new Date().toISOString()}`);
const wb = XLSX.utils.book_new();
@@ -402,4 +453,11 @@ async function downloadXlsx(data, locale, cb) {
cb(null);
}
-export { saveEntryCheckmark, setData, updatePath, downloadXlsx };
+export {
+ doFetch,
+ getFetchError,
+ saveEntryCheckmark,
+ setData,
+ updatePath,
+ downloadXlsx,
+};
diff --git a/tools/cldr-apps/js/src/esm/cldrTable.mjs b/tools/cldr-apps/js/src/esm/cldrTable.mjs
index 51d723bba76..d983f105687 100644
--- a/tools/cldr-apps/js/src/esm/cldrTable.mjs
+++ b/tools/cldr-apps/js/src/esm/cldrTable.mjs
@@ -26,7 +26,7 @@ import * as cldrXPathUtils from "./cldrXpathUtils.mjs";
const HEADER_ID_PREFIX = "header_";
const ROW_ID_PREFIX = "row_"; // formerly "r@"
-const CLDR_TABLE_DEBUG = true;
+const CLDR_TABLE_DEBUG = false;
/*
* NO_WINNING_VALUE indicates the server delivered path data without a valid winning value.
diff --git a/tools/cldr-apps/js/src/esm/cldrVote.mjs b/tools/cldr-apps/js/src/esm/cldrVote.mjs
index 81ce9e80c78..13ca7c72a46 100644
--- a/tools/cldr-apps/js/src/esm/cldrVote.mjs
+++ b/tools/cldr-apps/js/src/esm/cldrVote.mjs
@@ -13,7 +13,7 @@ import * as cldrSurvey from "./cldrSurvey.mjs";
import * as cldrTable from "./cldrTable.mjs";
import * as cldrText from "./cldrText.mjs";
-const CLDR_VOTE_DEBUG = true;
+const CLDR_VOTE_DEBUG = false;
/**
* The special "vote level" selected by the user, or zero for default.
diff --git a/tools/cldr-apps/js/src/views/DashboardWidget.vue b/tools/cldr-apps/js/src/views/DashboardWidget.vue
index 3100513b303..67f6c422a48 100644
--- a/tools/cldr-apps/js/src/views/DashboardWidget.vue
+++ b/tools/cldr-apps/js/src/views/DashboardWidget.vue
@@ -29,15 +29,27 @@
>
↻
-
-
+
+
+ {
+ catCheckmarkChanged(event, cat);
+ }
+ "
+ />
@@ -58,109 +70,94 @@
title="Hide checked items"
id="hideChecked"
v-model="hideChecked"
- />
+ />
-
-
-