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

Refactor file viewer #7270

Merged
merged 70 commits into from
Dec 22, 2024
Merged

Conversation

ch-iv
Copy link
Contributor

@ch-iv ch-iv commented Oct 19, 2024

Proposed Changes

This PR refactors the FileViewer component and some of the adjacent components. The changes include:

  • remove request to the /get_file endpoint
  • move text and binary file loading into their respective components
  • move HEIC/HEIF file handling from the FileViewer to the ImageViewer

These changes make the structure of the file viewer more logical and easier to change in the future. This also introduces a small frontend performance improvement by doing one less http request.

Screenshots of your changes (if applicable)
Associated documentation repository pull request (if applicable)

Type of Change

(Write an X or a brief description next to the type or types that best describe your changes.)

Type Applies?
🚨 Breaking change (fix or feature that would cause existing functionality to change)
New feature (non-breaking change that adds functionality)
🐛 Bug fix (non-breaking change that fixes an issue)
🎨 User interface change (change to user interface; provide screenshots)
♻️ Refactoring (internal change to codebase, without changing functionality) x
🚦 Test update (change that only adds or modifies tests)
📦 Dependency update (change that updates a dependency)
🔧 Internal (change that only affects developers or continuous integration)

Checklist

(Complete each of the following items for your pull request. Indicate that you have completed an item by changing the [ ] into a [x] in the raw text, or by clicking on the checkbox in the rendered description on GitHub.)

Before opening your pull request:

  • I have performed a self-review of my changes.
    • Check that all changed files included in this pull request are intentional changes.
    • Check that all changes are relevant to the purpose of this pull request, as described above.
  • I have added tests for my changes, if applicable.
    • This is required for all bug fixes and new features.
  • I have updated the project documentation, if applicable.
    • This is required for new features.
  • If this is my first contribution, I have added myself to the list of contributors.

After opening your pull request:

  • I have updated the project Changelog (this is required for all changes).
  • I have verified that the pre-commit.ci checks have passed.
  • I have verified that the CI tests have passed.
  • I have reviewed the test coverage changes reported by Coveralls.
  • I have requested a review from a project maintainer.

Questions and Comments

(Include any questions or comments you have regarding your changes.)

Copy link
Collaborator

@david-yz-liu david-yz-liu left a comment

Choose a reason for hiding this comment

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

@ch-iv great work! The code in this component is quite thorny, and you've done a good job teasing it apart. I've left a few comments on how you can streamline further (with a bit more aggressive refactoring).

One thing to watch out for as you are doing manual tests of your code is to make sure in the grading view that annotations are correctly rendered as files are switched, when switching between viewers of different types (e.g., pdf -> text) and between different files that use the same viewer.

});
}
});
this.setFileUrl(submission_file_id);
Copy link
Collaborator

Choose a reason for hiding this comment

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

Similar to my comment on your other PR, this function calls setState, which isn't synchronous. As you probably noticed, in the current implementation of setFileUrl, all of the URL-creation code is synchronous except the heic/heif image formats, which currently require a separate request to be made.

You can try doing the following:

  • Move the heic/heif handling into the ImageViewer component, where it more logically belongs
  • Change this.setFileUrl to this.getFileUrl and just (synchronously) return a URL, and then use that to set the initial url in the setState call on line 114
  • Ideally set the type state in that initial call (L114) as well, so that you don't need the if branch code on Lines 122-124
  • See also my comment on the plaintext/binary branch a bit further below

if (response.ok) {
return response.text();
}
const delimiter = this.props.selectedFileURL.includes("?") ? "&" : "?";
Copy link
Collaborator

Choose a reason for hiding this comment

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

This delimiter-handling part is quite awkward; I believe you're running into this because you're trying to unify the handling of multiple different URLs.

Rather than insert this here, please modify how the URLs are originally generated to include these query parameters directly. (I assume this will involve a few different locations, because of the different places this component is used.)

return response.text();
}
const delimiter = this.props.selectedFileURL.includes("?") ? "&" : "?";
fetch(
Copy link
Collaborator

Choose a reason for hiding this comment

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

Okay, so a larger change you can make (which I'd support) is to move the content fetching here into the TextViewer and BinaryViewer components. This will make the different viewer components more uniform in how they fetch data, and help simplify the FileViewer code.

@ch-iv ch-iv marked this pull request as ready for review December 15, 2024 23:56
@coveralls
Copy link
Collaborator

coveralls commented Dec 16, 2024

Pull Request Test Coverage Report for Build 12455551235

Details

  • 115 of 149 (77.18%) changed or added relevant lines in 9 files are covered.
  • 2 unchanged lines in 2 files lost coverage.
  • Overall coverage decreased (-0.002%) to 91.758%

Changes Missing Coverage Covered Lines Changed/Added Lines %
app/javascript/Components/Result/binary_viewer.jsx 27 30 90.0%
app/javascript/Components/Result/file_viewer.jsx 22 26 84.62%
app/javascript/Components/Result/text_viewer.jsx 35 44 79.55%
app/javascript/Components/Result/image_viewer.jsx 0 18 0.0%
Files with Coverage Reduction New Missed Lines %
app/javascript/Components/Result/file_viewer.jsx 1 72.41%
app/models/role.rb 1 93.22%
Totals Coverage Status
Change from base Build 12345653211: -0.002%
Covered Lines: 41180
Relevant Lines: 44201

💛 - Coveralls

@ch-iv ch-iv requested a review from david-yz-liu December 16, 2024 15:05
Copy link
Collaborator

@david-yz-liu david-yz-liu left a comment

Choose a reason for hiding this comment

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

Nice work, @ch-iv! In addition to the inline comments, you can go ahead and remove all traces of the get_file method. This includes the controller method, the routes.rb file, the submission_policy.rb file, and any related tests.

Changelog.md Outdated Show resolved Hide resolved
app/controllers/results_controller.rb Outdated Show resolved Hide resolved
app/controllers/submissions_controller.rb Outdated Show resolved Hide resolved
app/javascript/Components/Result/image_viewer.jsx Outdated Show resolved Hide resolved
app/javascript/Components/Result/file_viewer.jsx Outdated Show resolved Hide resolved

return (
<React.Fragment>
<div style={{display: this.state.loading || this.state.errorMessage ? "none" : "block"}}>
Copy link
Collaborator

Choose a reason for hiding this comment

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

This extra div is causing a rendering problem with the PDF viewer (height collapsing).

Fixing CSS is a bit tricky; instead of wrapping in a div, you can pass a display prop to each viewer, and use that to control the display of each component.

this.display_annotations();
this.adjustPictureSize();
this.rotateImage();
if (this.props.url && this.props.url !== prevProps.url) {
Copy link
Collaborator

Choose a reason for hiding this comment

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

Oh, sorry I realized that only the setImageURL should be guarded in this way. The this.display_annotations/this.adjustPictureSize/this.rotateImage should always trigger, otherwise the annotations won't show up properly. (Test in UI by creating an annotation and changing the zoom/rotation.)

@ch-iv ch-iv requested a review from david-yz-liu December 22, 2024 16:02
Copy link
Collaborator

@david-yz-liu david-yz-liu left a comment

Choose a reason for hiding this comment

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

Thank you, @ch-iv!

@david-yz-liu david-yz-liu merged commit c0bc5fd into MarkUsProject:master Dec 22, 2024
4 of 6 checks passed
@ch-iv ch-iv deleted the refactor-file-viewer branch December 24, 2024 04:32
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants