From 7d75cf384a0200e5764cc241a0b8d9977378f84a Mon Sep 17 00:00:00 2001 From: Forrest Collman Date: Mon, 18 Mar 2024 12:33:48 -0700 Subject: [PATCH] feat: adds display of tag descriptions and regexp searching for tags (#528) --- .../property_map.ts | 64 ++++++++++++++++--- src/ui/segment_list.ts | 10 ++- 2 files changed, 62 insertions(+), 12 deletions(-) diff --git a/src/segmentation_display_state/property_map.ts b/src/segmentation_display_state/property_map.ts index acb5131ca..aa1c79853 100644 --- a/src/segmentation_display_state/property_map.ts +++ b/src/segmentation_display_state/property_map.ts @@ -677,7 +677,7 @@ export function parseSegmentQuery( }); continue; } - if (labels === undefined) { + if (labels === undefined && tagNames.length == 0) { errors.push({ begin: startIndex, end: endIndex, @@ -778,7 +778,7 @@ export function parseSegmentQuery( }); continue; } - if (labels === undefined) { + if (labels === undefined && tagNames.length == 0) { errors.push({ begin: startIndex, end: endIndex, @@ -806,6 +806,7 @@ export interface TagCount { tag: string; tagIndex: number; count: number; + desc: string; } export interface PropertyHistogram { @@ -860,7 +861,10 @@ export function executeSegmentQuery( } const properties = inlineProperties?.properties; const totalIds = inlineProperties.ids.length / 2; + const totalTags = db?.tags?.tags?.length || 0; let indices = makeIndicesArray(totalIds, totalIds); + const showTags = makeIndicesArray(totalTags, totalTags); + showTags.fill(1); for (let i = 0; i < totalIds; ++i) { indices[i] = i; } @@ -877,16 +881,50 @@ export function executeSegmentQuery( } indices = indices.subarray(0, outIndex); }; + const filterByTagDescriptions = (regexp: RegExp) => { + const tagDescriptions = db!.tags!.tagDescriptions!; + const tags = db!.tags!.tags!; + + // reset showTags + showTags.fill(0); + + // iterate over tagDescriptions with a for loop + for (let i = 0; i < tagDescriptions.length; i++) { + if (tagDescriptions[i].match(regexp) !== null) { + showTags[i] = 1; + } + if (tags[i].match(regexp) !== null) { + showTags[i] = 1; + } + } + }; // Filter by label if (query.regexp !== undefined || query.prefix !== undefined) { - const values = db!.labels!.values; const { regexp, prefix } = query; - if (regexp !== undefined) { - filterIndices((index) => values[index].match(regexp) !== null); + if (db!.labels !== undefined) { + const values = db!.labels!.values; + if (regexp !== undefined) { + filterIndices((index) => values[index].match(regexp) !== null); + } + if (prefix !== undefined) { + filterIndices((index) => values[index].startsWith(prefix)); + } } - if (prefix !== undefined) { - filterIndices((index) => values[index].startsWith(prefix)); + // if the regular expression returns nothing + // then assudme the user wants to search through the tags + // and/or tag descriptions + if ( + (indices.length == 0 && regexp !== undefined) || + (db!.labels == undefined && regexp != undefined) + ) { + indices = makeIndicesArray(totalIds, totalIds); + for (let i = 0; i < totalIds; ++i) { + indices[i] = i; + } + filterByTagDescriptions(regexp); + // reset regexp to none so that it doesn't get applied again + query.regexp = undefined; } } @@ -924,7 +962,6 @@ export function executeSegmentQuery( const regexp = new RegExp(pattern); filterIndices((index) => values[index].match(regexp) !== null); } - let intermediateIndicesMask: IndicesArray | undefined; let intermediateIndices: IndicesArray | undefined; @@ -970,7 +1007,7 @@ export function executeSegmentQuery( let tagStatistics: TagCount[] = []; if (tagsProperty !== undefined) { const tagStatisticsInQuery: TagCount[] = []; - const { tags, values } = tagsProperty; + const { tags, values, tagDescriptions } = tagsProperty; const tagCounts = new Uint32Array(tags.length); for (let i = 0, n = indices.length; i < n; ++i) { const value = values[indices[i]]; @@ -983,9 +1020,16 @@ export function executeSegmentQuery( tagIndex < numTags; ++tagIndex ) { + if (showTags[tagIndex] === 0) continue; const count = tagCounts[tagIndex]; const tag = tags[tagIndex]; - const tagCount = { tag, tagIndex, count: tagCounts[tagIndex] }; + const tagDesc = tagDescriptions[tagIndex]; + const tagCount = { + tag, + tagIndex, + count: tagCounts[tagIndex], + desc: tagDesc, + }; if (query.includeTags.includes(tag) || query.excludeTags.includes(tag)) { tagStatisticsInQuery.push(tagCount); } else if (count > 0) { diff --git a/src/ui/segment_list.ts b/src/ui/segment_list.ts index 2d18e5f22..ccadae84a 100644 --- a/src/ui/segment_list.ts +++ b/src/ui/segment_list.ts @@ -990,12 +990,18 @@ function renderTagSummary( const filterQuery = queryResult.query as FilterQuery; const tagList = document.createElement("div"); tagList.classList.add("neuroglancer-segment-query-result-tag-list"); - for (const { tag, count } of tags) { + for (const { tag, count, desc } of tags) { const tagElement = document.createElement("div"); tagElement.classList.add("neuroglancer-segment-query-result-tag"); const tagName = document.createElement("span"); tagName.classList.add("neuroglancer-segment-query-result-tag-name"); - tagName.textContent = tag; + // if the tag is different than desc, show both + if (tag !== desc && desc !== undefined && desc !== "") { + tagName.textContent = tag + " (" + desc + ")"; + } else { + tagName.textContent = tag; + } + tagList.appendChild(tagElement); const included = filterQuery.includeTags.includes(tag); const excluded = filterQuery.excludeTags.includes(tag);