6f41c6a196697c175a2cf5abcbd247ba24a2fe8e
jcasper
Mon Jun 15 09:22:37 2026 -0700
Loading spinner for faceted composites (the page must first do the ajax metadata
fetch, then process the result with dataTables). refs #36320
diff --git src/hg/htdocs/style/facetedComposite.css src/hg/htdocs/style/facetedComposite.css
index 3802075d011..de4f4222cb7 100644
--- src/hg/htdocs/style/facetedComposite.css
+++ src/hg/htdocs/style/facetedComposite.css
@@ -1,215 +1,236 @@
/* SPDX-License-Identifier: MIT; (c) 2025 Andrew D Smith (author) */
+/* Loading indicator shown while metadata is fetched and the table is built */
+#faceted-loading {
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ justify-content: center;
+ gap: 0.75em;
+ padding: 3em 0;
+ color: #555;
+}
+#faceted-loading .faceted-spinner {
+ width: 36px;
+ height: 36px;
+ border: 4px solid #d3eaff; /* matches the table's light-blue accents */
+ border-top-color: #2b6cb0;
+ border-radius: 50%;
+ animation: faceted-spin 0.8s linear infinite;
+}
+@keyframes faceted-spin {
+ to { transform: rotate(360deg); }
+}
#container {
display: flex;
width: 100%;
gap: 1em;
align-items: flex-start;
box-sizing: border-box;
}
#filters {
display: flex;
flex-direction: column;
gap: 1em;
width: 300px;
flex-shrink: 0;
/* ADS: the lines below are for vertical scrolling if needed */
/* max-height: 80vh; */
/* overflow-y: auto; */
align-self: stretch;
box-sizing: border-box;
padding-right: 0.5em;
}
#filters > div {
display: flex;
flex-direction: column;
gap: 0.3em;
}
#filters label {
display: flex;
align-items: center;
gap: 0.3em;
cursor: pointer;
user-select: none;
}
#theMetaDataTable_wrapper {
flex: 1 1 auto;
min-width: 0;
box-sizing: border-box;
}
/* Horizontal-scroll box around metadata tables: a too-wide table scrolls
internally in this instead of spilling off-screen. The toolbar (Show N, paging)
sits outside this box, so it stays visible at the wrapper's width.
The genome browser page wraps this content in an old-school
that sizes to its content. The "width: 0; min-width: 100%"
pair stops the metadata table from reporting the (very large) content width,
the outer table sticks to the page width, the metadata table's min-width:100% makes
it expand to fill that, and then then metadata's overflow results in a scroll bar.
*/
.table-xscroll {
width: 0;
min-width: 100%;
overflow-x: auto;
}
#theMetaDataTable {
width: 100% !important;
box-sizing: border-box;
box-shadow: 0 0 0 1px #ddd;
}
/* Light blue background for the table header (column titles + search row) */
#theMetaDataTable thead th,
#theMetaDataTable thead td {
background-color: lightblue;
}
#theMetaDataTable td:nth-child(n+2),
#theMetaDataTable th:nth-child(n+2) {
vertical-align: top;
/* white-space: nowrap; */
/* overflow: hidden; /* hide overflow */
min-width: 100px; /* Adjust width as needed */
}
/* Override Select 3.0's custom checkbox styling to use native appearance */
table.dataTable input.dt-select-checkbox {
appearance: auto;
width: auto;
height: auto;
}
table.dataTable input.dt-select-checkbox:checked::after,
table.dataTable input.dt-select-checkbox:indeterminate::after {
display: none;
}
#theMetaDataTable input.row-select {
/* additional checkbox styling */
}
table.dataTable {
width: 100%;
border-collapse: collapse;
table-layout: auto;
}
table.dataTable tbody tr:nth-child(odd) {
background-color: #f0f0f0;
}
table.dataTable tbody tr:nth-child(even) {
background-color: #ffffff;
}
table.dataTable tbody tr:hover {
background-color: #d3eaff; /* Light blue on hover */
}
.color-box {
display: inline-block;
width: 1em;
height: 1em;
vertical-align: middle;
/* background-color set dynamically in JS */
}
.facet-heading {
cursor: pointer;
user-select: none;
}
.facet-heading::after {
content: " ❯";
margin-left: 0.3em;
display: inline-block;
transition: transform 0.2s;
transform: rotate(90deg); /* points down = expanded */
}
.facet-heading.collapsed::after {
transform: rotate(0deg); /* points right = collapsed */
}
.facet-body {
display: flex;
flex-direction: column;
gap: 0.3em;
}
.facet-body.collapsed {
display: none;
}
/* "All / Selected" segmented tabs in the toolbar */
#selected-filter {
display: inline-flex;
align-items: stretch;
order: -1; /* sit at the left edge, ahead of the page-length dropdown */
}
/* Top toolbar: both the page-length dropdown and the "show only selected"
toggle live inside .dt-length (the toggle is appended there in
facetedComposite.js). Stretch the cell full width and push the two
children apart: toggle on the left (via order: -1 above), dropdown right. */
#theMetaDataTable_wrapper .dt-layout-start:has(.dt-length) {
flex: 1 1 auto;
}
.dt-length {
display: flex !important;
width: 100%;
align-items: center;
justify-content: space-between;
}
/* Tab-like blocks for the "All / Selected" selection filter */
.filter-tab {
border: 1px solid #aaa;
background-color: #f0f0f0;
color: #333;
padding: 0.3em 0.9em;
font-size: 0.9em;
cursor: pointer;
user-select: none;
transition: background-color 0.15s;
}
.filter-tab:first-of-type {
border-radius: 4px 0 0 4px;
border-right: none; /* avoid a doubled border between the two tabs */
}
.filter-tab:last-of-type {
border-radius: 0 4px 4px 0;
}
.filter-tab:hover {
background-color: #e3e3e3;
}
.filter-tab.active {
background-color: lightblue; /* matches the table header band */
color: #000;
font-weight: bold;
}
/* Active filter chips bar */
#active-filters {
display: flex;
flex-wrap: wrap;
align-items: center;
justify-content: flex-start;
text-align: left;
gap: 0.4em;
padding: 0.5em 0;
border-top: 1px solid #ddd;
margin-top: 0.75em;
width: 100%;
}
.filter-chip-group-label {
font-weight: bold;
font-size: 0.85em;
margin-left: 0.3em;
}
.filter-chip {
display: inline-flex;
align-items: center;
background-color: #e8f0fe;
border: 1px solid #c2d6f2;
border-radius: 12px;
padding: 0.15em 0.5em;
font-size: 0.85em;
white-space: nowrap;
}
.filter-chip .remove-chip {
background: none;
border: none;
cursor: pointer;
font-size: 1em;
line-height: 1;
padding: 0 0 0 0.25em;
color: #666;
}
.filter-chip .remove-chip:hover {
color: #c00;
}