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

Reader / Player #1596

Open
wants to merge 60 commits into
base: develop
Choose a base branch
from
Open

Reader / Player #1596

wants to merge 60 commits into from

Conversation

kasperbirch1
Copy link
Contributor

@kasperbirch1 kasperbirch1 commented Dec 13, 2024

The `Reader` component supports both teaser and reserved material.
Use the `identifier` for teaser and `orderId` for reserved.

To load the reader, it is necessary to attach `readerAssets` to the `<head>` dynamically. A function called `appendAsset` has been introduced to manage this process.

Currently, there is a problem where `readerAssets` need to be reloaded each time the reader is initialized. As a temporary workaround, `window.location.reload();` is being used. A better solution will be implemented in the future.
The `Reader` needs to be shown in a full-screen modal.

Note:
It does not work with or need `FocusTrap`.  I have tested a proof-of-concept where it functions as expected. You can navigate pages using the arrow keys and close the modal with the Escape key.
This commit introduces functionality to try e-books.

To support closing the reader via the close/back button, the `closeModal` method was added to the `window` object and invoked using `javascript:window.closeModal('reader-modal-${identifier}')`.
…nal`

Consolidated the "Reader Teaser" logic into `MaterialButtonOnlineExternal` since not all online materials have a teaser.
…ialButtonsFindOnShelf`

These two should follow the same structure to ensure consistency in appearance.
To ensure the `MaterialButtonReaderTeaser` and `MaterialButtonsFindOnShelf` have identical behavior, I created a shared component to eliminate code duplication.
This ensures that the `ReaderModal` is loaded only when necessary, similar to how the `MaterialButtonReaderTeaser` is rendered.
I believe the issue isn't with the `useEffect` inside `Reader`, but rather with the modal logic itself. I have encountered similar issues with the `FindOnShelfModal`, where it doesn't load the second time. A ticket has been created for this issue: https://reload.atlassian.net/browse/DDFHER-140
The addition of `closeModal` to the `window` object is only necessary in the `ReaderModal`, so it is unnecessary to keep it in the generic `Modal` component.
Since loading the reader scripts requires a page shift, I developed the reader as a separate app with its own route in Drupal.
In the new architecture, `Reader` is now a standalone app.

Additionally I also found that there is no reason to wrap the stories in `Store`
After the `Reader` is no longer an app, the `ReaderModal` is no longer in use and has been removed.

A new component, `MaterialSecondaryLink`, has been added. It follows the same structure as `MaterialSecondaryButton` but is designed for links instead.

I considered reverting the `MaterialSecondaryButton` introduced in an earlier commit, but I believe the new structure is more readable. Additionally, it may be useful for the `Player` app/component in the future.

- Remove unnecessary `isFullScreen`  from `Modal`

With `Reader` now functioning as an  app on its own page, this property is no longer required.
This component is redundant as its functionality can be replaced with `MaterialSecondaryLink`.
Implemented the `Player` component, which functions similarly to the `Reader`. Considering moving the `Reader` component to the `components` folder for consistency in folder structure.
This enables the player teaser button for the correct material types: "lydbog (online)", "podcast", "musik (online)", "lydbog (bånd)" that opens in a modal `PlayerModal`.
These components are very similar and distinct from other components, so I have moved them together into the same folder and created a new story category called "ReaderPlayer" for them.
These will, in the future, not only be used for showing teasers.
The function now accepts a `@materialType`, which can also reference translatable strings such as `ebookText` or `audiobookText`.
This is now handled in `.player` CSS.
This caused issues when passing the URL parameter from Drupal into the `Reader` app.
…ublizon`

Sets the foundation for future changes where all Publizon (ereolen) will be handled internally, and as such, it will no longer be part of `MaterialButtonOnlineExternal`.

For the time being, I have added the new `MaterialButtonOnlinePublizon` below `MaterialButtonOnlineExternal`, but the TODO notes that separate logic will be required to manage the `Reader` / `Player` buttons/links.
This is no longer used as the `Reader` is now on its own app (page).
While investigating https://reload.atlassian.net/browse/DDFBRA-236, I identified an infinite loop in `MaterialDetailsModal`. This issue appears when the modal is opened via "user/loans" and potentially all places it is used
Since all data from loans comes from Publizon, we don’t have a `workId`, which is required to link to the material. Therefore, I use the `useComplexSearchWithPaginationQuery` to find the work using the ISBN (identifier) and construct the `workId` into a URL, e.g., `/work/work-of:870970-basis:62986115`.  Additionally, if it finds the material type, it adds that as a parameter.
This commit ensures that users who already have a digital loan (OrderId) at Publizon will see a button labeled either "Read" or "Listen to @materialType."

### Changes Made:
1. Added `orderId` to `LoanType` and included it in the return of `mapPublizonLoanToLoanType`.
2. Introduced a new hook, `useReaderPlayerButtons`, which accepts `selectedManifestations` and returns an `orderId` if the identifier for the `selectedManifestations` exists in the Publizon loans retrieved from `useLoans`.

### Known Issues:
This is a WIP because an issue with `FocusTrap` still needs to be resolved. Currently, `FocusTrap` is temporarily set to `active={false}` to avoid blocking further testing.
This change was required to avoid removing the `FocusTrap` from the modal, as it states: "Your focus-trap must have at least one container with at least one tabbable node in it at all times."

I spent some time attempting to use `createPortal` to manage a single modal in `material.tsx`, but this approach does not work because the modal is only rendered after it is opened.

The solution is to include both modals in `material.tsx` and utilize the existing `useReaderPlayerButtons` hook to determine whether the modal should render. This necessitated a small refactor to handle scenarios where there are no `manifestations`.

Additionally, I made a minor cleanup, but further improvements are still needed.
Eliminate redundant code, prevent duplicate invocations of `getMaterialTypes`, and consolidate the logic into a single function for determining the type. Use the `type` from `useReaderPlayerButtons` to control when player logic is enabled.
Copy link
Contributor

@spaceo spaceo left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have some suggestions

src/apps/loan-list/modal/material-details.tsx Show resolved Hide resolved
src/apps/loan-list/modal/material-details.tsx Show resolved Hide resolved
src/core/utils/useGetWorkUrlFromIdentifier.ts Outdated Show resolved Hide resolved
src/core/utils/useReaderPlayer.tsx Outdated Show resolved Hide resolved
src/core/utils/useReaderPlayer.tsx Outdated Show resolved Hide resolved
src/core/utils/useReaderPlayer.tsx Show resolved Hide resolved
src/core/utils/useReaderPlayer.tsx Outdated Show resolved Hide resolved
@kasperbirch1 kasperbirch1 requested a review from spaceo January 7, 2025 14:18
kasperbirch1 and others added 22 commits January 7, 2025 15:47
…anonymous-and-logged-in-user-can-try-ebook

DDFBRA-183 - Anonymous and logged in user can try ebook
…logged-in-user-can-go-from-digital-loans-to-material-to-listen-read

DDFBRA-236 - Logged in user can go from digital loans to material
…logged-in-user-can-open-their-loaned-e-material

WIP: DDFBRA-251 - User can open their loaned e materials
The `isVisible` function in `src/core/utils/featureFlag.ts` manages feature visibility in development and pull request environments. This ensures that new code can be integrated without being exposed to users on production sites.

During the interim period before the reader/player logic is fully released, some translatable texts and URLs will be hardcoded. This approach simplifies pull requests and avoids unnecessary complexity.
This was partially removed by misunderstanding in [DDFBRA-49](https://reload.atlassian.net/browse/DDFBRA-49).

It has now been reintroduced along with an update to `MaterialAvailabilityTextOnline`, ensuring it checks for all e-materials (Publizon).

Here is the commit where it was removed:
[b28c225](b28c225)
Replaced `Object.fromEntries` with `reduce` to build the `availabilityTextMap`, making the logic more explicit and readable while preserving the original functionality.
Renamed variables to `shouldShowOnlineAvailability` and `shouldShowPhysicalAvailability` to better reflect their purpose.
Added a condition to return default values when ´manifestations` or 'data?.loans` is `null`, and removed redundant null checks for loans after ensuring its presence earlier.
…ntifier`

The updated name provides better clarity, making the function's purpose and usage more apparent to developers.

Note: Filename changes will be addressed in a separate commit for git history purposes.
We don’t use the `warn` approach in the codebase. Therefore, this is an intentional error for development purposes until a better solution is implemented.
…in user loans.

This hides the link to Erolen when the manifestation is already in the user's loans.
…tensibility

- Replaces single function (isVisible) with a featureFlag object
- Introduces a features list (name, active), allowing for future expansion
- Implements an isActive() method to handle feature toggles more cleanly
Add descriptions for `identifier` and `orderId` and make the fields editable.
@kasperbirch1 kasperbirch1 force-pushed the reader-player-feature branch from c2ec163 to d08cbca Compare January 7, 2025 14:53
Copy link
Contributor

@spaceo spaceo left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM 👍

The identifier must always be returned, and `useGetV1UserLoans` is only executed when a user is present. To optimize this, the check for `data?.loans` has been moved.

Additional improvements:
- Added a check for `manifestations.length === 0` to prevent potential errors if an empty array (manifestations) is provided.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants