-
Notifications
You must be signed in to change notification settings - Fork 89
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: shareable URLs for library components and searches [FC-0076] #1575
feat: shareable URLs for library components and searches [FC-0076] #1575
Conversation
to support optional parameters in test paths
Thanks for the pull request, @pomegranited! What's next?Please work through the following steps to get your changes ready for engineering review: 🔘 Get product approvalIf you haven't already, check this list to see if your contribution needs to go through the product review process.
🔘 Provide contextTo help your reviewers and other members of the community understand the purpose and larger context of your changes, feel free to add as much of the following information to the PR description as you can:
🔘 Get a green buildIf one or more checks are failing, continue working on your changes until this is no longer the case and your build turns green. 🔘 Let us know that your PR is ready for review:Who will review my changes?This repository is currently maintained by Where can I find more information?If you'd like to get more details on all aspects of the review process for open source pull requests (OSPRs), check out the following resources:
When can I expect my changes to be merged?Our goal is to get community contributions seen and reviewed as efficiently as possible. However, the amount of time that it takes to review and merge a PR can vary significantly based on factors such as:
💡 As a result it may take up to several weeks or months to complete a review and merge your PR. |
8fdbe9b
to
aa25827
Compare
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## master #1575 +/- ##
==========================================
+ Coverage 92.97% 92.98% +0.01%
==========================================
Files 1075 1077 +2
Lines 21205 21353 +148
Branches 4562 4542 -20
==========================================
+ Hits 19715 19855 +140
- Misses 1424 1432 +8
Partials 66 66 ☔ View full report in Codecov by Sentry. |
5fad981
to
18b4e2d
Compare
* Restructures LibraryLayout so that LibraryContext can useParams() to initialize its componentId/collectionId instead of having to parse complicated route strings. Initialization-from-URL can be disabled for the content pickers by passing skipUrlUpdate to the LibraryContext -- which is needed by the component picker. * Clicking/selecting a ComponentCard/CollectionCard navigates to an appropriate component/collection route given the current page. * Adds useLibraryRoutes() hook so components can easily navigate to the best available route without having to know the route strings or maintain search params. * Moves ContentType declaration to the new routes.ts to avoid circular imports. * Renames openInfoSidebar to openLibrarySidebar, so that openInfoSidebar can be used to open the best sidebar for a given library/component/collection.
18b4e2d
to
ce4c27e
Compare
This triggers a navigate() call when the searchbox changes, which resets the query counts in testing. So had to reduce the expected counts by 1.
d202a8d
to
b1fe66e
Compare
Related changes: * adds "manage-team" sidebar action to trigger LibraryInfo's "Manage Team" modal * moves useStateWithUrlSearchParam up to hooks.ts so it can be used by search-manager and library-authoring * splits sidebarCurrentTab and additionalAction out of SidebarComponentInfo -- they're now managed independently as url parameters. This also simplifies setSidebarComponentInfo: we can simply set the new id + key, and so there's no need to merge the ...prev state and new state. * shortens some sidebar property names: sidebarCurrentTab => sidebarTab sidebarAdditionalAction => sidebarAction * test: Tab changes now trigger a navigate() call, which invalidates the within(sidebar) element in the tests. So using screen instead.
b1fe66e
to
f6b46c4
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ooops, queued up these comments when I took an early look at this PR and forgot to submit. Not sure if they're still useful.
<Routes> | ||
<Route | ||
path={ROUTES.COMPONENTS} | ||
element={context(<LibraryAuthoringPage />)} | ||
/> | ||
<Route | ||
path={ROUTES.COLLECTIONS} | ||
element={context(<LibraryAuthoringPage />)} | ||
/> | ||
<Route | ||
path={ROUTES.COMPONENT} | ||
element={context(<LibraryAuthoringPage />)} | ||
/> | ||
<Route | ||
path={ROUTES.COLLECTION} | ||
element={context(<LibraryCollectionPage />)} | ||
/> | ||
<Route | ||
path={ROUTES.HOME} | ||
element={context(<LibraryAuthoringPage />)} | ||
/> | ||
</Routes> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this entire <Routes>...</Routes>
worth having instead of just putting <LibraryAuthoringPage />
? I guess it will show a 404 error if none of the routes match? Or does it?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The advantage of spelling out all of these routes is that it's now very easy to get the :componentId
/ :collectionId
from useParams()
and use these as the defaults in LibraryProvider
.
I also had to wrap the page element
in that context()
wrapper so that the routes would be available when LibraryProvider
is created -- previously <LibraryLayout>
wrapped <LibraryProvider>
around the <Routes>
, and so <LibraryProvider>
couldn't use them.
fa8783d
to
f6bf086
Compare
Related features: 1. We can filter on more than one tag, so the search field needs to store an Array of values which still need to be sanitised. This change adds useListHelpers to assist with the parsing and validating of an Array of Typed values. 2. SearchManager takes a skipUrlUpdate property which should disable using search params for state variables. Our "sort" parameters respected this, but none of the other search parameters did. So this change also adds a wrapper hook called useStateOrUrlSearchParam that handles this switch cleanly. This feature also revealed two bugs fixed in useStateWithUrlSearchParam: 1. When the returnSetter is called with a function instead of a simple value, we need to pass in previous returnValue to the function so it can generate the new value. 2. When the returnSetter is called multiple times by a single callback (like with clearFilters), the latest changes to the UrlSearchParams weren't showing up. To fix this, we had to use the location.search string as the "latest" previous url search, not the prevParams passed into setSearchParams, because these params may not have the latest updates.
f6bf086
to
37370d2
Compare
Combine blockTypesFilter + problemTypesFilter into a single state variable called typesFilter, and store selected block + problem types in the url search parameters. These two states are heavily interdependent, and so the FilterByBlockType code was quite complicated. Then storing these states as url search params caused re-renders that disrupted the delicate dependencies between these two states. This change stores both these type lists using a single TypesFilterData class to simplify the code and avoid bugs when updating their states.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@pomegranited Everything works nicely! Great work!.
Just found a small bug: On collections page, select any collection so that its sidebar appears (this appends st=manage
to query params), now click on Library Info
button and then on Manage Access
button. This opens up Library Team
modal and adds sa=manage-team
query param to url. On refreshing page, modal is not displayed, most probably due to presence of st=manage
query param in url. We should remove it on clicking Library info
button or display of library sidebar.
Using the Component card menu, select Add to Collection. Note that ?sa=jump-to-add-collections is in the query string, and this is preserved when you navigate between components.
Should we also add this parameter when author clicks on Add to Collection
button in the component sidebar?
c59f109
to
9d3e049
Compare
clearing out the selected component/collection.
when editing component collections.
on library authoring page.
using typescript function overloading. Multiple values in the search params are provided by UrlSearchParams.getAll(). Also: * removes the useListHelpers hook added in previous commits -- it is no longer needed. * moves the useStateOrUrlSearchParam hook and TypesFilterData class to a separate search-manager/hooks.ts * sanitizes tag search parameters using oel RESERVED_TAG_CHARS * uses getBlockType to sanitize usage key search parameters
Re-implements the functionality added by openedx#1576 which was broken when storing types in search params. Had to use a different approach to avoid "insecure operation" errors from React. Also adds tests.
need to wrap the ManageCollections element in SidebarProvider.
Since search filters are now stored in the URL, they can leak between tests.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@pomegranited This looks and works great. Thank you for taking this up. 💯
- I tested this: (Played around with different pages and options and verified change in URL.)
- I read through the code
- I checked for accessibility issues
- Includes documentation
static #sep1 = ','; // separates the individual types | ||
|
||
static #sep2 = '|'; // separates the block types from the problem types |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@pomegranited NIT: we can replace these separators with browser's ability to parse multiple query params with same name. But I don't wish to block this as we can update it in a separate PR (if we want to) in the future. Currently, I cannot find any real problems with this approach and it works perfectly.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @navinkarkera -- I'd love to do this, but I'm not sure how, since there's two different "types" that we're tracking here -- block and problem types -- I can't just use a simple ?type=html&type=checkboxes&...
solution. I had to merge them into a single object because they're so entwined (see FilterByBlockType
), and so when I then tried to manage their state with URLSearchParams, I got all kinds of terrible react errors.
I'm open to ideas though -- I just can't see a clear way to doing this.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@pomegranited I was thinking about using a different param name for problem types for example:
?type=html&type=checkboxes&problemType=multiplechoiceresponse&problemType=optionresponse
But if it is causing issues with react and requires a lot of time to implement, we can skip it for now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It won't take much time to split these params -- it's just a reversion back to where they were before. And it's possible the stuff I did to workaround the URLSearchParams state issue has fixed the react problems I was having when I combined them.
@ChrisChV what do you think?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think Jill's solution is fine for now. I'm not one to risk errors in reverting when they've already been fixed. In any other case, it could be done later. What do you think?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good 👍 Only one nit about the coverage
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@pomegranited I think we should get to 100% test coverage for this file. do you think it's possible?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure thing -- I'll write some tests.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Couldn't get routes.ts coverage to 100% (because of MemoryRouter), but I could do 95%.
static #sep1 = ','; // separates the individual types | ||
|
||
static #sep2 = '|'; // separates the block types from the problem types |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think Jill's solution is fine for now. I'm not one to risk errors in reverting when they've already been fixed. In any other case, it could be done later. What do you think?
Description
This PR adds new routes and URL parameters to use when viewing and performing searches on library components. These changes allow these pages to be bookmarked or shared by copy/pasting the browser's current URL.
No changes were made to the UI.
Use cases covered:
Supporting information
See #1499
Private-ref: FAL-3984
Testing instructions
Code review: Recommend going commit-by-commit, since this is a pretty big change.
This change involved some route additions and changes, and so to manually test them, please refresh your browser page after each step, and ensure you see exactly the same page -- including selected tabs, sidebar, and sidebar tabs. Please also check that using the back/forth buttons navigate smoothly between pages as expected, but that changing the library search parameters doesn't append to the browser history.
/libraries
and create or select an existing library from the list."All Content" should still be the default tab you see.
Note that the component picker still works as expected.
Note that the URL changes to
/library/:libraryId/component/:componentId
, and the component is visible in the sidebar.Note that the URL changes to reflect the new componentId.
Note that the URL changes to
/library/:libraryId/:collectionId
, and the collection is visible in the sidebar.Note that the URL changes to reflect the new collectionId.
Note that the sidebar tab is preserved (where possible) when navigating between components or collections.
Clicking Library Info should reset the sidebar back to the Library.
Note that you remain in the selected tab when switching between selected components/collections.
Note that the URL changes to
/library/:libraryId/collection/:collectionId
, and the collection is visible in the sidebar.Note that the URL changes to
/library/:libraryId/collection/:collectionId/:componentId
, and the component is visible in the sidebar.Note that the sidebar tab is preserved (where possible) when navigating between components.
Clicking Collection Info should reset the sidebar back to the Collection.
Note that your keywords are stored in a
q
query string parameter, and preserved on page refresh.Note that these are store in the query string, and cleared when de-selecting or clicking Clear Filters.
Note that
?sa=manage-team
is the query string, and refreshing the page re-opens the Library Team modal.Note that
?sa=jump-to-add-collections
is in the query string, and this is preserved when you navigate between components.Clicking "Confirm" or "Cancel" removes this parameter and closes the collection manager.
Author Notes & Concerns
TODO
s in the SearchManager code about sanitizing parameters, because sanitising user input is good practice. But I'm not sure how best to sanitise things like search keywords?clearFilters
on multiple search filters that are now using theuseStateWithUrlSearchParam
hook -- more details in this comment. Advice or alternatives welcome!<MemoryRouter>
used withtestUtils
, and none of the other available<Router>
classes would let me set the baselibraryId
path. Advice welcome :)