Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Library Content Block Editor #411

Closed
wants to merge 49 commits into from
Closed
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
49 commits
Select commit Hold shift + click to select a range
27b2c6f
feat: initial editor layout + ui
connorhaugh Oct 17, 2023
1a3bc9b
feat: wip
rayzhou-bit Oct 25, 2023
668c184
feat: wip2
rayzhou-bit Nov 1, 2023
d8a45fd
feat: wip3
rayzhou-bit Nov 6, 2023
3f1887c
feat: polish
rayzhou-bit Nov 8, 2023
d66c7e9
test: add reducers and selectors tests
connorhaugh Nov 8, 2023
3734e01
fix: proptypes
rayzhou-bit Nov 9, 2023
2b263ee
test: add blockselector test
connorhaugh Nov 9, 2023
ffd2e63
test: add three more JS tests
connorhaugh Nov 10, 2023
65eee08
fix: add tests for all but hooks.js
connorhaugh Nov 10, 2023
2674607
fix: initialize
rayzhou-bit Nov 13, 2023
e74c90f
fix: library api
rayzhou-bit Nov 13, 2023
3d8d5b0
feat: api tweaks mostly with index.js
rayzhou-bit Nov 15, 2023
5d2f783
feat: libraryselector.js api update
rayzhou-bit Nov 15, 2023
a9989da
feat: blockselector.js rewrite wip
rayzhou-bit Nov 16, 2023
56c4c39
feat: general changes to api usage
rayzhou-bit Nov 17, 2023
2b92174
feat: table saves candidates
rayzhou-bit Nov 22, 2023
21bd883
feat: load candidate works
rayzhou-bit Nov 28, 2023
ab3e541
feat: v1 library selection
rayzhou-bit Nov 29, 2023
fe65556
fix: loading issue and polish
rayzhou-bit Nov 30, 2023
45b9f40
fix: saving blocks
rayzhou-bit Nov 30, 2023
1339322
feat: sort library dropdown alphabetically
rayzhou-bit Nov 30, 2023
c92c35f
feat: lint and some fixes
rayzhou-bit Dec 1, 2023
48ce342
feat: lint and tests
rayzhou-bit Dec 5, 2023
8fb9644
feat: candidate tuples and more tests
rayzhou-bit Dec 5, 2023
9a56336
feat: more tests
rayzhou-bit Dec 6, 2023
4bdf570
feat: why are tests so hard
rayzhou-bit Dec 6, 2023
535bc14
feat: more tests
rayzhou-bit Dec 7, 2023
2699795
feat: lint
rayzhou-bit Dec 7, 2023
be6cbe8
feat: merge main
rayzhou-bit Dec 7, 2023
bcc0984
feat: lint
rayzhou-bit Dec 7, 2023
be615b6
feat: some more api tests
rayzhou-bit Dec 7, 2023
87fbfa7
feat: selectors test
rayzhou-bit Dec 7, 2023
6265f5a
feat: remove fetchV2LibraryMetadata
rayzhou-bit Dec 19, 2023
3a689a4
feat: v1 library api update
rayzhou-bit Dec 19, 2023
0d51711
feat: fix
rayzhou-bit Dec 19, 2023
b09ed11
feat: lint
rayzhou-bit Dec 19, 2023
32c4fb3
feat: library version should be string
rayzhou-bit Dec 19, 2023
208070a
feat: failure tests
rayzhou-bit Dec 20, 2023
02f6ac3
feat: merge conflict
rayzhou-bit Dec 20, 2023
3ea40d8
feat: LCB children
rayzhou-bit Dec 28, 2023
fd6d530
feat: new fetch children api
rayzhou-bit Jan 3, 2024
1c783d7
feat: test and lint
rayzhou-bit Jan 3, 2024
0bfa336
feat: use usage id and fetch v1 library block
rayzhou-bit Jan 5, 2024
8553ac6
feat: regex fix and v1 library version
rayzhou-bit Jan 5, 2024
9db63b4
feat: lint
rayzhou-bit Jan 6, 2024
0c68a92
feat: v1 api fix and candidate saving
rayzhou-bit Jan 16, 2024
01f6296
feat: test and lint
rayzhou-bit Jan 19, 2024
d94280c
Merge remote-tracking branch 'upstream/main' into feat--library-conte…
kdmccormick Feb 9, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion src/editors/containers/EditorContainer/index.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,7 @@ export const EditorContainer = ({
/>
</div>
</ModalDialog.Header>
<ModalDialog.Body className="pb-6">
<ModalDialog.Body className="pb-6 min-vh-100">
rayzhou-bit marked this conversation as resolved.
Show resolved Hide resolved
{isInitialized && children}
</ModalDialog.Body>
<EditorFooter
Expand Down
133 changes: 133 additions & 0 deletions src/editors/containers/LibraryContentEditor/BlocksSelector.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,133 @@
import React, { useCallback } from 'react';
import { connect } from 'react-redux';
import PropTypes from 'prop-types';
import { FormattedMessage, injectIntl, intlShape } from '@edx/frontend-platform/i18n';
import { CheckboxControl, DataTable } from '@edx/paragon';

import messages from './messages';
import { actions, selectors } from './data';
import { useBlocksHook } from './hooks';
import { modes } from './constants';
import { getCandidates } from './utils';

export const RowCheckbox = ({ row }) => {
const {
indeterminate,
checked,
...toggleRowSelectedProps
} = row.getToggleRowSelectedProps();

return (
<div className='text-center'>
<CheckboxControl
{...toggleRowSelectedProps}
title="Toggle row selected"
checked={checked}
isIndeterminate={false}
/>
</div>
);
};

export const BlocksSelector = ({
initialRows,
mode,
// redux
blocksInSelectedLibrary,
setCandidatesForLibrary,
selectedLibraryId,
}) => {

const {
blocksTableData,
} = useBlocksHook({
blocksInSelectedLibrary,
selectedLibraryId,
});

const columns = [
{
Header: 'Name',
accessor: 'display_name',
},
{
Header: 'Block Type',
accessor: 'block_type',
},
];

const selectColumn = {
id: 'selection',
Header: () => null,
Cell: RowCheckbox,
disableSortBy: true,
};

const onSelectedRowsChanged = useCallback(
(selected) => setCandidatesForLibrary({
libraryId: selectedLibraryId,
candidates: getCandidates({
blocks: blocksInSelectedLibrary,
rows: selected,
}),
}),
[blocksInSelectedLibrary]
);

if (selectedLibraryId === null || mode !== modes.selected.value) {
return <></>;
}

return (
<div className='mb-5 pt-3 border-top'>
<label>
<FormattedMessage {...messages.tableInstructionLabel} />
</label>
<DataTable
key={selectedLibraryId}
columns={columns}
data={blocksTableData}
itemCount={blocksTableData.length}
isSelectable
isPaginated
isSortable
initialState={{ selectedRowIds: initialRows }}
manualSelectColumn={selectColumn}
onSelectedRowsChanged={onSelectedRowsChanged}
>
<DataTable.TableControlBar />
<DataTable.Table />
<DataTable.EmptyTable content="No blocks found." />
<DataTable.TableFooter />
</DataTable>
</div>
);
};

BlocksSelector.defaultProps = {
blocksInSelectedLibrary: [],
initialRows: {},
mode: '',
selectedLibraryId: null,
};

BlocksSelector.propTypes = {
initialRows: PropTypes.shape({}),
mode: PropTypes.string,
// redux
blocksInSelectedLibrary: PropTypes.arrayOf(PropTypes.shape({})),
setCandidatesForLibrary: PropTypes.func.isRequired,
selectedLibraryId: PropTypes.string,
};

export const mapStateToProps = (state) => ({
blocksInSelectedLibrary: selectors.blocksInSelectedLibrary(state),
mode: selectors.mode(state),
selectedLibraryId: selectors.selectedLibraryId(state),
});

export const mapDispatchToProps = {
setCandidatesForLibrary: actions.setCandidatesForLibrary,
};

export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(BlocksSelector));
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
import React from 'react';
import { render, fireEvent } from '@testing-library/react';
import { IntlProvider } from '@edx/frontend-platform/i18n';
import { BlocksSelector } from './BlocksSelector';

jest.unmock('@edx/paragon');
jest.unmock('@edx/paragon/icons');

jest.mock('./hooks', () => ({
useBlocksHook: jest.fn().mockReturnValue({
blockLinks: {
1: 'link1',
2: 'link2',
},
blocksTableData: [
// Mocked blocksTableData
{ id: 1, display_name: 'Block 1', block_type: 'Type A' },
{ id: 2, display_name: 'Block 2', block_type: 'Type B' },
],
tempCandidates: [],
setTempCandidates: jest.fn(),
}),
}));

function renderComponent(props) {
return render(
<IntlProvider locale="en">
<BlocksSelector {...props} />
</IntlProvider>,
);
}

const mockProps = {
candidates: {
},
mode: 'selected',
studioEndpointUrl: 'https://example.com',
blocksInSelectedLibrary: [{}],
setCandidatesForLibrary: jest.fn(),
selectedLibraryId: 'exampleLibraryId',
};

describe('BlocksSelector', () => {
it('renders when selectedLibraryId is not null', () => {
const { queryByText } = renderComponent(mockProps);

// make sure that the relevant columns are there
expect(queryByText('Name')).toBeTruthy();
expect(queryByText('Block Type')).toBeTruthy();
// make sure that the relevant rows are there
});

it('does not render when selectedLibraryId is null', () => {
const { container } = renderComponent({ ...mockProps, selectedLibraryId: null });
expect(container.firstChild).toBeFalsy();
});
it('renders when mode is selected', () => {
const { queryByText } = renderComponent(mockProps);
expect(queryByText('Name')).toBeTruthy();
});
it('does not render when mode is not selected', () => {
const { container } = renderComponent({ ...mockProps, mode: 'soMeThingElse' });
expect(container.firstChild).toBeFalsy();
});
it('renders rows and triggers setCandidatesForLibrary when a row is selected', () => {
const { getAllByTestId } = renderComponent(mockProps);
const rowcheckboxes = getAllByTestId('datatable-select-column-checkbox-cell');
expect(rowcheckboxes.length).toBe(2);
const checkboxElement = rowcheckboxes[0];
expect(checkboxElement.checked).toBe(false);
fireEvent.click(checkboxElement);
expect(checkboxElement.checked).toBe(true);
});
});
89 changes: 89 additions & 0 deletions src/editors/containers/LibraryContentEditor/LibrarySelector.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
import React from 'react';
import { connect } from 'react-redux';
import PropTypes from 'prop-types';
import { Dropdown } from '@edx/paragon';
import { FormattedMessage, injectIntl, intlShape } from '@edx/frontend-platform/i18n';

import messages from './messages';
import { selectors } from './data';
import { useLibrarySelectorHook } from './hooks';
import { getLibraryName } from './utils';

export const LibrarySelector = ({
// redux
libraries,
selectedLibraryId,
settings,
// injected
intl,
}) => {
const {
onLibrarySelect,
} = useLibrarySelectorHook({
libraries,
settings,
});

if (Object.keys(libraries).length === 0) {
return (
<div className='mb-3'>
<span>
<FormattedMessage {...messages.noLibraryMessage} />
</span>
</div>
);
}

return (
<div className='mb-3'>
<Dropdown className='w-100'>
<Dropdown.Toggle
className='w-100'
id='library-selector'
variant='outline-primary'
>
{selectedLibraryId
? getLibraryName(libraries[selectedLibraryId])
: intl.formatMessage(messages.librarySelectorDropdownDefault)}
</Dropdown.Toggle>
<Dropdown.Menu>
<Dropdown.Item key={-1} onClick={() => onLibrarySelect(null)}>
<FormattedMessage {...messages.librarySelectorDropdownDefault} />
</Dropdown.Item>
{Object.entries(libraries).map(([id, library], index) => (
<Dropdown.Item key={index} onClick={() => onLibrarySelect(id)}>
{getLibraryName(library)}
</Dropdown.Item>
))}
</Dropdown.Menu>
</Dropdown>
</div>
);
};

LibrarySelector.defaultProps = {
libraries: [],
selectedLibraryId: null,
settings: {},
};

LibrarySelector.propTypes = {
// redux
libraries: PropTypes.array,
selectedLibraryId: PropTypes.string,
settings: PropTypes.object,
// injected
intl: intlShape.isRequired,
};

export const mapStateToProps = (state) => ({
libraries: selectors.libraries(state),
selectedLibraryId: selectors.selectedLibraryId(state),
settings: selectors.settings(state),
});

export const mapDispatchToProps = {};

export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(LibrarySelector));


Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
import React from 'react';
import { render, fireEvent } from '@testing-library/react';
import { IntlProvider } from '@edx/frontend-platform/i18n';
import { LibrarySelector } from './LibrarySelector';

jest.mock('./data/api', () => ({
fetchLibraryContent: jest.fn().mockReturnValue({
blocks: 'SoMe BLOcKs',
}),
fetchLibraryProperty: jest.fn().mockReturnValue({
version: 'lIkE a VeRsiOn',
}),
}));

function renderComponent(props) {
return render(
<IntlProvider locale="en">
<LibrarySelector {...props} />
</IntlProvider>,
);
}

describe('LibrarySelector',()=>{
const mocklibraries = [
{
display_name: 'lIb1',
library_key: 'LiB KEy 1',
},
{
display_name: 'LIb2',
library_key: 'LIb KEy 2',
},
{
display_name: 'liB3',
library_key: 'lIb keY 3',
}
]
const props = {
studioEndpointUrl: 'eXaMplE.com',
libraries: mocklibraries,
settings: {
[mocklibraries[0].library_key]: {
value: 'SoMethIng'
}
},
unloadLibrary: jest.fn(),
}
it('Renders as expected with default props',()=>{
const {container, queryByText, queryByTestId} = renderComponent(props);

expect(queryByTestId('dropdown')).toBeTruthy();
// It renders the selected library as the title.
expect(queryByText(mocklibraries[0].display_name)).toBeTruthy();
// The other members of the library are not rendered.
expect(queryByText(mocklibraries[1].display_name)).toBeFalsy();
expect(queryByText(mocklibraries[2].display_name)).toBeFalsy();

//Clicking on the dropdown displays the options
fireEvent.click(container.querySelector("#library-selector"));
expect(queryByText(mocklibraries[1].display_name)).toBeTruthy();
expect(queryByText(mocklibraries[2].display_name)).toBeTruthy();
expect(queryByText('FormattedMessage')).toBeTruthy();
});
it('Does not render dropdown when there are no libraries',()=>{
const {container, queryByTestId} = renderComponent({
...props,
libraries: null,
});
expect(queryByTestId('dropdown')).toBeFalsy();
});
});
Loading
Loading