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: fetch file from url + more #115

Open
wants to merge 29 commits into
base: main
Choose a base branch
from

Conversation

avanavana
Copy link

@avanavana avanavana commented Mar 10, 2025

Hi Theo, long time sub/viewer, but first time contributor to one of your projects.

I love this simple little app, but time and again I found myself repeatedly running into a limitation in the original featureset. A very common use case for me is to need to edit images using the tool that exist online, or that I find online via a web search. As it stands, this requires me to download a file every time and then re-upload it in order to use the tool.

This PR and feature branch, provides this missing functionality, in the form of an alternative way of inputting images in each of the three tools—fetch file from URL. Basically, when using any of the tools, a user can either click upload/drag and drop/paste a file to upload, or—with these changes—they can copy and paste any URL that resolves to an image and the app will fetch that file, process it, and use that image instead.

TL;DR, Major Changes

  • Added ability to fetch files from URL
  • Standardized & generalized processing logic between existing file upload and new file fetching functionality
  • Improved error-handling, especially in terms of user experience
  • App-wide accessibility improvements and 100% keyboard navigability
  • Improved usability at smaller, mobile screen sizes
  • Calculate and display scale of preview images for users, when uploaded images are larger than container and sized to fit
  • Preview SVGs on light background as well as default dark background before converting to PNG
  • Add transparent background option to square image tool
  • Add "cover" image-fit mode to existing "contain" mode for square image tool, with a new toggle/option selector, as an alternative way to derive a square image from an existing image
  • Improvements on input and validation for custom SVG scale and custom border radius input
  • Improved, standardized styling across the app, while staying true to its minimalist origins
  • General organizational improvements and consolidation of duplicate code into components

BTW, I'm currently looking for work. I'm what is more and more being called a "Design Engineer"—I have 15+ years experience both as a senior/lead product designer and as a developer (full stack, but most of my work tends to be more front-end-y). React+TypeScript is my center of gravity and has been since 2018. See: www.avanavana.com

Videos

🎥 Demoing fetch file from URL and SVG scale factor improvements
https://github.com/user-attachments/assets/e49b3890-6f14-4ed5-97aa-3f835197b7d0

🎥 Demoing some of the new features, dynamic preview scale, keyboard navigation, etc
https://github.com/user-attachments/assets/deb28194-ca78-41c3-8576-468a4a66e12b

🎥 Demoing corner rounder proportional border radius improvement
https://github.com/user-attachments/assets/7115ffec-a247-403c-9a81-397b0f770dd2

🎥 Demoing behavior of "too-wide" and "too-tall" images with screen-size change
https://github.com/user-attachments/assets/3b1fec9b-8f6e-47ed-90e2-cea994246b62

Summary

In implementing the fetch file from URL feature, I aimed for maximum parity and mutual intelligibility between the existing useFileUploader() hook by implementing it with a new custom hook, useFileFetcher(), which uses React's new-ish useActionState() hook on the back-end, as well as by abstracting/generalizing the file processing logic that was previously located entirely within useFileUploader() as its own useProcessFile() hook, which now serves both hooks/methods of retrieving files.

Along with implementing this feature I also implemented better error handling, especially from the point of view of users. Instead of console errors and alert(), there are now real error messages whose state lives at the top level of each tool and whose set method is passed into the useFileUploader(), useClipboardPaste(), and new useFileFetcher() hooks, allowing each method of file retrieval to produce and render its own distinct errors in a simple, centralized way.

I also implemented accessible keyboard navigation for all interactive elements, using a powerful new, custom useKeydown() hook so the entire app can be controlled by means of keyboard control, and I implemented WAI-ARIA best practices.

For the option selector components like OptionSelector, SvgScaleSelector, and BorderRadiusSelector, at small screen sizes, those with many options (especially SvgScaleSelector) were overflowing in the x-direction. I implemented a simple useMediaQuery() hook to conditionally render these option selector components with a new custom Select component at small screen sizes. Initially I tried implementing it with a native <select> element, but the conditional mounting led to a problem where, due to browser restrictions, the newly-mounted native element was unable to be operated with a single click/touch event—the first click/touch event would focus the element but the browser would then prevent its options from being shown—until after a second click/touch event. This was unacceptable from a UX perspective, so I decided to implement it as a custom Select component. This component is fully accessible and keyboard-navigable, using all the same events as the native element.

Beyond this, I did some work standardizing and improving styling across the app, while keeping it true to the original look and feel, and I consolidated and standardized various pieces of UI shared across the three tools.

A full list of the updates is provided below, along with the testing I performed.

Additions & Updates (Full List)

Home

  • Center paragraph text
  • Make links tab-navigable and accessible and give them hover+focus styles
  • Standardize default, hover, and focus styles of all header and footer links
  • Add icons to 3 primary tool links to help distinguish and highlight them
  • Standardize capitalization of all tool links and style them more like other links (e.g. hover/focus styles should be either color change or underline, but not a mix of both throughout app—I went for color change)

All Tools

  • Standardize capitalization of page titles on all tools, to keep them consistent with capitalization on home page
  • Add page titles to tool screens by passing down title from metadata object in each tool's page.tsx file
  • Standardize default, hover, and focus styles of all header and footer links
  • Encapsulate footer into shared, reusable component
  • Standardize capitalization of FileDropzone text
  • Add same upload icon to FileDropZone component
  • Standardize text color between FileDropzone and UploadBox
  • Initial tool screen
    • Add fetch file from URL form
    • Add fetch file from URL custom hook
    • Standardize file upload and fetch from URL hook return objects
    • Generalize file upload and fetch from URL hooks to both use the same new useProcessFile() custom hook, which performs standardized file processing logic behind the scenes and allows for standard implementations of multiple methods of file input
    • Add tab navigation between file upload, url input, & url submit
    • Add escape key exit to home screen
    • Add enter & space key triggers for file upload
    • Standardize error reporting for all user input methods
      • Create ErrorMessage component
      • Standardize error throwing and catching to use new Error Message component and handle errors globally with tool-level error state as a single source of truth, so that errors resulting from any input method are each displayed using the same top-level error state and update correctly
      • File upload
        • Upload via upload form
          • Invalid/non-image file displays error
          • Invalid image type displays error
          • Invalid/corrupted file displays error
        • Upload via paste from clipboard
          • Invalid/non-image file displays error
          • Invalid image type displays error
          • Invalid/corrupted file displays error
        • Upload via drag and drop
          • Invalid/non-image file displays error
          • Invalid image type displays error
          • Invalid/corrupted file displays error
      • Fetch from URL
        • Invalid/non-image file displays error
        • Invalid file type displays error
        • Invalid/corrupt file displays error
        • Invalid url displays error
        • Missing url displays error
        • Unresolvable url displays error
        • Protected/forbidden url displays error
        • CORS protection displays error
        • Remote server error displays error
        • Malformed request displays error
        • Invalid HTTPS/TLS certificate displays error
        • Permanently moved resource displays error
    • Standardize styles for all input elements, including hover & focus states
    • Standardize styles for all button elements, including hover & focus states
    • Make file drop zone slightly wider to allow for a little more room for new fetch from URL form below
    • Replace ugly file upload icon with nicer file upload icon from lucide icons
    • Add "allows paste from clipboard" message to all initial tool screens
  • Preview screen
    • Add dynamic preview scale % above image preview
    • Add escape key to trigger cancel
    • Calculate width & height for SVGs without explicit width and height from SVG viewBox attribute, instead of arbitrary 300x150 size, which caused issues with centering user-provided content in the square image tool
    • Display all images under maximum width/height dimensions (determined by container) at actual size
    • Scale all images with at least one dimension larger than maximum container width/height down according to their largest dimension
    • Add "0.5x" value to scale options
    • Validate and format input for custom scale
    • render as select dropdown below xs (29rem—i.e. 25rem + 2rem padding) breakpoint
      • Create custom Select component, since browser restrictions prevent the use of native <select> (when the native select mounts after screen resize, the browser prevents initial click events, requiring users to click twice with the menu flashing open and closed very quickly and awkwardly—unacceptable)
        • Maintain all standard input styles, including hover and focus styles
        • Handle all native-equivalent keydown events for accessibility
          • ArrowUp/ArrowDown/Enter/Space to open select menu options
          • ArrowUp/ArrowDown to navigate menu options, with infinite cycling at first and last options
          • Enter/Space to select a menu option
          • Escape to close menu options and/or blur select element
        • Handle all native-equivalent mouse events
        • Integrate component with OptionSelector/SvgScaleSelector/BorderRadiusSelector state
        • Implement all necessary WAI-ARIA attributes for accessibility
        • Replace native <select> element with new custom Select component, conditionally rendering it in OptionSelector/SvgScaleSelector/BorderRadiusSelector when the screen is below the "xs" breakpoint, and in a 50/50 flex-row arrangement with the "custom" input field when it is present
    • Add native-style keyboard controls (spacebar + down arrow to open, arrow keys to change selected option, option/alt or cmd + arrow keys to jump to top/bottom of options, enter or spacebar to confirm selection) to <select> version of scale selector
    • Add "Allows pasting images from clipboard" message to all initial tool screens
    • Add clipboard paste icon to "Allows pasting images from clipboard" message
    • Drag and drop
      • Render actual drag and drop error messages below UI instead of alert dialogs and console errors

SVG to PNG tool

  • Initial tool screen
    • Paste from clipboard
      • Only accept *.svg files from paste from clipboard
  • Preview screen
    • Add preview scale % above image preview
    • Center original/scaled size text

Square image tool

  • Preview screen
    • Add preview scale % above image preview
    • Center original/square size text
    • Add "transparent" option for export
    • Transparent option shows very faint "checkboard" pattern in extra portions of the new square image that help distinguish it from the default black background and give proper feedback to users after they choose "transparent" as an option
    • Fix bug where SVGs without explicit width and height can't be center positioned in square (fixed by using viewbox attribute to size SVG)
    • Encapsulate "Save Image" element into a separate component defined in square-tool.tsx
    • Add new OptionSelector component to allow users to toggle between "contain" (existing behavior) and "cover" image-fit/object-fit modes/methods of "squaring" a source image.

Rounded corner tool

  • Preview screen
    • Add preview scale % above image preview
    • Center actual size text
    • Add "transparent" option for export
    • Transparent option shows very faint "checkboard" pattern behind rounded corners that help distinguish it from the default black background and give proper feedback to users after they choose "transparent" as an option
    • Add "1px" value to radius options
    • Validate and format input for custom radius

General/components

  • Group and sort all imports in a standardized way in all files as 1. react+installed packages (react first, then next, then other installed packages in ascending lexicographical order), 2. components (asc. lex. order), 3. hooks/lib/utils (in that order, and then asc. lex. order within each group), and finally, 4. imported types (asc. lex. order), with a single newline space between import groups
  • Standardize button styles across app (no more "success" and "destructive" bg colors—nothing about "cancel" is irreversible or causes loss of data, and in other contexts, primary CTA uses the "blue" color. Cleaner and simpler design if all buttons just use the same style, cancel can use a secondary "ghosted" style.
  • Standardize input styles across app. Unify heights, padding, typography, colors, and hover/focus states.
  • Distinguish external from internal links with up-and-to-the-right arrow icon
  • Encapsulate footer into shared, reusable component
    • Add subtle GitHub logo icon to GitHub link
  • UploadBox
    • Slightly widen upload box to give more room for fetch from URL form below
    • Replace ugly icon with better-proportioned icon
    • Make text color consistent and better-balanced
  • FileDropzone
    • Add same updated upload icon from UploadBox
    • Review usage of useCallback() throughout app to avoid premature or negligible optimization, i.e. unnecessary usage of useCallback that may add unnecessary overhead (useCallback() only makes sense in cases where 1. heavy computation is necessary, 2. referential equality is necessary, and 3. in general, where it actually improves performance in a way that can be measured (i.e. measure performance before & after before employing it). useCallback() is generally unnecessary when it is used with both no dependencies and the function it wraps is not being passed to any props in child components. In other words, if it is used to wrap handlers in a function that are not passed down and only used in that component, with no dependencies, and no heavy computation, it is unnecessary and just adding to overhead.

Testing

main screen

  • tab navigation between 3 main tool links
  • links should not be un-focusable/blur-able with Escape key (as with all native link tab focus)
  • all links have consistent font styles, including hover & focus states (other than color, which is used to indicate external
  • main content should be vertically and horizontally centered in window

all tools

  • initial tool screen
    • tab navigation between file upload, url input, & url submit
    • escape exits to main page
    • enter opens file upload
    • space opens file upload
    • if file upload button is not focused, enter and space do not open file upload
    • escape does not blur file upload when focused (ensure native behavior)
    • focus on file upload does not block escape key exit to main page
    • enter attempts to submit url form when either url input or submit button are focused
    • enter does not submit url form when neither url input nor submit button are focused
    • attempt to submit empty url form results in focused input
    • enter does not submit url form when url input is empty
    • empty url form submission does not trigger native input required popover
    • empty url form submission results in "URL is required" error message, without any network requests being sent
    • escape blurs url input when focused (ensure native behavior)
    • escape does not blur url submit button when focused (ensure native behavior)
    • focus on url submit button does not block escape key exit to main page
    • all input elements have consistent styles, including hover & focus states
    • all button elements have consistent styles, including hover & focus states
    • file upload + url upload UI is fixed width above sm (30rem) breakpoint
    • file upload + url upload UI is 100% fluid width (minus padding) below "xs" (29rem—i.e. 25rem + 2rem padding) breakpoint
    • smooth transition between fixed and fluid layouts across "xs" breakpoint
    • main content should be vertically and horizontally centered in window
    • use file upload button to upload
      • invalid file type displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid/corrupt file displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
    • drag and drop to upload
      • invalid file type displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid/corrupt file displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
    • paste from clipboard to upload
      • non-image file type displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid file type displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid/corrupt file displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
    • fetch from url to upload
      • non-image file displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid file type displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid/corrupt file displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid url displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • missing url displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • unresolvable url displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • protected/forbidden url displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • CORS protection displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • remote server error displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • malformed request displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • invalid HTTPS/TLS certificate displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
      • permanently moved resource displays error
        • SVG to PNG
        • Square Image
        • Rounded Corners
    • existing error message is replaced with any new error messages (only one error message displayed at once)
    • errors dismissed after valid upload or url fetch, or navigation to main page and back again to initial tool screen
  • preview screen
    • paste new image over existing image in preview
    • tab navigation between all option inputs, cancel, and save buttons
    • all input elements have consistent styles, including hover & focus states
    • all button elements have consistent styles, including hover & focus states
    • cancel button exits to initial tool screen
    • escape cancels to initial tool screen
    • escape blurs focused inputs, buttons (including option selector component buttons), & links before canceling to initial tool screen
    • preview scale calculated correctly
    • preview scale changes with screen + image preview resize
    • preview image scales smoothly through different breakpoints
    • if image is shown at actual size (i.e. image dimensions are both less than the maximum container size), preview scale should not be shown
    • if any dimension of an image is greater than the maximum container size, the preview scale should always be shown, with the scale percentage displayed equal to the maximum container size divided by the image's largest dimension
    • main content should be vertically and horizontally centered in screen

svg-to-png-tool

  • initial tool screen
    • upload
      • smaller than max size in both dimensions
      • wider than max size
      • taller than max size
      • larger than max size in both dimensions
      • paste from clipboard to upload should only accept *.svg files
    • fetch from URL
      • smaller than max size in both dimensions
      • wider than max size
      • taller than max size
      • larger than max size in both dimensions
      • enter submits url form
  • preview screen
    • choose preview background
      • preview on light bg
      • preview on dark bg
    • choose export scale
      • fixed scales
        • scaled (fixed) size is calculated correctly
        • export at fixed scale
      • custom scale
        • show placeholder when empty
        • set value to 1 if empty or 0
        • set value to 64 if user manually enters a value above 64
        • manually enter any real number 0 < x <= 64
        • leading and trailing whitespace is removed
        • decimal values always have only one leading zero
        • trailing zeroes are removed
        • user must not be able to enter non-numeric characters
        • step values up and down by 1 with mouse clicks on buttons
        • step values up and down by 10 with mouse clicks on buttons + shift key
        • step values up and down by 0.1 with mouse clicks on buttons + alt/option key
        • tab to step up/down buttons
        • step values up and down by 1 with arrow keys
        • step values up and down by 10 with arrow keys + shift key
        • step values up and down by 0.1 with arrow keys + alt/option keys
        • scaled (custom) size is calculated correctly
        • scaled (custom) size is only ever shown with a maximum of 2 decimal places in all locations other than the actual user's input in the custom scale input field.
        • custom scales all result in images with whole number dimensions
        • user can successfully export at any custom scale
      • render as custom Select component below "xs" (29rem—i.e. 25rem + 2rem inline padding) breakpoinirgvt gfvv
      • open and operate Select with native keyboard controls (spacebar + down arrow to open, arrow keys to change selected option, option/alt or cmd + arrow keys to jump to top/bottom of options, enter or spacebar to confirm selection)
      • selected item should be preserved between display mode changes (between the normal and Select display modes across the 30rem breakpoint

square-tool

  • initial tool screen
    • upload
      • smaller than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • wider than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • taller than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • larger than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • [x ] webp
    • fetch from URL
      • smaller than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • wider than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • taller than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • larger than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
  • preview
    • choose background
      • white bg
        • preview updates correctly
        • export saves correctly
      • black bg
        • preview updates correctly
        • export saves correctly
      • transparent bg
        • preview updates correctly
        • export saves correctly
    • choose image fit mode
      • contain
        • preview updates correctly
        • sizes update correctly
        • export saves correctly
      • cover
        • preview updates correctly
        • sizes update correctly
        • export saves correctly

rounded-tool

  • initial tool screen
    • upload
      • smaller than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • wider than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • taller than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • larger than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • enter opens file upload
      • space opens file upload
      • if file upload button is not focused, enter and space do not open file upload
    • fetch from URL
      • smaller than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • wider than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • taller than max size
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • larger than max size in both dimensions
        • svg with transparency
        • svg with full bleed
        • png with transparency
        • png with full bleed
        • jpg
        • webp
      • enter submits url form
      • escape cancels to initial tool screen
  • preview screen
    • choose background
      • white bg
        • preview updates correctly
        • export saves correctly
      • black bg
        • preview updates correctly
        • export saves correctly
      • transparent bg
        • preview updates correctly
        • export saves correctly
    • choose border radius
      • fixed radii
        • border radius is calculated correctly
        • border radius is scaled correctly in preview according to image preview scale
        • exports correctly
      • custom radius
        • show placeholder when empty = -1
        • set value to 1 if empty or 0 = 1
        • set value to 999 if user manually enters a value gte 1000,
        • manually enter any real number 0 < x < 1000 (e.g. a user can manually enter numbers arbitrarily close to 0 and 1000)
        • leading and trailing whitespace is removed
        • decimal values always have only one leading zero
        • trailing zeroes are removed
        • user must not be able to enter non-numeric characters
        • step values up and down by 1 with mouse clicks on buttons
        • step values up and down by 10 with mouse clicks on buttons + shift key
        • step values up and down by 0.1 with mouse clicks on buttons + alt/option key
        • tab to step up/down buttons
        • step values up and down by 1 with arrow keys
        • step values up and down by 10 with arrow keys + shift key
        • step values up and down by 0.1 with arrow keys + alt/option keys
        • custom radii are applied to the preview image at the correct scale depending on whether the preview image itself is scaled to fit its container (i.e. whether its dimensions exceed the container or not
      • render as custom Select component below "xs" (29rem—i.e. 25rem + 2rem inline padding)
      • open and operate Select with native keyboard controls (spacebar + down arrow to open, arrow keys to change selected option, option/alt or cmd + arrow keys to jump to top/bottom of options, enter or spacebar to confirm selection)
      • selected item should be preserved between display mode changes (between the normal and Select display modes across the 30rem breakpoint

avanavana added 27 commits March 9, 2025 03:23
Added core functionality for "fetch from URL" method of file upload,
for all three tools, using custom "useFileFetcher" hook, which uses
"useActionState" behind the scenes.  Refactored and generalized existing
"useFileUploader" hook along with the new "useFileFetcher" hook to both
use a common "useProcessFile" hook.  A "FetchFromUrlForm" component was
also created to sit alongside the existing "UploadBox" component to
accommodate user interaction in the UI for both methods of retrieving
files for all three tools, the container holding these two components
was widened a touch to accommodate the new FetchFromUrlForm, and some
styles were updated to standardize styling between the two methods.
Errors are now handled in a standardized, universal way for all three tools.
A shared ErrorMessage component displays the error, and each tool manages
a single piece of error state at its top level, to make it easy to manage
errors when users trigger multiple errors, even between different methods
of file retrieval (e.g. as opposed to managing error state as a FIFO or list
data structure).  The error state and set method are simply passed as arguments
to the useFileFetcher and useFileUploader hooks where they are used at the top
level of each tool, and then all of the various issues that can trigger errors
germane to each tool can easily set custom error messages.  If a setError/onError
method is not passed to the hooks, they will default to the original (somewhat
crude, tbh) method of displaying errors via alert().
Oops—forgot to add @Keyframes for the Spinner subcomponent of FetchFromUrlForm,
which displays inside the submit button when useFileFetcher's useActionState sets
form state as "pending".
Removed 3 unnecessary uses of useCallback. (useCallback only provides a benefit
when 1. dependencies are specified in the dependency array, and/or 2. the function
wrapped in useCallback is being passed as a prop to another component, in order to
provide either referential equality across renders, or to memoize expensive
calculations—none of which is the case for these three simple handlers that belong
solely to this component, require no dependencies, and do not perform expensive
calculations.  The fourth usage of useCallback, however, makes sense.  Unnecessary
usage of useCallback adds unnecessary overhead and overcomplicates code.

Also slightly improved the styling of FileDropzone—added an "upload" icon as in the
regular/non-dnd UploadBox component for consistency, and standardized text colors.
Switched ugly "upload" icon in UploadBox component out for much more
balanced, tighter "upload" icon from lucide icons, which I had previously
also added to the FileDropzone component, so this also makes the two
components consistent.
The same strategy of handling errors in the file upload box/form, drag and drop
component, and fetch file from url form has been extended to the paste file from
clipboard functionality, for all 3 tools.
Consolidated many inconsistencies among text styles, unified button styles,
standardized capitalization of text in various places.
The structure, styles, and size of all preview images for each of the three tools
has been standardized.  More importantly, whereas previously images larger than
the maximum container width would be scaled to fit, but the user had no idea what
that scale was (and border radius in the corner rounder tool was not actually being
scaled in the preview according to the real dimensions of the image), now, if an
uploaded (or fetched) image is larger than the maximum container dimensions, a
preview scale will appear above the preview image showing the user exactly at what
scale they are currently viewing the preview image.  This preview is dynamic and
connected to a ResizeObserver, so as the user changes their window size (and this
is particularly important for mobile users and users with smaller window sizes,
where nearly all uploaded images must be scaled), the preview scale will change
dynamically as well.
Enter/escape key functionality on initial tool screen and previews,
implemented with a powerful, custom "useKeydown" hook with typesafety,
modifiers, excluded modifiers, conditionals, targeting, capture phase,
preventDefault and stopPropagation.  A DOM utility function has been added
to detect focused interactive elements in order to be able to conditionally
target keydown events with the hook and work around the native behavior of
certain keys without overriding them and losing their functionality.
A number of improvements were made on the various option selector components
(generic option selector, svg scale selector, and border radius selector).

- Designed and implemented standardized focus states (esp. for keyboard interaction)
- Implemented custom increment/decrement buttons on custom number inputs
- Made custom number inputs with new increment/decrement buttons fully accessible +
  keyboard navigable
- Generalized handler function for custom number inputs so that it can be used both for
  input keydown events as well as increment/decrement button click events, just by passing
  a typed context property along with its event object
- Significantly improved the range of numbers a user can type into the custom number inputs,
  and implemented foolproof validation.  Users can now remove all characters from the custom
  number inputs in order to begin typing something like "0.25", alt/option and shift click
  key commands still work for small/large increments/decrements, and user-entered values such
  as 63.9999 and 24.179 will be rounded to a maximum of 2 decimal places, as values like 64
  and 24.18, respectively.
- Handle null/empty string values better
- Improved hover styles on options
At smaller screen sizes, the regular option selector (plus svg scale selector and
border radius selector) component can be too wide to fit on the screen and cannot
resize or wrap to fit.  This commit implements a custom select component that is
swapped for the linear scale selector below the "xs" breakpoint (25rem, the width
of the upload box + fetch from url form, plus 2rem inline padding on either side
= 29rem).  The custom select element is fully accessible and keyboard navigable
and implements its keyboard functionality with the custom useKeydown hook.
Conditional rendering across the "xs" breakpoint is handled with a custom
useMediaQuery hook powered by React's new useSyncExternalStore, which is meant to
interact with browser APIs like this.  When the Select component is rendered instead
of the regular option selector components and the "custom" value is selected the
custom value input is rendered in a row beside the custom Select component, instead
of in a column layout as with the regular option selector.  For the SVG scale
selector, a new value of 0.5 or 1/2 scale was added as the smallest scale, since it
is an extremely common use case for users to halve the size of an image.
All app icons have been consolidated into a single shared component file with multiple exported
icon components, each sharing common default props, and exposing a handful of props like strokeWidth
and className. In refactoring icons across the app in this manner, I have also added icons to a few
important places.  A download icon has been added to the main "save" buttons in each of the three
tools, which really helps to communicate the fact that those buttons are primary CTAs and highlights
their action in an immediate, visual way, and a GitHub logo icon has been added to the footer link
to the app's repo on GitHub. This helps balance and standardize the GitHub link in the footer with
the "Back" link in the header. with its icon.
Footer content within a <footer> element has been consolidated into a
shared <Footer> component.
The blocks under the preview image on each tool which display information
such as "original (size)" and "actual (size)", "scaled (size)", etc have had
white-space: nowrap applied to them to ensure that at smaller screen sizes
the layout does not become messy.
A page title, fed by the title property in the page metadata has been
added to each tool page, rendered in a shared <PageTitle> component,
which fits nicely in line with the "Back" button on each tool page.
This page title makes it a lot more obvious for users which tool they
are using, since otherwise they can look very similar, and balances
the header more visually.
Previously, if users uploaded a transparent SVG with dark (esp. black) artwork,
the image would be completely unreadable on the site's dark/black background.

This commit adds an option selector to the svg-to-png tool that allows users to
toggle between dark and light preview backgrounds (i.e. not included in PNG export)
so that they can actually see the SVGs they upload in such cases.
An option to make an image square, with the extra background transparent has
been added to the square-tool, and both square-tool and rounded-tool now render
their "transparent" options with a classic "checkerboard" style pattern, because
otherwise it is impossible to tell where transparency is if the image has a black
background or black artwork, given the black background of the site.
User input that is >= 1000 should be clamped to an even 999, while still allowing
users to manually input values arbitrarily close to 1000 such as 999.9999999999,
or use the increment button with the alt/option key to enter 999.9.
Because the footer link pushes the main content/container up by 60px, the main
content isn't vertically centered.  The "back" button and the PageTitle component
that was implemented to emulate its design are not part of the static flow of the
page, so they do not push the content down equally. Therefore we have to add a top
margin equal to the height of the footer in order to keep the content centered both
on the main page.tsx and the tool page layout.tsx.
At some point I was sure that the native Space and Enter for activating the native
file input was working, but this usage of the useKeyDown() hook ensures it works and
is fully keyboard navigable.
Copy link

vercel bot commented Mar 10, 2025

The latest updates on your projects. Learn more about Vercel for Git ↗︎

Name Status Preview Comments Updated (UTC)
quickpic ✅ Ready (Inspect) Visit Preview 💬 Add feedback Mar 11, 2025 4:13pm

If a user uploaded an SVG image with no explicit width and height
attribute defined, this would cause the exported image to be malformed.

This was fixed by reverting to using viewBox attribute dimensions if
those explicit dimensions are missing.  SVGs missing both width/height
and viewBox throw errors at upload/fetch time, so SVGs are guaranteed
to have valid dimensions in either explicit width/height attributes or
in a viewBox attribute.
The square image tool so far has only ever taken a source image and "contained"
it within a square representing the largest dimension of the source image.  However,
there is another way to square an image, and that is to take the smallest dimension
of the source image as the square size, and then "cover" this square with the source
image. The "object-fit" property in CSS is very similar.  I implemented this with an
instance of the OptionSelector component that toggles between the two modes, with
"contain", the status quo, as the default value.  Now, users can toggle between these
two methods of "squaring" an image at will.

I also fixed issues that were related to differences in how the HTML canvas "drawImage"
method handles SVG vs. raster data.  This was causing some SVG images to be incorrectly
positioned in "cover" mode.
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.

1 participant