Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Category dashboard with collapsible filters #934

Open
wants to merge 35 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
d3987bf
custom techreport header
sarahfossheim Aug 28, 2024
b21cc6e
add prototype categories page
sarahfossheim Aug 28, 2024
04628eb
add filters in sidebar and change page based on amount of techs
sarahfossheim Aug 30, 2024
07171f7
add new nav and filter structure (wip
sarahfossheim Sep 4, 2024
c312252
responsive filters prototype
sarahfossheim Sep 17, 2024
706d6f0
wip: add collapsible filter sidebar
sarahfossheim Sep 24, 2024
fa8c689
fix bug with double close button
sarahfossheim Sep 24, 2024
e58f7ce
add aria
sarahfossheim Sep 24, 2024
7f837c1
merge in master
sarahfossheim Sep 25, 2024
6b24d96
add checkboxes on category page, update texts, minor bugfixes
sarahfossheim Oct 18, 2024
cf7c0e8
select correct technology in comparison view, remove category selecto…
sarahfossheim Oct 23, 2024
bf11a13
fix css linting
sarahfossheim Oct 23, 2024
ce8df02
fix linting
sarahfossheim Oct 23, 2024
bc5af01
Merge branch 'main' into cwvtech-category-dashboard
sarahfossheim Oct 30, 2024
b74de3b
fix linting
sarahfossheim Oct 30, 2024
728d42f
Merge remote-tracking branch 'refs/remotes/origin/cwvtech-category-da…
sarahfossheim Oct 30, 2024
0d062cc
fix sorting order multi apps
sarahfossheim Oct 30, 2024
aea33e1
sort table alternative correctly
sarahfossheim Nov 8, 2024
7aaf182
merge main into branch
sarahfossheim Nov 18, 2024
fea35f0
close <nav>
sarahfossheim Nov 19, 2024
c16fe45
merge master into branch
sarahfossheim Dec 3, 2024
3987516
update tests
sarahfossheim Dec 18, 2024
5a45a40
format tests
sarahfossheim Dec 18, 2024
d67c907
Update src/js/techreport/tableLinked.js
sarahfossheim Dec 18, 2024
7820ed5
fix geo/rank bug, long category list bug, adoption bug, add category …
sarahfossheim Jan 7, 2025
cd10b24
add technologies to summary in category page
sarahfossheim Jan 8, 2025
273d19b
add timestamp to category page and format large numbers
sarahfossheim Jan 12, 2025
37dbaeb
remove console.log
sarahfossheim Jan 12, 2025
d861090
select correct technology in comparison view filters
sarahfossheim Jan 15, 2025
3a125ad
make menu responsive on mobile + track aria-expanded status
sarahfossheim Jan 16, 2025
062bf5b
give remove button icon alt that matches the label
sarahfossheim Jan 16, 2025
f664a28
filter out data with empty content
sarahfossheim Jan 16, 2025
7fb726e
fix linting
sarahfossheim Jan 16, 2025
9a905c3
add width and height to close filter button
sarahfossheim Jan 16, 2025
163cdc3
remove placeholder texts
sarahfossheim Jan 16, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
72 changes: 72 additions & 0 deletions config/techreport.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,11 @@
"name": "Technology Report",
"summary": "The Core Web Vitals Technology Report is a dashboard combining the powers of real-user experiences in the [Chrome User Experience Report (CrUX)](https://developers.google.com/web/tools/chrome-user-experience-report/) dataset with web technology detections available in HTTP Archive, to allow analysis of the way websites are both built and experienced.",
"config": {
"default_apps": {
"drilldown": [ "ALL" ],
"comparison": [ "ALL", "WordPress", "Wix", "Next.js" ]
},
"default_category": "CMS",
"cwv_subcategories": [
"CLS",
"LCP",
Expand Down Expand Up @@ -1118,6 +1123,73 @@
}
},
"description": "Comparison placeholder"
},
"category": {
"id": "category",
"title": "Categories",
"subtitle": "Technology Report",
"config": {
"default": {
"category": "CMS",
"app": ["ALL", "WordPress", "Drupal"],
"series": {
"breakdown": "app"
}
},
"tech_comparison_summary": {
"id": "tech_comparison_summary",
"table": {
"caption": "Summary",
"columns": [
{
"key": "technology",
"name": "Tech",
"type": "heading"
},
{
"key": "origins",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Values appear as 3222201 which can be harder to read. Could we format with commas, ie 3,222,201? It'd also be helpful for comparing values if the text in this column is right-aligned, like a spreadsheet.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Formatted them with commas everywhere now

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In addition to the number of origins, I think a primary use case for this data would be to see the proportion of origins that adopt this category relative to all origins that adopt this category.

For example, there are 3.2M WordPress origins and the /categories endpoint says that there are 8.8M origins in the CMS category, so we could show that "market share" value as 36%.

What do you think would be the best way to show the 36% stat? It could be a new column, combined with the # of origins somehow, or replace the # origin column entirely. Alternatively, would you hate a pie chart at the top of the page?

Aside: I suspect the 8.8M number is wrong as it's double-counting origins that use multiple CMSs. WordPress market share should be more like 75% of the CMS category. This is something we'll need to fix in BigQuery.

cc @max-ostapenko

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What do you think would be the best way to show the 36% stat? It could be a new column, combined with the # of origins somehow, or replace the # origin column entirely.

If a primary use case is to see the market share, I'm inclined to say replace the origins with the market share, so the table doesn't become overcrowded with info. The other numbers in the table (% of good CWV etc) feel more important than the absolute number of origins to provide a summary of the category data as well, so if a column has to be hidden to make space for the market share, I think origins is the safest one to remove.

Alternatively, would you hate a pie chart at the top of the page?

What about just for the top 5/top 10 technologies within a category? I fear anything more than a handful of items will become too busy/hard to read, at which point it's not very useful anymore and the table becomes a better alternative.

Aside: I suspect the 8.8M number is wrong as it's double-counting origins that use multiple CMSs

If some origins can have multiple CMSes, then adding all the market share percentages can end up being more than 100%, and a pie chart doesn't work anymore? So if that's the case, a different way of highlighting the top techs within a category might be better?

→ Sounds like marketshare might be better for a follow-up issue & PR as well, so we can think through the specifics of it a bit more?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

SGTM

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When filtering to the top 1k rank, I'm seeing technologies with "null" origin counts

https://technology-report-dot-httparchive.uk.r.appspot.com/reports/techreport/category?tech=&geo=ALL&rank=Top+1k&category=CMS

image

"name": "Origins",
"breakdown": "subcategory",
"subcategory": "adoption",
"endpoint": "adoption",
"metric": "origins"
},
{
"key": "good_pct",
"name": "Overall good CWV",
"breakdown": "subcategory",
"subcategory": "overall",
"suffix": "%",
"className": "main-cell pct-value",
"endpoint": "vitals",
"metric": "good_pct"
},
{
"key": "good_pct",
"name": "FID",
"breakdown": "subcategory",
"subcategory": "FID",
"suffix": "%",
"endpoint": "vitals",
"metric": "good_pct"
},
{
"key": "medium_score_pct",
"name": "Accessibility",
"breakdown": "subcategory",
"subcategory": "accessibility",
"endpoint": "lighthouse",
"metric": "median_score_pct"
},
{
"key": "client",
"name": "Client",
"className": "client"
}
]
}
}
}
}
},

Expand Down
61 changes: 59 additions & 2 deletions server/routes.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,15 +66,68 @@ def reports():

return render_template("reports.html", reports=all_reports)


@app.route("/reports/techreport/<page_id>", strict_slashes=False)
def techreport(page_id):
def techreportlanding(page_id):
# Needed for the header dropdown
all_reports = report_util.get_reports()

# Get the configuration for the tech report
tech_report = tech_report_util.get_report()

# Get the settings for the current page
active_tech_report = tech_report.get("pages").get(page_id)

# Add the technologies requested in the URL to the filters
# Use the default configured techs as fallback
# Use ["ALL"] if there is nothing configured
requested_technologies = active_tech_report.get("config").get("default").get(
"app"
) or ["ALL"]

if request.args.get("tech"):
requested_technologies = request.args.get("tech").split(",")

# Get the filters
requested_geo = request.args.get("geo") or "ALL"
requested_rank = request.args.get("rank") or "ALL"
requested_category = request.args.get("category") or "ALL"
filters = {
"geo": requested_geo,
"rank": requested_rank,
"app": requested_technologies,
"category": requested_category,
}

active_tech_report["filters"] = filters

return render_template(
"techreport/%s.html" % page_id,
active_page=page_id,
tech_report_labels=tech_report.get("labels"),
tech_report_config=tech_report.get("config"),
tech_report_page=active_tech_report,
custom_navigation=True,
reports=all_reports,
)

@app.route("/reports/techreport/tech", strict_slashes=False)
def techreport():
# Needed for the header dropdown
all_reports = report_util.get_reports()

# Get the configuration for the tech report
tech_report = tech_report_util.get_report()

# Get the current page_id
requested_technologies = ["ALL"]
if request.args.get("tech"):
requested_technologies = request.args.get("tech").split(",")

if len(requested_technologies) > 1:
page_id = "comparison"
else:
page_id = "drilldown"

# Get the settings for the current page
active_tech_report = tech_report.get("pages").get(page_id)

Expand All @@ -91,20 +144,24 @@ def techreport(page_id):
# Get the filters
requested_geo = request.args.get("geo") or "ALL"
requested_rank = request.args.get("rank") or "ALL"
requested_category = request.args.get("category") or "ALL"
filters = {
"geo": requested_geo,
"rank": requested_rank,
"app": requested_technologies,
"category": requested_category,
}

active_tech_report["filters"] = filters

return render_template(
"techreport/%s.html" % page_id,
active_page=page_id,
requested_page="technology",
tech_report_labels=tech_report.get("labels"),
tech_report_config=tech_report.get("config"),
tech_report_page=active_tech_report,
custom_navigation=True,
reports=all_reports,
)

Expand Down
2 changes: 1 addition & 1 deletion src/js/components/drilldownHeader.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { DataUtils } from "../techreport/utils/data";

function setTitle(title) {
const mainTitle = document.querySelector('h2 span.main-title');
const mainTitle = document.querySelector('h1 span.main-title');
mainTitle.textContent = title;
}

Expand Down
43 changes: 35 additions & 8 deletions src/js/components/filters.js
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@
/* Get the geo and rank filter */
const geo = document.getElementsByName('geo')[0].value;
const rank = document.getElementsByName('rank')[0].value;
const categories = document.getElementsByName('categories')[0].value;

/* Create a string of technologies */
let technologies = [];
Expand All @@ -63,8 +64,13 @@
url.searchParams.delete('rank');
url.searchParams.append('rank', rank);

/* Scroll to the report content */
url.hash = '#report-content';
if(categories) {
url.searchParams.delete('category');
url.searchParams.append('category', categories);
}

// /* Scroll to the report content */
// url.hash = '#report-content';

/* Update the url */
location.href = url;
Expand All @@ -75,7 +81,7 @@
this.technologies = DataUtils.filterDuplicates(this.technologies, 'technology');

/* Get existing tech selectors on the page */
const allTechSelectors = document.querySelectorAll('select.tech');
const allTechSelectors = document.querySelectorAll('select[name="tech"]');
const techNames = this.technologies.map(element => element.app);

/* Update the options inside all of the selectors */
Expand All @@ -95,6 +101,21 @@
techs.unshift({ technology: 'ALL' });

/* Add one option per technology */
<<<<<<< HEAD
Fixed Show fixed Hide fixed
if(document.getElementById('filter-option')) {
techs.forEach((technology) => {
const optionTmpl = document.getElementById('filter-option').content.cloneNode(true);
const option = optionTmpl.querySelector('option');
const formattedTech = technology.technology;
option.textContent = technology.technology;
option.value = formattedTech;
if(formattedTech === techSelector.getAttribute('data-selected')) {
option.selected = true;
}
techSelector.append(optionTmpl);
});
}
=======
techs.forEach((technology) => {
const optionTmpl = document.getElementById('filter-option').content.cloneNode(true);
const option = optionTmpl.querySelector('option');
Expand All @@ -106,6 +127,7 @@
}
techSelector.append(optionTmpl);
});
>>>>>>> main
});
}

Expand Down Expand Up @@ -145,7 +167,7 @@

/* Update the list with categories */
updateCategories() {
const selects = document.querySelectorAll('select[name="categories"]');
const selects = document.querySelectorAll('select[name="categories"]') || document.querySelectorAll('select[name="category"]');

if(this.categories) {
selects.forEach(select => {
Expand All @@ -159,6 +181,9 @@
const sortedCategories = this.categories.sort((a, b) => a.category !== b.category ? a.category < b.category ? -1 : 1 : 0);
sortedCategories.forEach((category) => {
const option = document.createElement('option');
if(category.category === select.getAttribute('data-selected')) {
option.selected = true;
}
option.value = category.category;
option.innerHTML = category.category;
select.append(option);
Expand Down Expand Up @@ -212,16 +237,18 @@
const techId = `tech-${document.querySelectorAll('select.tech[name="tech"]').length + 1}`;
selectElement.setAttribute('id', techId);
labelElement.setAttribute('for', techId);
removeButton.dataset.tech = techId;

categorySelect.setAttribute('id', `${techId}-category`);
categoryLabel.setAttribute('for', `${techId}-category`);
categorySelect.setAttribute('data-tech', techId);

removeButton.classList.remove('hidden');
if(removeButton) {
removeButton.dataset.tech = techId;
removeButton.classList.remove('hidden');

/* Bind functionality to the button */
removeButton.addEventListener('click', this.removeTechnology);
/* Bind functionality to the button */
removeButton.addEventListener('click', this.removeTechnology);
}

/* Fill in all techs and select the first one */
selectElement.innerHTML = document.querySelector('select.tech').innerHTML;
Expand Down
Loading
Loading