From a468f6f29d620726bdfecde4f2d3047f6dc780ce Mon Sep 17 00:00:00 2001 From: Richard Tibbles Date: Tue, 17 Dec 2024 15:12:07 -0800 Subject: [PATCH] Add primary language filtering to useBaseSearch. Allow language selection logic to be reused. --- .../composables/__tests__/useSearch.spec.js | 438 ------------------ .../assets/src/composables/useContentLink.js | 5 + .../assets/src/views/LibraryPage/index.vue | 84 +++- .../assets/src/views/TopicsPage/index.vue | 16 +- .../assets/test/views/library-page.spec.js | 23 +- .../kolibri-common/components/SearchChips.vue | 17 +- .../AccordionSelectGroup.vue | 26 +- .../SearchFiltersPanel/LanguageSelector.vue | 65 +++ .../SearchFiltersPanel/SelectGroup.vue | 54 +-- .../components/SearchFiltersPanel/index.vue | 6 +- .../composables/__mocks__/useBaseSearch.js | 5 + .../__tests__/useBaseSearch.spec.js | 53 ++- .../composables/useBaseSearch.js | 194 ++++++-- packages/kolibri/utils/i18n.js | 26 +- 14 files changed, 411 insertions(+), 601 deletions(-) delete mode 100644 kolibri/plugins/learn/assets/src/composables/__tests__/useSearch.spec.js create mode 100644 packages/kolibri-common/components/SearchFiltersPanel/LanguageSelector.vue diff --git a/kolibri/plugins/learn/assets/src/composables/__tests__/useSearch.spec.js b/kolibri/plugins/learn/assets/src/composables/__tests__/useSearch.spec.js deleted file mode 100644 index 3b0d5e1284e..00000000000 --- a/kolibri/plugins/learn/assets/src/composables/__tests__/useSearch.spec.js +++ /dev/null @@ -1,438 +0,0 @@ -import { get, set } from '@vueuse/core'; -import VueRouter from 'vue-router'; -import Vue, { nextTick, ref } from 'vue'; -import ContentNodeResource from 'kolibri-common/apiResources/ContentNodeResource'; -import { coreStoreFactory } from 'kolibri/store'; -import { AllCategories, NoCategories } from 'kolibri/constants'; -import useUser, { useUserMock } from 'kolibri/composables/useUser'; // eslint-disable-line -import useSearch from '../useSearch'; -import coreModule from '../../../../../../core/assets/src/state/modules/core'; - -Vue.use(VueRouter); - -jest.mock('kolibri/composables/useUser'); - -const name = 'not important'; - -function prep(query = {}, descendant = null) { - const store = coreStoreFactory({ - state: () => ({ - route: { - query, - name, - }, - }), - mutations: { - SET_QUERY(state, query) { - state.route.query = query; - }, - }, - }); - store.registerModule('core', coreModule); - const router = new VueRouter(); - router.push = jest.fn().mockReturnValue(Promise.resolve()); - return { - ...useSearch(descendant, store, router), - router, - store, - }; -} - -describe(`useSearch`, () => { - beforeEach(() => { - ContentNodeResource.fetchCollection = jest.fn(); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - useUser.mockImplementation(() => useUserMock()); - }); - describe(`searchTerms computed ref`, () => { - it(`returns an object with all relevant keys when query params are empty`, () => { - const { searchTerms } = prep(); - expect(get(searchTerms)).toEqual({ - accessibility_labels: {}, - categories: {}, - grade_levels: {}, - languages: {}, - learner_needs: {}, - learning_activities: {}, - keywords: '', - }); - }); - it(`returns an object with all relevant keys when query params have other keys`, () => { - const { searchTerms } = prep({ - search: { - this: true, - }, - keyword: 'how about this?', - }); - expect(get(searchTerms)).toEqual({ - accessibility_labels: {}, - categories: {}, - grade_levels: {}, - languages: {}, - learner_needs: {}, - learning_activities: {}, - keywords: '', - }); - }); - it(`returns an object with all relevant keys when query params are specified`, () => { - const { searchTerms } = prep({ - accessibility_labels: 'test1,test2', - keywords: 'I love paris in the springtime!', - categories: 'notatest,reallynotatest,absolutelynotatest', - grade_levels: 'lowerprimary,uppersecondary,adult', - languages: 'ar-jk,en-pr,en-gb', - learner_needs: 'internet,pencil,rolodex', - learning_activities: 'watch', - }); - expect(get(searchTerms)).toEqual({ - accessibility_labels: { - test1: true, - test2: true, - }, - categories: { - notatest: true, - reallynotatest: true, - absolutelynotatest: true, - }, - grade_levels: { - lowerprimary: true, - uppersecondary: true, - adult: true, - }, - languages: { - 'ar-jk': true, - 'en-pr': true, - 'en-gb': true, - }, - learner_needs: { - internet: true, - pencil: true, - rolodex: true, - }, - learning_activities: { - watch: true, - }, - keywords: 'I love paris in the springtime!', - }); - }); - it(`setting relevant keys will result in a router push`, () => { - const { searchTerms, router } = prep(); - set(searchTerms, { - keywords: 'test', - categories: { - cat1: true, - cat2: true, - }, - }); - expect(router.push).toHaveBeenCalledWith({ - name, - query: { - keywords: 'test', - categories: 'cat1,cat2', - }, - }); - }); - it(`removing keys will be propagated to the router`, () => { - const { searchTerms, router } = prep({ - keywords: 'test', - categories: 'cat1,cat2', - grade_levels: 'level1', - }); - set(searchTerms, { - keywords: '', - categories: { - cat2: true, - }, - }); - expect(router.push).toHaveBeenCalledWith({ - name, - query: { - categories: 'cat2', - }, - }); - }); - it(`setting keywords to null will be propagated to the router`, () => { - const { searchTerms, router } = prep({ - keywords: 'test', - categories: 'cat1,cat2', - grade_levels: 'level1', - }); - set(searchTerms, { - keywords: null, - categories: { - cat2: true, - }, - }); - expect(router.push).toHaveBeenCalledWith({ - name, - query: { - categories: 'cat2', - }, - }); - }); - }); - describe('displayingSearchResults computed property', () => { - const searchKeys = [ - 'learning_activities', - 'categories', - 'learner_needs', - 'accessibility_labels', - 'languages', - 'grade_levels', - ]; - it.each(searchKeys)('should be true when there are any values for %s', key => { - const { displayingSearchResults } = prep({ - [key]: 'test1,test2', - }); - expect(get(displayingSearchResults)).toBe(true); - }); - it('should be true when there is a value for keywords', () => { - const { displayingSearchResults } = prep({ - keywords: 'testing testing one two three', - }); - expect(get(displayingSearchResults)).toBe(true); - }); - }); - describe('search method', () => { - it('should call ContentNodeResource.fetchCollection when searchTerms changes', async () => { - const { store } = prep(); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - store.commit('SET_QUERY', { categories: 'test1,test2' }); - await nextTick(); - expect(ContentNodeResource.fetchCollection).toHaveBeenCalledWith({ - getParams: { - categories: ['test1', 'test2'], - max_results: 25, - include_coach_content: false, - }, - }); - }); - it('should not call ContentNodeResource.fetchCollection if there is no search', () => { - const { search } = prep(); - ContentNodeResource.fetchCollection.mockClear(); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - search(); - expect(ContentNodeResource.fetchCollection).not.toHaveBeenCalled(); - }); - it('should clear labels and more if there is no search', () => { - const { search, labels, more } = prep(); - set(labels, ['test']); - set(more, { test: 'test' }); - search(); - expect(get(labels)).toBeNull(); - expect(get(more)).toBeNull(); - }); - it('should call ContentNodeResource.fetchCollection if there is no search but a descendant is set', () => { - const { search } = prep({}, ref({ tree_id: 1, lft: 10, rght: 20 })); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - search(); - expect(ContentNodeResource.fetchCollection).toHaveBeenCalledWith({ - getParams: { - tree_id: 1, - lft__gt: 10, - rght__lt: 20, - max_results: 1, - include_coach_content: false, - }, - }); - }); - it('should set labels and clear more if there is no search but a descendant is set', async () => { - const { labels, more, search } = prep({}, ref({ tree_id: 1, lft: 10, rght: 20 })); - const labelsSet = { - available: ['labels'], - languages: [], - }; - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({ labels: labelsSet })); - set(more, { test: 'test' }); - search(); - await nextTick(); - expect(get(more)).toBeNull(); - expect(get(labels)).toEqual(labelsSet); - }); - it('should call ContentNodeResource.fetchCollection when searchTerms exist', () => { - const { search } = prep({ categories: 'test1,test2' }); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - search(); - expect(ContentNodeResource.fetchCollection).toHaveBeenCalledWith({ - getParams: { - categories: ['test1', 'test2'], - max_results: 25, - include_coach_content: false, - }, - }); - }); - it('should ignore other categories when AllCategories is set and search for isnull false', () => { - const { search } = prep({ categories: `test1,test2,${NoCategories},${AllCategories}` }); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - search(); - expect(ContentNodeResource.fetchCollection).toHaveBeenCalledWith({ - getParams: { categories__isnull: false, max_results: 25, include_coach_content: false }, - }); - }); - it('should ignore other categories when NoCategories is set and search for isnull true', () => { - const { search } = prep({ categories: `test1,test2,${NoCategories}` }); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - search(); - expect(ContentNodeResource.fetchCollection).toHaveBeenCalledWith({ - getParams: { categories__isnull: true, max_results: 25, include_coach_content: false }, - }); - }); - it('should set keywords when defined', () => { - const { search } = prep({ keywords: `this is just a test` }); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - search(); - expect(ContentNodeResource.fetchCollection).toHaveBeenCalledWith({ - getParams: { - keywords: `this is just a test`, - max_results: 25, - include_coach_content: false, - }, - }); - }); - it('should set results, labels, and more with returned data', async () => { - const { labels, more, results, search } = prep({ categories: 'test1,test2' }); - const expectedLabels = { - available: ['labels'], - languages: [], - }; - const expectedMore = { - cursor: 'adalskdjsadlkjsadlkjsalkd', - }; - const expectedResults = [{ id: 'node-id1' }]; - ContentNodeResource.fetchCollection.mockReturnValue( - Promise.resolve({ - labels: expectedLabels, - results: expectedResults, - more: expectedMore, - }), - ); - search(); - await nextTick(); - expect(get(labels)).toEqual(expectedLabels); - expect(get(results)).toEqual(expectedResults); - expect(get(more)).toEqual(expectedMore); - }); - }); - describe('searchMore method', () => { - it('should not call anything when not displaying search terms', () => { - const { searchMore } = prep(); - ContentNodeResource.fetchCollection.mockClear(); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - searchMore(); - expect(ContentNodeResource.fetchCollection).not.toHaveBeenCalled(); - }); - it('should not call anything when more is null', () => { - const { more, searchMore } = prep({ categories: 'test1' }); - ContentNodeResource.fetchCollection.mockClear(); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - set(more, null); - searchMore(); - expect(ContentNodeResource.fetchCollection).not.toHaveBeenCalled(); - }); - it('should not call anything when moreLoading is true', () => { - const { more, moreLoading, searchMore } = prep({ categories: 'test1' }); - ContentNodeResource.fetchCollection.mockClear(); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - set(more, {}); - set(moreLoading, true); - searchMore(); - expect(ContentNodeResource.fetchCollection).not.toHaveBeenCalled(); - }); - it('should pass the more object directly to getParams', () => { - const { more, searchMore } = prep({ categories: `test1,test2,${NoCategories}` }); - ContentNodeResource.fetchCollection.mockReturnValue(Promise.resolve({})); - const moreExpected = { test: 'this', not: 'that' }; - set(more, moreExpected); - searchMore(); - expect(ContentNodeResource.fetchCollection).toHaveBeenCalledWith({ getParams: moreExpected }); - }); - it('should set results, more and labels', async () => { - const { labels, more, results, searchMore, search } = prep({ - categories: `test1,test2,${NoCategories}`, - }); - const expectedLabels = { - available: ['labels'], - languages: [], - }; - const expectedMore = { - cursor: 'adalskdjsadlkjsadlkjsalkd', - }; - const originalResults = [{ id: 'originalId', content_id: 'first' }]; - ContentNodeResource.fetchCollection.mockReturnValue( - Promise.resolve({ - labels: expectedLabels, - results: originalResults, - more: expectedMore, - }), - ); - search(); - await nextTick(); - const expectedResults = [{ id: 'node-id1', content_id: 'second' }]; - ContentNodeResource.fetchCollection.mockReturnValue( - Promise.resolve({ - labels: expectedLabels, - results: expectedResults, - more: expectedMore, - }), - ); - set(more, {}); - searchMore(); - await nextTick(); - expect(get(labels)).toEqual(expectedLabels); - expect(get(results)).toEqual(originalResults.concat(expectedResults)); - expect(get(more)).toEqual(expectedMore); - }); - }); - describe('removeFilterTag method', () => { - it('should remove a filter from the searchTerms', () => { - const { removeFilterTag, router } = prep({ - categories: 'test1,test2', - }); - removeFilterTag({ value: 'test1', key: 'categories' }); - expect(router.push).toHaveBeenCalledWith({ - name, - query: { - categories: 'test2', - }, - }); - }); - it('should remove keywords from the searchTerms', () => { - const { removeFilterTag, router } = prep({ - keywords: 'test', - }); - removeFilterTag({ value: 'test', key: 'keywords' }); - expect(router.push).toHaveBeenCalledWith({ - name, - query: {}, - }); - }); - it('should not remove any other filters', () => { - const { removeFilterTag, router } = prep({ - categories: 'test1,test2', - learning_activities: 'watch', - }); - removeFilterTag({ value: 'test1', key: 'categories' }); - expect(router.push).toHaveBeenCalledWith({ - name, - query: { - categories: 'test2', - learning_activities: 'watch', - }, - }); - }); - }); - describe('clearSearch method', () => { - it('should remove all filters from the searchTerms', () => { - const { clearSearch, router } = prep({ - categories: 'test1,test2', - learning_activities: 'watch', - keywords: 'this', - }); - clearSearch(); - expect(router.push).toHaveBeenCalledWith({ - name, - query: {}, - }); - }); - }); -}); diff --git a/kolibri/plugins/learn/assets/src/composables/useContentLink.js b/kolibri/plugins/learn/assets/src/composables/useContentLink.js index d5bce334591..00447fa62f6 100644 --- a/kolibri/plugins/learn/assets/src/composables/useContentLink.js +++ b/kolibri/plugins/learn/assets/src/composables/useContentLink.js @@ -2,6 +2,7 @@ import { get } from '@vueuse/core'; import isEmpty from 'lodash/isEmpty'; import pick from 'lodash/pick'; import { computed, getCurrentInstance } from 'vue'; +import { primaryLanguageKey } from 'kolibri-common/composables/useBaseSearch'; import { ExternalPagePaths, PageNames } from '../constants'; function _decodeBackLinkQuery(query) { @@ -17,6 +18,10 @@ export default function useContentLink(store) { function _makeNodeLink(id, isResource, query, deviceId) { const params = get(route).params; + const oldQuery = get(route).query || {}; + if (!isResource && oldQuery[primaryLanguageKey]) { + query[primaryLanguageKey] = oldQuery[primaryLanguageKey]; + } return { name: isResource ? PageNames.TOPICS_CONTENT : PageNames.TOPICS_TOPIC, params: pick({ id, deviceId: deviceId || params.deviceId }, ['id', 'deviceId']), diff --git a/kolibri/plugins/learn/assets/src/views/LibraryPage/index.vue b/kolibri/plugins/learn/assets/src/views/LibraryPage/index.vue index b0499ab0f87..4a9e997c1b8 100644 --- a/kolibri/plugins/learn/assets/src/views/LibraryPage/index.vue +++ b/kolibri/plugins/learn/assets/src/views/LibraryPage/index.vue @@ -49,9 +49,16 @@ v-else-if="!displayingSearchResults && !rootNodesLoading" data-test="channels" > -

- {{ channelsLabel }} -

+ + +

+ {{ channelsLabel }} +

+
+ + + +

@@ -122,6 +130,7 @@ v-model="searchTerms" data-test="side-panel" :width="`${sidePanelWidth}px`" + :showLanguages="displayingSearchResults" /> @@ -176,20 +185,22 @@ diff --git a/packages/kolibri-common/components/SearchFiltersPanel/SelectGroup.vue b/packages/kolibri-common/components/SearchFiltersPanel/SelectGroup.vue index cf26123bff5..0a94c833d44 100644 --- a/packages/kolibri-common/components/SearchFiltersPanel/SelectGroup.vue +++ b/packages/kolibri-common/components/SearchFiltersPanel/SelectGroup.vue @@ -1,16 +1,9 @@