2024-03-11 17:02:15 +02:00
|
|
|
import * as React from 'react';
|
2024-05-02 18:05:25 +02:00
|
|
|
import { mockMobilePlatform, setupDatabaseAndSynchronizer, switchClient } from '@joplin/lib/testing/test-utils';
|
2024-03-11 17:02:15 +02:00
|
|
|
|
|
|
|
import { render, screen, userEvent, waitFor } from '@testing-library/react-native';
|
|
|
|
import '@testing-library/react-native/extend-expect';
|
|
|
|
|
|
|
|
import pluginServiceSetup from './testUtils/pluginServiceSetup';
|
2024-05-02 18:05:25 +02:00
|
|
|
import createMockReduxStore from '../../../../utils/testing/createMockReduxStore';
|
2024-06-04 10:57:52 +02:00
|
|
|
import WrappedPluginStates from './testUtils/WrappedPluginStates';
|
|
|
|
import { AppState } from '../../../../utils/types';
|
|
|
|
import { Store } from 'redux';
|
|
|
|
import mockRepositoryApiConstructor from './testUtils/mockRepositoryApiConstructor';
|
|
|
|
import { resetRepoApi } from './utils/useRepoApi';
|
2024-03-11 17:02:15 +02:00
|
|
|
|
2024-03-29 14:40:54 +02:00
|
|
|
const expectSearchResultCountToBe = async (count: number) => {
|
|
|
|
await waitFor(() => {
|
|
|
|
expect(screen.queryAllByTestId('plugin-card')).toHaveLength(count);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2024-06-04 10:57:52 +02:00
|
|
|
// The search box is initially read-only -- waits for it to be editable.
|
|
|
|
const getEditableSearchBox = async () => {
|
2024-06-15 11:00:21 +02:00
|
|
|
const searchBox = await screen.findByPlaceholderText('Search for plugins...');
|
2024-06-04 10:57:52 +02:00
|
|
|
expect(searchBox).toBeVisible();
|
|
|
|
|
|
|
|
await waitFor(() => {
|
|
|
|
expect(searchBox.props.editable).toBe(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
return searchBox;
|
|
|
|
};
|
|
|
|
|
|
|
|
let reduxStore: Store<AppState>;
|
|
|
|
|
|
|
|
describe('PluginStates.search', () => {
|
2024-03-11 17:02:15 +02:00
|
|
|
beforeEach(async () => {
|
|
|
|
await setupDatabaseAndSynchronizer(0);
|
|
|
|
await switchClient(0);
|
2024-06-04 10:57:52 +02:00
|
|
|
reduxStore = createMockReduxStore();
|
|
|
|
pluginServiceSetup(reduxStore);
|
|
|
|
mockMobilePlatform('android');
|
|
|
|
resetRepoApi();
|
|
|
|
|
|
|
|
await mockRepositoryApiConstructor();
|
2024-03-11 17:02:15 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should find results', async () => {
|
2024-06-04 10:57:52 +02:00
|
|
|
const wrapper = render(<WrappedPluginStates initialPluginSettings={{}} store={reduxStore}/>);
|
2024-03-11 17:02:15 +02:00
|
|
|
|
|
|
|
const user = userEvent.setup();
|
2024-06-04 10:57:52 +02:00
|
|
|
const searchBox = await getEditableSearchBox();
|
2024-03-11 17:02:15 +02:00
|
|
|
await user.type(searchBox, 'backlinks');
|
|
|
|
|
|
|
|
// Should find one result
|
2024-03-29 14:40:54 +02:00
|
|
|
await expectSearchResultCountToBe(1);
|
2024-03-11 17:02:15 +02:00
|
|
|
|
|
|
|
// Clearing the search input should hide all results
|
|
|
|
await user.clear(searchBox);
|
2024-03-29 14:40:54 +02:00
|
|
|
await expectSearchResultCountToBe(0);
|
|
|
|
|
|
|
|
// Typing a space should show all results
|
|
|
|
await user.type(searchBox, ' ');
|
2024-03-11 17:02:15 +02:00
|
|
|
await waitFor(() => {
|
2024-03-29 14:40:54 +02:00
|
|
|
expect(screen.queryAllByTestId('plugin-card').length).toBeGreaterThan(2);
|
2024-03-11 17:02:15 +02:00
|
|
|
});
|
2024-06-04 10:57:52 +02:00
|
|
|
|
|
|
|
wrapper.unmount();
|
2024-03-11 17:02:15 +02:00
|
|
|
});
|
2024-03-29 14:40:54 +02:00
|
|
|
|
|
|
|
it('should only show recommended plugin search results on iOS-like environments', async () => {
|
|
|
|
// iOS uses restricted install mode
|
2024-06-04 10:57:52 +02:00
|
|
|
mockMobilePlatform('ios');
|
|
|
|
await mockRepositoryApiConstructor();
|
2024-03-29 14:40:54 +02:00
|
|
|
|
2024-06-04 10:57:52 +02:00
|
|
|
const wrapper = render(<WrappedPluginStates initialPluginSettings={{}} store={reduxStore}/>);
|
2024-03-29 14:40:54 +02:00
|
|
|
|
|
|
|
const user = userEvent.setup();
|
2024-06-04 10:57:52 +02:00
|
|
|
const searchBox = await getEditableSearchBox();
|
|
|
|
|
|
|
|
await user.press(searchBox);
|
2024-03-29 14:40:54 +02:00
|
|
|
await user.type(searchBox, 'abc');
|
|
|
|
|
2024-06-04 10:57:52 +02:00
|
|
|
expect(searchBox.props.value).toBe('abc');
|
|
|
|
|
2024-03-29 14:40:54 +02:00
|
|
|
// Should find recommended plugins
|
|
|
|
await expectSearchResultCountToBe(1);
|
|
|
|
|
|
|
|
// Should not find non-recommended plugins
|
|
|
|
await user.clear(searchBox);
|
|
|
|
await user.type(searchBox, 'backlinks');
|
|
|
|
await expectSearchResultCountToBe(0);
|
|
|
|
|
|
|
|
await user.clear(searchBox);
|
|
|
|
await user.type(searchBox, ' ');
|
|
|
|
await expectSearchResultCountToBe(1);
|
|
|
|
expect(screen.getByText(/ABC Sheet Music/i)).toBeTruthy();
|
|
|
|
expect(screen.queryByText(/backlink/i)).toBeNull();
|
2024-06-04 10:57:52 +02:00
|
|
|
|
|
|
|
wrapper.unmount();
|
2024-03-29 14:40:54 +02:00
|
|
|
});
|
2024-04-03 19:51:09 +02:00
|
|
|
|
|
|
|
it('should mark incompatible plugins as incompatible', async () => {
|
2024-06-04 10:57:52 +02:00
|
|
|
const wrapper = render(<WrappedPluginStates initialPluginSettings={{}} store={reduxStore}/>);
|
2024-04-03 19:51:09 +02:00
|
|
|
|
|
|
|
const user = userEvent.setup();
|
2024-06-04 10:57:52 +02:00
|
|
|
const searchBox = await getEditableSearchBox();
|
|
|
|
await user.press(searchBox);
|
2024-04-03 19:51:09 +02:00
|
|
|
await user.type(searchBox, 'abc');
|
2024-06-04 10:57:52 +02:00
|
|
|
expect(searchBox.props.value).toBe('abc');
|
2024-04-03 19:51:09 +02:00
|
|
|
|
|
|
|
await expectSearchResultCountToBe(1);
|
|
|
|
expect(screen.queryByText('Incompatible')).toBeNull();
|
|
|
|
|
|
|
|
await user.clear(searchBox);
|
|
|
|
await user.type(searchBox, 'side bar toggle');
|
|
|
|
await expectSearchResultCountToBe(1);
|
|
|
|
expect(await screen.findByText(/Note list and side bar/i)).toBeVisible();
|
|
|
|
expect(await screen.findByText('Incompatible')).toBeVisible();
|
|
|
|
|
2024-06-04 10:57:52 +02:00
|
|
|
wrapper.unmount();
|
2024-04-03 19:51:09 +02:00
|
|
|
});
|
2024-03-11 17:02:15 +02:00
|
|
|
});
|