diff --git a/ui/src/common/params.ts b/ui/src/common/params.ts new file mode 100644 index 0000000000..68ae78d349 --- /dev/null +++ b/ui/src/common/params.ts @@ -0,0 +1,7 @@ +export enum Params { + Query = 'query', + SortBy = 'sortBy', + Category = 'category', + Kind = 'kind', + Catalog = 'catalog' +} diff --git a/ui/src/containers/App/__snapshots__/App.test.tsx.snap b/ui/src/containers/App/__snapshots__/App.test.tsx.snap index 457d8790f3..0a3b9f9a0a 100644 --- a/ui/src/containers/App/__snapshots__/App.test.tsx.snap +++ b/ui/src/containers/App/__snapshots__/App.test.tsx.snap @@ -8,6 +8,9 @@ exports[`App should render the component correctly and match the snapshot 1`] = + + +
@@ -245,8 +248,8 @@ exports[`App should render the component correctly and match the snapshot 1`] =
- - + + @@ -254,7 +257,7 @@ exports[`App should render the component correctly and match the snapshot 1`] = - +
diff --git a/ui/src/containers/App/index.tsx b/ui/src/containers/App/index.tsx index 4a4b637789..62d7626eaa 100644 --- a/ui/src/containers/App/index.tsx +++ b/ui/src/containers/App/index.tsx @@ -10,6 +10,7 @@ import Footer from '../../components/Footer'; import Resources from '../Resources'; import Authentication from '../../containers/Authentication'; import Details from '../Details'; +import ParseUrl from '../ParseUrl'; import { createProvider } from '../../store/root'; import './App.css'; @@ -19,6 +20,7 @@ const App: React.FC = observer(() => { return ( + } className="hub-page"> diff --git a/ui/src/containers/ParseUrl/ParseUrl.test.tsx b/ui/src/containers/ParseUrl/ParseUrl.test.tsx new file mode 100644 index 0000000000..e7d3017cf0 --- /dev/null +++ b/ui/src/containers/ParseUrl/ParseUrl.test.tsx @@ -0,0 +1,24 @@ +import { when } from 'mobx'; +import { FakeHub } from '../../api/testutil'; +import { createProviderAndStore } from '../../store/root'; + +const TESTDATA_DIR = `src/store/testdata`; +const api = new FakeHub(TESTDATA_DIR); +const { root } = createProviderAndStore(api); + +describe('ParseUrl component', () => { + it('it can set url params to resource store', (done) => { + const { resources } = root; + when( + () => { + return !resources.isLoading; + }, + () => { + resources.setURLParams('?/query=ansible'); + expect(resources.urlParams).toBe('?/query=ansible'); + + done(); + } + ); + }); +}); diff --git a/ui/src/containers/ParseUrl/index.tsx b/ui/src/containers/ParseUrl/index.tsx new file mode 100644 index 0000000000..398c192a39 --- /dev/null +++ b/ui/src/containers/ParseUrl/index.tsx @@ -0,0 +1,23 @@ +import React from 'react'; +import { useMst } from '../../store/root'; +import { Params } from '../../common/params'; + +const ParseUrl: React.FC = () => { + const { resources } = useMst(); + + if (window.location.search) { + const searchParams = new URLSearchParams(window.location.search); + if (searchParams.has(Params.Query)) { + resources.setSearch(searchParams.get(Params.Query)); + } + if (searchParams.has(Params.SortBy)) { + resources.setSortBy(searchParams.get(Params.SortBy)); + } + + // Storing url params to store inorder to parse the url only after successfully resource load + resources.setURLParams(window.location.search); + } + + return <> ; +}; +export default ParseUrl; diff --git a/ui/src/containers/Resources/__snapshots__/Resources.test.tsx.snap b/ui/src/containers/Resources/__snapshots__/Resources.test.tsx.snap index e89791eff8..f32981f038 100644 --- a/ui/src/containers/Resources/__snapshots__/Resources.test.tsx.snap +++ b/ui/src/containers/Resources/__snapshots__/Resources.test.tsx.snap @@ -5,7 +5,7 @@ exports[`Resource Component should render the resources component 1`] = ` - +
@@ -740,7 +740,7 @@ exports[`Resource Component should render the resources component 1`] = `
-
+
diff --git a/ui/src/containers/Resources/index.tsx b/ui/src/containers/Resources/index.tsx index b824bc0b21..f0abf418f4 100644 --- a/ui/src/containers/Resources/index.tsx +++ b/ui/src/containers/Resources/index.tsx @@ -1,5 +1,5 @@ import React from 'react'; -import { useObserver } from 'mobx-react'; +import { observer } from 'mobx-react'; import { EmptyState, EmptyStateIcon, @@ -14,12 +14,24 @@ import { useHistory } from 'react-router-dom'; import { useMst } from '../../store/root'; import { IResource } from '../../store/resource'; import Cards from '../../components/Cards'; +import { UpdateURL } from '../../utils/updateUrl'; import './Resources.css'; -const Resources = () => { - const { resources } = useMst(); +const Resources: React.FC = observer(() => { + const { resources, categories } = useMst(); + const { catalogs, kinds, search, sortBy } = resources; const history = useHistory(); + + React.useEffect(() => { + const selectedcategories = categories.selectedByName.join(','); + const selectedKinds = [...kinds.selected].join(','); + const selectedCatalogs = catalogs.selectedByName.join(','); + + const url = UpdateURL(search, sortBy, selectedcategories, selectedKinds, selectedCatalogs); + if (!resources.isLoading) history.replace(`?${url}`); + }, [search, sortBy, categories.selectedByName, kinds.selected, catalogs.selected]); + const clearFilter = () => { resources.clearAllFilters(); history.push('/'); @@ -43,12 +55,10 @@ const Resources = () => { ); }; - return useObserver(() => - resources.resources.size === 0 ? ( - - ) : ( - {checkResources(resources.filteredResources)} - ) + return resources.resources.size === 0 ? ( + + ) : ( + {checkResources(resources.filteredResources)} ); -}; +}); export default Resources; diff --git a/ui/src/containers/Search/index.tsx b/ui/src/containers/Search/index.tsx index f31fbc3e18..2c35bd1ad8 100644 --- a/ui/src/containers/Search/index.tsx +++ b/ui/src/containers/Search/index.tsx @@ -1,4 +1,4 @@ -import React, { useEffect } from 'react'; +import React from 'react'; import { useHistory } from 'react-router-dom'; import { useObserver } from 'mobx-react'; import { TextInput } from '@patternfly/react-core'; @@ -9,35 +9,13 @@ import './Search.css'; const Search: React.FC = () => { const { resources } = useMst(); - // to get query params from the url - const searchParams = new URLSearchParams(window.location.search); - const query = searchParams.get('query') || ' '; - - useEffect(() => { - if (query !== ' ') { - resources.setSearch(query); - } - }, [query, resources]); - - const setParams = ({ query = '' }) => { - const searchParams = new URLSearchParams(); - searchParams.set('query', query); - return searchParams.toString(); - }; - - const updateURL = (text: string) => { - const url = setParams({ query: text }); - if (window.location.pathname === '/') history.replace(`?${url}`); - }; - const onSearchChange = useDebounce(resources.search, 400); const history = useHistory(); const onSearchKeyPress = (e: React.KeyboardEvent) => { if (e.key === 'Enter') { e.preventDefault(); - history.push('/'); - updateURL(resources.search); + if (window.location.pathname !== '/') history.push('/'); } return; }; @@ -48,7 +26,6 @@ const Search: React.FC = () => { type="search" onChange={(resourceName: string) => { resources.setSearch(resourceName); - updateURL(resourceName); return onSearchChange; }} onKeyPress={onSearchKeyPress} diff --git a/ui/src/store/catalog.test.ts b/ui/src/store/catalog.test.ts index 0caf02a1df..ba214e1480 100644 --- a/ui/src/store/catalog.test.ts +++ b/ui/src/store/catalog.test.ts @@ -90,4 +90,25 @@ describe('Store Object', () => { done(); }); + + it('should toggle catalogs by name and can get selected catlogs by name', (done) => { + const store = CatalogStore.create({}); + + const item = Catalog.create({ + id: 1, + name: 'tekton', + type: 'community' + }); + + store.add(item); + + store.toggleByName('Tekton'); + const catalogs = store.items.get('1'); + assert(catalogs); + + expect(catalogs.selected).toBe(true); + expect(store.selectedByName).toEqual(['tekton']); + + done(); + }); }); diff --git a/ui/src/store/catalog.ts b/ui/src/store/catalog.ts index d1fdf9ebc6..5ce7743c56 100644 --- a/ui/src/store/catalog.ts +++ b/ui/src/store/catalog.ts @@ -1,5 +1,6 @@ import { Instance, types } from 'mobx-state-tree'; import { Icons } from '../common/icons'; +import { titleCase } from '../common/titlecase'; const icons: { [catalog: string]: Icons } = { official: Icons.Cat, @@ -41,6 +42,13 @@ export const CatalogStore = types self.items.forEach((c) => { c.selected = false; }); + }, + toggleByName(name: string) { + self.items.forEach((c) => { + if (titleCase(c.name) === name) { + c.selected = true; + } + }); } })) @@ -58,5 +66,13 @@ export const CatalogStore = types }); return list; + }, + + /* This view returns list of the selected catalos's name instead of id + to avoid loop on it inorder to get catalogs name */ + get selectedByName() { + return Array.from(self.items.values()) + .filter((c: ICatalog) => c.selected) + .reduce((acc: string[], c: ICatalog) => [...acc, c.name], []); } })); diff --git a/ui/src/store/category.test.ts b/ui/src/store/category.test.ts index 5d7bb52447..542ba39e24 100644 --- a/ui/src/store/category.test.ts +++ b/ui/src/store/category.test.ts @@ -122,4 +122,54 @@ describe('Store functions', () => { } ); }); + + it('can toggle the category by name', (done) => { + const store = CategoryStore.create({}, { api }); + expect(store.count).toBe(0); + expect(store.isLoading).toBe(true); + + when( + () => !store.isLoading, + () => { + expect(store.count).toBe(5); + expect(store.isLoading).toBe(false); + + store.toggleByName('Build Tools'); + + const categories = store.items.get('1'); + assert(categories); + expect(categories.selected).toBe(true); + + done(); + } + ); + }); + + it('can return the all selected catgories in a list', (done) => { + const store = CategoryStore.create({}, { api }); + expect(store.count).toBe(0); + expect(store.isLoading).toBe(true); + + when( + () => !store.isLoading, + () => { + expect(store.count).toBe(5); + expect(store.isLoading).toBe(false); + + // Gets the category with id as 1 + const c1 = store.items.get('1'); + assert(c1); + c1.toggle(); + + // Gets the category with id as 2 + const c2 = store.items.get('2'); + assert(c2); + c2.toggle(); + + expect(store.selectedByName.sort()).toEqual(['Build Tools', 'CLI'].sort()); + + done(); + } + ); + }); }); diff --git a/ui/src/store/category.ts b/ui/src/store/category.ts index ce8f2b1c25..3d5bb49adc 100644 --- a/ui/src/store/category.ts +++ b/ui/src/store/category.ts @@ -45,6 +45,13 @@ export const CategoryStore = types return Array.from(self.items.values()); }, + // This returns list of selected category's name + get selectedByName() { + return Array.from(self.items.values()) + .filter((c: ICategory) => c.selected) + .reduce((acc: string[], c: ICategory) => [...acc, c.name], []); + }, + get selectedTags() { return new Set( Array.from(self.items.values()) @@ -67,6 +74,14 @@ export const CategoryStore = types self.items.forEach((c) => { c.selected = false; }); + }, + + toggleByName(name: string) { + self.items.forEach((c) => { + if (c.name === name) { + c.selected = true; + } + }); } })) diff --git a/ui/src/store/resource.test.ts b/ui/src/store/resource.test.ts index 2eeb18f65d..e915173b33 100644 --- a/ui/src/store/resource.test.ts +++ b/ui/src/store/resource.test.ts @@ -679,4 +679,27 @@ describe('Store functions', () => { } ); }); + + it('it should parse the url and can update the store', (done) => { + const store = ResourceStore.create( + {}, + { + api, + categories: CategoryStore.create({}, { api }) + } + ); + expect(store.isLoading).toBe(true); + when( + () => !store.isLoading, + () => { + store.setURLParams('?category=Automation%2CBuild+Tools&catalog=tekton'); + store.parseUrl(); + + expect(store.filteredResources.length).toBe(1); + expect(store.categories.selectedByName).toEqual(['Build Tools']); + + done(); + } + ); + }); }); diff --git a/ui/src/store/resource.ts b/ui/src/store/resource.ts index 17b4365e7f..8ce15dfbe2 100644 --- a/ui/src/store/resource.ts +++ b/ui/src/store/resource.ts @@ -7,6 +7,7 @@ import { Api } from '../api'; import { Catalog, CatalogStore } from './catalog'; import { Kind, KindStore } from './kind'; import { assert } from './utils'; +import { Params } from '../common/params'; export const updatedAt = types.custom({ name: 'momentDate', @@ -111,6 +112,7 @@ export const ResourceStore = types sortBy: types.optional(types.enumeration(Object.values(SortByFields)), SortByFields.Unknown), tags: types.optional(types.map(Tag), {}), search: '', + urlParams: '', err: '', isLoading: true }) @@ -139,6 +141,9 @@ export const ResourceStore = types setSortBy(field: string) { const key: SortByFields = SortByFields[field as keyof typeof SortByFields]; self.sortBy = key; + }, + setURLParams(url: string) { + self.urlParams = url; } })) @@ -149,6 +154,32 @@ export const ResourceStore = types self.categories.clearSelected(); self.setSearch(''); self.setSortBy(SortByFields.Unknown); + }, + + parseUrl() { + const searchParams = new URLSearchParams(self.urlParams); + if (searchParams.has(Params.Category)) { + const categoriesParams = searchParams.getAll(Params.Category)[0].split(','); + categoriesParams.forEach((t: string) => { + self.categories.toggleByName(t); + }); + } + + if (searchParams.has(Params.Catalog)) { + const catalogsParams = searchParams.getAll(Params.Catalog)[0].split(','); + catalogsParams.forEach((t: string) => { + self.catalogs.toggleByName(t); + }); + } + + if (searchParams.has(Params.Kind)) { + const kindsParams = searchParams.getAll(Params.Kind)[0].split(','); + kindsParams.forEach((t: string) => { + const kind = self.kinds.items.get(t); + assert(kind); + kind.toggle(); + }); + } } })) @@ -251,6 +282,9 @@ export const ResourceStore = types r.versions.push(r.latestVersion); self.add(r); }); + + // Url parsing after resource load + if (self.urlParams) self.parseUrl(); } catch (err) { self.err = err.toString(); } diff --git a/ui/src/utils/updateUrl.test.ts b/ui/src/utils/updateUrl.test.ts new file mode 100644 index 0000000000..45fa0ce1a1 --- /dev/null +++ b/ui/src/utils/updateUrl.test.ts @@ -0,0 +1,8 @@ +import { UpdateURL } from './updateUrl'; + +describe('Test UpdateUrl function', () => { + it('Test UpdateUrl function', () => { + const val = UpdateURL('ansible', 'rating', 'cli', 'task', 'Tekton'); + expect(val).toEqual('query=ansible&sortBy=rating&category=cli&kind=task&catalog=Tekton'); + }); +}); diff --git a/ui/src/utils/updateUrl.ts b/ui/src/utils/updateUrl.ts new file mode 100644 index 0000000000..ee222bf364 --- /dev/null +++ b/ui/src/utils/updateUrl.ts @@ -0,0 +1,48 @@ +import { Params } from '../common/params'; +import { SortByFields } from '../store/resource'; +import { titleCase } from '../common/titlecase'; + +// This function returns all selected filters in a combination of params +export const UpdateURL = ( + search: string, + sort: string, + categories: string, + kinds: string, + catalogs: string +) => { + // To get URLSearchParams object instance + const searchParams = new URLSearchParams(window.location.search); + + // To delete query params if search is empty + if (!search && searchParams.has(Params.Query)) searchParams.delete(Params.Query); + + // Sets query params + if (search) searchParams.set(Params.Query, search); + + //To delete sort params if doesn't selected any sort + if (sort === SortByFields.Unknown && searchParams.has(Params.SortBy)) + searchParams.delete(Params.SortBy); + + // Set sort params + if (sort !== SortByFields.Unknown) searchParams.set(Params.SortBy, sort); + + // To delete category params if doesn't selected any category + if (!categories && searchParams.has(Params.Category)) searchParams.delete(Params.Category); + + // Sets category params + if (categories) searchParams.set(Params.Category, categories); + + // To delete kind params if doesn't selected any kind + if (!kinds && searchParams.has(Params.Kind)) searchParams.delete(Params.Kind); + + // Sets Kind params + if (kinds) searchParams.set(Params.Kind, kinds); + + // To delete catalog params if doesn't selected any catalog + if (!catalogs && searchParams.has(Params.Catalog)) searchParams.delete(Params.Catalog); + + // Sets catalos params + if (catalogs) searchParams.set(Params.Catalog, titleCase(catalogs)); + + return searchParams.toString(); +};