From cdd926cfc128e2b136fc5b9f3f98d3755e86e4b3 Mon Sep 17 00:00:00 2001
From: Joli <69351113+L-Joli@users.noreply.github.com>
Date: Mon, 16 Oct 2023 10:55:14 -0400
Subject: [PATCH] feat: add fuzzy search to projects list dropdown (#208)
---
src/components/fuzzySearch.ts | 93 +++++++++++++-------
src/views/home/HomeViewProvider.ts | 55 +++++++++---
src/views/inspector/InspectorViewProvider.ts | 30 ++++---
src/webview/homeView.ts | 13 +++
src/webview/inspectorView.ts | 4 +
5 files changed, 143 insertions(+), 52 deletions(-)
diff --git a/src/components/fuzzySearch.ts b/src/components/fuzzySearch.ts
index f5ace0c..f6609a3 100644
--- a/src/components/fuzzySearch.ts
+++ b/src/components/fuzzySearch.ts
@@ -3,15 +3,23 @@ import { WebviewApi } from "vscode-webview";
export enum SearchType {
variables = 'variables',
- features = 'features'
+ features = 'features',
+ projects = 'projects',
}
+
+type LocalStorageKey = `${SearchType}${string}` | SearchType
+
//View Provider
-export const getCustomDropdown = (optionElements: string, selectedValue: string) => {
+export const getCustomDropdown = (optionElements: string, selectedValue: string, folderIndex?: number, dataType?: string) => {
+ const folderAttribute = folderIndex !== undefined ? `data-folder="${folderIndex}"` : ''
+ const dataTypeAttribute = dataType ? `data-type="${dataType}"` : ''
+ const inputId = folderIndex !== undefined ? `id="dropdown-input-${folderIndex}"` : ''
+ const optionsListId = folderIndex !== undefined ? `id="dropdown-optionsList-${folderIndex}"` : ''
return `
-
+
^
-
`
@@ -28,11 +36,13 @@ export const getCustomDropdown = (optionElements: string, selectedValue: string)
*/
export function handleFuzzySearchDropdown(
vscode: WebviewApi
,
- searchType: SearchType,
+ localStorageKey: LocalStorageKey,
fuseOptions: Fuse.IFuseOptions = {
keys: ['label', 'value'],
threshold: 0.5,
- }
+ },
+ inputSelector: string = '.dropdown-input',
+ optionsListSelector: string = '.dropdown-options'
) {
window.addEventListener('message', (event) => {
const message = event.data
@@ -42,17 +52,10 @@ export function handleFuzzySearchDropdown(
}
})
- const localStorageData = localStorage.getItem(searchType)
-
- if (!localStorageData) {
- return
- }
-
- const input = document.querySelector('.dropdown-input') as HTMLInputElement
- const optionsList = document.querySelector('.dropdown-options') as HTMLDivElement
-
- const searchData: { label: string, value?: string }[] = JSON.parse(localStorageData)
- const fuse = new Fuse([...searchData], fuseOptions)
+ const input = document.querySelector(inputSelector) as HTMLInputElement
+ const optionsList = document.querySelector(optionsListSelector) as HTMLDivElement
+ const folderIndex = optionsList?.getAttribute('data-folder')
+ const selectedValueKey = `selectedValue${folderIndex || ''}`
const createAndAppendOption = (displayValue: string, dataAttributeValue?: string) => {
const option = document.createElement('div')
@@ -63,15 +66,57 @@ export function handleFuzzySearchDropdown(
// Event listeners to handle dropdown behavior
document.addEventListener('click', () => {
+ if (!optionsList?.classList.contains('visible')) {
+ return
+ }
+ const selectedValue = localStorage.getItem(selectedValueKey)
+ if (selectedValue) {
+ input.value = selectedValue
+ }
optionsList?.classList.remove('visible')
})
+ optionsList?.addEventListener('click', (event) => {
+ const selectedOption = event.target as HTMLElement
+ if (selectedOption.tagName === 'DIV') {
+ const selectedValue = selectedOption.getAttribute('data-value')
+ const selectedType = optionsList.getAttribute('data-type')
+ if (selectedValue) {
+ input.value = selectedOption.innerText
+ vscode.postMessage({
+ type: selectedType || 'key',
+ value: selectedValue,
+ ...(optionsList.dataset.folder && { folderIndex: optionsList.dataset.folder })
+ })
+ optionsList.classList.remove('visible')
+ }
+ }
+ })
+
input?.addEventListener('click', (event) => {
+ const { value } = event.target as HTMLInputElement
+ localStorage.setItem(selectedValueKey, value)
event.stopPropagation()
optionsList?.classList.toggle('visible')
})
+
+ let localStorageData: string | null = localStorage.getItem(localStorageKey)
+ const refreshLocalStorageData = () => {
+ localStorageData = localStorage.getItem(localStorageKey)
+ }
input?.addEventListener('input', () => {
+ if (!localStorageData) {
+ vscode.postMessage({
+ type: 'setFuseData',
+ ...(folderIndex && { folderIndex: folderIndex })
+ })
+ refreshLocalStorageData()
+ return
+ }
+
+ const searchData: { label: string, value?: string }[] = JSON.parse(localStorageData)
+ const fuse = new Fuse([...searchData], fuseOptions)
const inputValue = input?.value.toLowerCase().trim()
optionsList.innerHTML = ''
@@ -95,18 +140,4 @@ export function handleFuzzySearchDropdown(
}
})
- optionsList.addEventListener('click', (event) => {
- const selectedOption = event.target as HTMLElement
- if (selectedOption.tagName === 'DIV') {
- const selectedValue = selectedOption.getAttribute('data-value')
- if (selectedValue) {
- vscode.postMessage({
- type: 'key',
- value: selectedValue,
- })
- optionsList.classList.remove('visible')
- }
- }
- })
-
}
\ No newline at end of file
diff --git a/src/views/home/HomeViewProvider.ts b/src/views/home/HomeViewProvider.ts
index 801fdae..d943cfe 100644
--- a/src/views/home/HomeViewProvider.ts
+++ b/src/views/home/HomeViewProvider.ts
@@ -1,24 +1,27 @@
import * as vscode from 'vscode'
import { getNonce } from '../../utils/getNonce'
-import { BaseCLIController, OrganizationsCLIController, ProjectsCLIController } from '../../cli'
+import { BaseCLIController, OrganizationsCLIController, Project, ProjectsCLIController } from '../../cli'
import path from 'path'
import { COMMAND_LOGOUT } from '../../commands/logout'
import { KEYS, StateManager } from '../../StateManager'
import { updateRepoConfig } from '../../utils/updateRepoConfigProject'
import { executeRefreshAllCommand } from '../../commands'
import { loginAndRefresh } from '../../utils/loginAndRefresh'
+import { SearchType, getCustomDropdown } from '../../components/fuzzySearch'
type HomeViewMessage =
| { type: 'project' | 'organization', value: string, folderIndex: number }
| { type: 'config', folderIndex: number }
| { type: 'login', folderIndex: number }
| { type: 'logout' }
+ | { type: 'setFuseData', folderIndex: number }
export class HomeViewProvider implements vscode.WebviewViewProvider {
_view?: vscode.WebviewView
_doc?: vscode.TextDocument
isRefreshing: boolean = false
webviewIsDisposed: boolean = false
+ projectOptions: Record = []
constructor(private readonly _extensionUri: vscode.Uri) {}
@@ -46,6 +49,11 @@ export class HomeViewProvider implements vscode.WebviewViewProvider {
const folder = vscode.workspace.workspaceFolders?.[data.folderIndex]
if (!folder) { return }
+ if (data.type === 'setFuseData') {
+ this.setFuseSearchData(this.projectOptions[folder.index], folder)
+ return
+ }
+
if (data.type === 'login') {
await loginAndRefresh([folder])
webviewView.webview.html = await this._getHtmlForWebview(webviewView.webview)
@@ -116,6 +124,21 @@ export class HomeViewProvider implements vscode.WebviewViewProvider {
await this.refreshAll()
}
+ private postMessageToWebview(message: unknown) {
+ this._view?.webview.postMessage(message)
+ }
+
+ private setFuseSearchData(data: Data[], folder: vscode.WorkspaceFolder) {
+ this.postMessageToWebview({ type: 'searchData', searchType: `${SearchType.projects}${folder.index}`, value: JSON.stringify(data)})
+ }
+
+ private getSelectedProject(activeProjectKey: string | undefined, projects: Record) {
+ if (!activeProjectKey) {
+ return ''
+ }
+ return projects[activeProjectKey]?.name || projects[activeProjectKey]?.key
+ }
+
private async getBodyHtml(folder: vscode.WorkspaceFolder, showHeader?: boolean): Promise {
const organizationsController = new OrganizationsCLIController(folder)
const projectsController = new ProjectsCLIController(folder)
@@ -125,7 +148,13 @@ export class HomeViewProvider implements vscode.WebviewViewProvider {
const activeOrganizationName = StateManager.getFolderState(folder.name, KEYS.ORGANIZATION)?.name
const isLoggedIn = StateManager.getFolderState(folder.name, KEYS.LOGGED_IN)
- const projectId = `project${folder.index}`
+ const projectOptions = Object.values(projects).map((project) => ({
+ label: project.name || project.key,
+ value: project.key,
+ }))
+ this.projectOptions[folder.index] = projectOptions
+ this.setFuseSearchData(projectOptions, folder)
+
const organizationId = `organization${folder.index}`
const editButtonId = `editConfigButton${folder.index}`
@@ -133,13 +162,10 @@ export class HomeViewProvider implements vscode.WebviewViewProvider {
const orgOptions = Object.values(organizations).map((organization) =>
`${organization.display_name || organization.name}`
)
- const projectOptions = Object.values(projects).map((project) =>
- `${project.name}`
+ const projectOptionElements = Object.values(projects).map((project) =>
+ `${project.name}
`
)
- if (!activeProjectKey) {
- projectOptions.unshift(`Select a project...`)
- }
-
+ const selectedProject = this.getSelectedProject(activeProjectKey, projects)
return `
${
showHeader ? `
@@ -162,9 +188,7 @@ export class HomeViewProvider implements vscode.WebviewViewProvider {
-
- ${projectOptions.join('')}
-
+ ${getCustomDropdown(projectOptionElements.join(''), selectedProject, folder.index, 'project')}
@@ -207,6 +231,14 @@ export class HomeViewProvider implements vscode.WebviewViewProvider {
const homeViewStylesUri = webview.asWebviewUri(
vscode.Uri.joinPath(this._extensionUri, 'media', 'styles', 'homeView.css'),
)
+ const dropdownStylesUri = webview.asWebviewUri(
+ vscode.Uri.joinPath(
+ this._extensionUri,
+ 'media',
+ 'styles',
+ 'customDropdown.css',
+ ),
+ )
const webViewUri = webview.asWebviewUri(
vscode.Uri.joinPath(this._extensionUri, 'out', 'homeView.js'),
)
@@ -236,6 +268,7 @@ export class HomeViewProvider implements vscode.WebviewViewProvider {
+
diff --git a/src/views/inspector/InspectorViewProvider.ts b/src/views/inspector/InspectorViewProvider.ts
index a70cc3c..8be1e37 100644
--- a/src/views/inspector/InspectorViewProvider.ts
+++ b/src/views/inspector/InspectorViewProvider.ts
@@ -32,6 +32,7 @@ type InspectorViewMessage =
}
| { type: 'folder'; value: number }
| { type: 'jsonReadonly'; value: string }
+ | { type: 'setFuseData' }
const htmlMessage = (message: string) => `
@@ -96,16 +97,8 @@ export class InspectorViewProvider implements vscode.WebviewViewProvider {
this.selectedKey = ''
- // Send variables data to the webview to initialize fuse
- const variableOptions = this.orderedVariables.map(variable => ({
- label: variable.key
- }))
- const featureOptions = this.orderedFeatures.map(feature => ({
- label: feature.name || feature.key,
- value: feature._id
- }))
- this.postMessageToWebview({type: 'searchData', searchType: SearchType.variables, value: JSON.stringify(variableOptions)})
- this.postMessageToWebview({type: 'searchData', searchType: SearchType.features, value: JSON.stringify(featureOptions)})
+ // Send data to the webview to initialize Fuse
+ this.setFuseSearchData()
} catch (e) {
vscode.window.showErrorMessage(
`Error initializing features and variables in inspector: ${e}`,
@@ -133,6 +126,11 @@ export class InspectorViewProvider implements vscode.WebviewViewProvider {
webviewView.webview.onDidReceiveMessage(
async (data: InspectorViewMessage) => {
+ if (data.type === 'setFuseData') {
+ this.setFuseSearchData()
+ return
+ }
+
if (data.type === 'variableOrFeature') {
this.selectedType = data.value
this.selectedKey = ''
@@ -175,6 +173,18 @@ export class InspectorViewProvider implements vscode.WebviewViewProvider {
this.webviewIsDisposed = false
}
+ private setFuseSearchData() {
+ const variableOptions = this.orderedVariables.map(variable => ({
+ label: variable.key
+ }))
+ const featureOptions = this.orderedFeatures.map(feature => ({
+ label: feature.name || feature.key,
+ value: feature._id
+ }))
+ this.postMessageToWebview({ type: 'searchData', searchType: SearchType.variables, value: JSON.stringify(variableOptions) })
+ this.postMessageToWebview({ type: 'searchData', searchType: SearchType.features, value: JSON.stringify(featureOptions) })
+ }
+
private async refreshInspectorView() {
if (!this._view?.webview) {
return
diff --git a/src/webview/homeView.ts b/src/webview/homeView.ts
index 34a88d0..ceb54b2 100644
--- a/src/webview/homeView.ts
+++ b/src/webview/homeView.ts
@@ -1,4 +1,5 @@
import { provideVSCodeDesignSystem, vsCodeDropdown, vsCodeOption, Dropdown } from "@vscode/webview-ui-toolkit";
+import { SearchType, handleFuzzySearchDropdown } from "../components/fuzzySearch";
provideVSCodeDesignSystem().register(vsCodeDropdown(), vsCodeOption());
@@ -46,6 +47,18 @@ function main() {
dropdown.addEventListener('change', handleDropdownValueChange);
}
+ const optionsLists = document.querySelectorAll('.dropdown-options') as NodeListOf
+ optionsLists.forEach(optionsList => {
+ const folderIndexAttribute = optionsList.getAttribute('data-folder')
+ const folderIndex = folderIndexAttribute ? folderIndexAttribute : null
+
+ if (folderIndex) {
+ handleFuzzySearchDropdown(vscode, `${SearchType.projects}${folderIndex}`, undefined, `#dropdown-input-${folderIndex}`, `#dropdown-optionsList-${folderIndex}`)
+ } else {
+ handleFuzzySearchDropdown(vscode, SearchType.projects)
+ }
+ })
+
const editConfigButtons = document.getElementsByClassName("edit-config-button")
for (let i = 0; i < editConfigButtons.length; i++) {
// Edit config button
diff --git a/src/webview/inspectorView.ts b/src/webview/inspectorView.ts
index 7d6d1a4..0cc8453 100644
--- a/src/webview/inspectorView.ts
+++ b/src/webview/inspectorView.ts
@@ -20,6 +20,10 @@ if (focusedElement) {
window.addEventListener('message', (event) => {
const message = event.data
+ // if message type is 'searchData' then we don't want to post it back to the extension
+ if (message.type === 'searchData') {
+ return
+ }
vscode.postMessage(message)
})