remix-img
An image optimization library for Remix (Beta).
An image optimization library for Remix inspired by Next.js’s <Image />
component. I built this library because I needed a working solution that offers dynamic image optimization with server-side processing, responsive images, lazy loading, and caching similar to Next.js Image. I deploy my projects on environments like Linode (with persistent storage) and Vercel (serverless), so the default file-system caching solution works well for my needs. Since I’m already using something very similar to this in my projects, I decided to package it as an easy-to-use library.
- Github Repo remix-img.
Note: This library is currently in beta.
-
Dynamic Image Optimization:
Uses Sharp for on-demand image processing (resizing, format conversion, and optimization). -
Caching & Revalidation:
- Optimizes images on-demand and caches them using a file-system based adapter (default:
.cache/images
). - Mimics Next.js’s caching strategy by storing each image variant in its own subdirectory with a unique filename and accompanying metadata.
- Supports smart cache headers (e.g.
x-remix-img-cache
:MISS
,STALE
, orHIT
) and usesCache-Control
headers (public, max-age=<TTL>, stale-while-revalidate=<timeout>
) so that downstream caches (or CDNs) can quickly serve images. - Includes a config flag (
useEdgeCache
) that allows bypassing local caching to rely solely on CDN caching in serverless environments.
- Optimizes images on-demand and caches them using a file-system based adapter (default:
-
Responsive Images:
Automatically generatessrcset
andsizes
attributes based on configured device sizes, ensuring the browser downloads the optimal image for the viewport. -
Native Lazy Loading & Priority:
Leverages browser-native lazy loading with an IntersectionObserver fallback and supports eager loading for priority images. -
Fallback Placeholder & Blur-Up:
Supports providing a custom placeholder image. (TODO: Implement auto-generated blurred placeholders.) -
Custom Loader Support:
Allows you to plug in your own loader function to tailor image processing if desired. -
Pluggable Caching:
Exposes a cache adapter interface so that you can replace the default file-system cache with a custom adapter (e.g., one that uses S3, Cloudflare KV, or a hybrid cache like@next-boost/hybrid-disk-cache
for more robust production use).
Install via npm:
npm install remix-img
-
Configure the Library
Modify your entry.server.tsx (project entry) in your remix project with the following:
import { updateConfig } from "remix-img"; // override config updateConfig({ baseUrl: "http://localhost:5173", // Match your dev base url so that both SSR and client generate the same urls. allowedDomains: [ "example.com", "cdn.example.com", "localhost", "mycustomdomain.com", ], cacheDir: process.env.REMIX_IMG_CACHE_DIR || ".cache/images", useEdgeCache: process.env.NODE_ENV === "production", defaultQuality: process.env.DEFAULT_QUALITY ? Number(process.env.DEFAULT_QUALITY) : 85, }); // ... rest of your entry.server.ts code
The library loads these overrides and merges them with the default configuration.
-
Add Server-Side Image Optimization
In your Remix App, create a route that points to the image optimizer loader. For example, in
app/routes/optimized-image.ts
:// app/routes/optimized-image.ts import { loader as imageLoader } from "remix-img/server"; export { loader };
This loader:
- Validates the image URL and allowed domains.
- Processes the image using Sharp (on a cache MISS or stale condition).
- Leverages caching (or bypasses it if
useEdgeCache
is true) and sets smart Cache-Control headers.
-
Use the Image Component
Import and use the
<RemixImage>
component in your Remix routes or components:// app/routes/gallery.jsx import Image from "remix-img/client"; export default function Gallery() { return ( <div> <h1>Gallery</h1> <RemixImage src="https://example.com/my-image.jpg" alt="A sample image" width={800} height={600} sizes="(max-width: 768px) 100vw, 50vw" // custom sizes override placeholder="blur" // Use a custom placeholder URL (auto-blur TODO) loading="lazy" // Defaults to lazy loading onError={(e) => console.error("Image failed to load", e)} /> </div> ); }
With the updated package entry point (see next section), you can import directly from
remix-img
(see below).
-
src:
string
The source URL of the image. -
alt:
string
Alt text for the image, used for accessibility. -
width:
number | string
The desired width of the image. Can be provided as a number (e.g.800
) or as a string literal (e.g."800"
). -
height:
number | string
The desired height of the image. -
fill:
boolean
If true, the image will fill its container (ignoring explicit width/height).
Used for “fill” layout mode. -
sizes:
string
Allows the user to override the defaultsizes
attribute on the<img>
element.
For example:(max-width: 1200px) 100vw, 1200px
. -
loader:
(src: string, width: number, quality?: number) => string
Custom loader function that overrides the default URL builder for image optimization. -
quality:
number | string
The desired image quality (0–100). Accepts either a number or a string literal. -
outputFormat:
"auto" | "webp" | "jpeg" | "jpg" | "png"
Specifies the output format of the image. If set to"auto"
, the original format is preserved (though our loader defaults to JPEG when auto is used). -
priority:
boolean
If true, forces the image to load immediately (eager loading) rather than lazy loading. -
loading:
"eager" | "lazy"
Determines the browser’s native loading behavior. Defaults to"lazy"
if not specified. -
placeholder:
"blur" | "empty" | string
Indicates how to handle the placeholder.- If set to
"blur"
, it currently requires a custom placeholder URL (TODO: auto-generate a blurred placeholder). - Otherwise, it can be a URL to a custom placeholder image or
"empty"
.
- If set to
-
blurDataURL:
string
A data URL for a pre-generated blurred placeholder image (planned for auto-generation in a future version). -
unoptimized:
boolean
If true, bypasses image optimization and returns the original image URL. -
overrideSrc:
string
If provided, this URL is used directly for the<img>
element, completely bypassing optimization logic. -
onLoadingComplete:
(img: HTMLImageElement) => void
Callback function that is invoked once the image has fully loaded. -
layout:
"fill" | "fixed" | "intrinsic" | "responsive"
Determines the layout behavior of the image, similar to Next.js Image. -
objectFit:
React.CSSProperties["objectFit"]
CSS property for controlling how the image should be resized to fit its container (commonly used withfill
layout). -
objectPosition:
React.CSSProperties["objectPosition"]
CSS property for positioning the image within its container. -
lazyBoundary:
string
Defines the boundary for lazy loading. -
lazyRoot:
Element | null
Specifies the root element for lazy loading (useful for advanced lazy loading configurations).
The optimizer loader (in server/optimized-image.ts
) handles:
- Domain validation.
- Processing the image using Sharp.
- Caching via a pluggable cache adapter (default is a file-system adapter that stores each variant in its own subdirectory).
- Setting smart Cache-Control headers (
public, max-age=<TTL>, stale-while-revalidate=<timeout>
) and a customx-remix-img-cache
header.
- minimumCacheTTL: Number (seconds)
- staleWhileRevalidate: Number (seconds)
-
baseUrl:
string
-
allowedDomains:
string[]
-
cacheDir: Directory path for caching (default:
.cache/images
) - defaultQuality: Number (0-100)
-
deviceSizes & imageSizes:
number[]
– Used for generating responsivesrcset
-
formats:
string[]
– Supported output formats - cdnConfig: Optional CDN settings (not fully implemented yet)
- useEdgeCache: Boolean – If true, bypasses local caching in favor of edge/CDN caching
-
cacheAdapter: Optionally, users can supply a custom adapter implementing the
CacheAdapter
interface. -
Custom Config File:
Users can override defaults by creating a.remix-img.config.json
file at the project root.
-
Caching Strategy:
Mimics Next.js’s approach by storing each image variant in its own subdirectory.- MISS: First-time processing and caching.
- STALE: Cached image is outdated; served immediately while revalidated in the background.
- HIT: Cached image is fresh.
-
Smart Cache Headers:
The response includes:Cache-Control: public, max-age=<TTL>, stale-while-revalidate=<timeout>
-
x-remix-img-cache
: Indicates the cache status.
-
Pluggable Cache Adapter:
The default is a file-system adapter (ideal for persistent environments like Linode/VMs). For serverless deployments (e.g. Vercel), a custom adapter can be provided to offload caching to an external service or rely on CDN caching (controlled via theuseEdgeCache
flag).
-
Support Environment Variables:
At the moment users can override config using a .json file. I need to implement support for using environment variables for config options like
useEdgeCase
,cdnConfig
, andcacheDir
. -
Caching in Serverless:
The default file-system caching is not ideal for serverless environments due to ephemeral storage. A more robust hybrid solution (similar to @next-boost/hybrid-disk-cache) would be needed for production-level serverless deployments. -
Custom Cache Adapter:
Future improvements include allowing users to easily plug in external storage (like S3, Cloudflare KV, or Redis) for caching. -
Auto-Generated Blur Placeholder:
Currently, the library accepts a placeholder URL. Future versions should auto-generate a blurred placeholder (e.g., a base64 data URL). -
Extended CDN Integration:
Further enhancements may include better integration with edge networks and more configurable cache headers. - Testing and Edge Cases: I need to do further testing and identify bugs from edge cases before I can take the project out of beta.
-
Clone the Repository
Clone the repository to your local machine. -
Install Dependencies
Run:npm install
-
Build the Library
Compile the TypeScript:npm run build
-
Link Locally
Usenpm link
to link the library into your local Remix project:cd remix-img npm link # In your Remix project: npm link remix-img
-
Run Tests
Execute:npm test
Contributions are welcome! To contribute:
- Fork the repository.
- Create a feature branch:
git checkout -b feature/your-feature
. - Commit your changes.
- Push to your fork and open a Pull Request.
Please ensure your code adheres to best practices and includes tests for new features or bug fixes.
This project is licensed under the MIT License.
- Inspired by Next.js’s
<Image />
component. - Uses Sharp for efficient image processing.