Bitmaps in SVG
The SVG backend provides support for bitmap images through <image>
elements. Unlike raster-based backends, SVG doesn't load actual pixel data. Instead, it creates image references that the browser loads and renders.
How SVG Handles Bitmaps
SVG handles bitmaps differently from other backends:
- No pixel manipulation: SVG creates
<image>
elements withhref
attributes pointing to image sources. - Browser-handled loading: the browser handles fetching images and respecting CORS policies.
- Vector context: bitmaps are embedded in a vector graphics context and can have SVG transformations and filters applied.
Loading Bitmap References
In the SVG backend, "loading" a bitmap creates an SvgImageRef
which stores the image URL and optional dimensions. The specifier type is String
(representing a URL or data URI).
// Load from URL
val imageRef = "https://example.com/image.png".loadBitmap[SvgImageRef]
// Load from data URI
val dataUri = "..."
val dataRef = dataUri.loadBitmap[SvgImageRef]
// Load from relative path
val localRef = "images/logo.svg".loadBitmap[SvgImageRef]
Note* that loadBitmap
returns an IO[SvgImageRef]
. Use .unsafeRunSync()
to extract the value in examples, or handle the IO
properly in production code.
Specifying Dimensions
You can optionally specify dimensions when creating image references:
// With both width and height
val withDims = SvgLoadBitmap.withDimensions("image.jpg", width = 200, height = 150)
// With width only (height auto-calculated by browser)
val withWidth = SvgLoadBitmap.withWidth("image.png", width = 300)
// With height only (width auto-calculated by browser)
val withHeight = SvgLoadBitmap.withHeight("image.gif", height = 250)
If dimensions are not specified, the SVG backend uses a default size of 100x100 pixels for bounding box calculations.
Converting to Pictures
An SvgImageRef
can be converted to a Picture
using the toPicture
method:
val imageRefIO = "photo.jpg".loadBitmap[SvgImageRef]
val imageRef = imageRefIO.unsafeRunSync()
val picture = imageRef.toPicture
// Or load and convert in one step
val directPictureIO = "photo.jpg".loadBitmap[SvgImageRef].toPicture
val directPicture = directPictureIO.unsafeRunSync()
Composing with Vector Graphics
Since bitmap references become regular Pictures
, they can be composed with vector graphics:
val logoIO = for {
imageRef <- "logo.png".loadBitmap[SvgImageRef]
logo = imageRef.toPicture
} yield logo
.on(circle(150).fillColor(Color.lightBlue))
.beside(text("Welcome").fillColor(Color.black))
val composition = logoIO.unsafeRunSync()
SVG Filters on Bitmaps
SVG filters can be applied to bitmap images just like any other Picture
:
val filteredIO = for {
imageRef <- "photo.jpg".loadBitmap[SvgImageRef]
image = imageRef.toPicture
} yield image
.blur(3.0)
.dropShadow(4, 4, 2, Color.black.alpha(0.5.normalized))
val filtered = filteredIO.unsafeRunSync()
Complete Example
This example demonstrates loading an image, applying transformations, and composing with vector graphics:
val program = for {
// Load the image reference
imageRef <- "https://example.com/logo.png".loadBitmap[SvgImageRef]
// Convert to Picture
logo = imageRef.toPicture
// Create composition
composition = logo
.scale(0.75, 0.75)
.on(square(200).fillColor(Color.lightGray))
.above(text("Company Name").fillColor(Color.darkGray))
} yield composition
// In production, handle IO properly
// For examples, we can use unsafeRunSync
val result = program.unsafeRunSync()
Important Considerations
CORS Policies
When loading images from external domains, be aware of CORS (Cross-Origin Resource Sharing) policies.
No Pixel Access
Unlike Java2D or Canvas backends, SVG cannot:
- Access individual pixels.
- Convert Pictures to bitmap data.
- Apply pixel-level manipulations.
For these operations, use a raster-based backend.