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

support to crop/resize images in build-time #522

Open
Tracked by #197
losnappas opened this issue Oct 7, 2024 · 12 comments
Open
Tracked by #197

support to crop/resize images in build-time #522

losnappas opened this issue Oct 7, 2024 · 12 comments
Labels
enhancement New feature or request

Comments

@losnappas
Copy link

There is a cool feature in Next.js, basically makes images responsive by putting in a srcset etc. attributes. Nextjs had a separate image compression "server endpoint" last I checked, however personally I think creating a couple of resized images for the srcset, during build, is good enough.

Any thoughts on supporting such a feature?

@aralroca aralroca mentioned this issue Oct 7, 2024
26 tasks
@aralroca
Copy link
Collaborator

aralroca commented Oct 7, 2024

Thank you, @losnappas, for the proposal. I have added it to the 1.0 route map, and we will discuss its implementation here.

instead of making a Server Component (in Brisa <Image /> is a Server Component), I would make something that can also be used inside the Web Components without using slots/children.

If all we want is the crop/resize action, we could support it with:

<img src="..." width="500" height="500" cropImage />

Since the srcset already exists with the native element, we can add this attribute extension so that at build time, a Bun macro will do this job. (similar than renderOn="build").

The name of this attribute, or a better way to do it, remains open. But it would prevent it from being a Server Component for use anywhere.

@aralroca aralroca added the enhancement New feature or request label Oct 7, 2024
@aralroca aralroca changed the title <Image /> component or image compression pipeline during build support to crop images in build-time Oct 7, 2024
@aralroca
Copy link
Collaborator

aralroca commented Oct 7, 2024

surely as a result we could put the suffix of the hash of the image in case an image is cut in different places.

/my-image -> /my-image-empdnk6q.jpeg

Or add the size:

/my-image -> /my-image-500x500.jpeg

@AlbertSabate
Copy link
Collaborator

This would be a very nice to have yeah. It would be nice if we can choose between:

  • cropImage or resizeImage, if we crop de image, what would it be the strategy?
  • Define a quality
  • If jpeg or png, generate a webp ??
  • check for srcset and generate the images from src if they are not present (if present use the defined ones)
  • For dynamic images, give the option of use a cloud provider image optimisation or external cdn. So, ignore whatever is an external link.

Thinking further, maybe this would be a nice idea for a Brisa plugin or part of a cloud adapter, instead of having it on Brisa core ??

@aralroca aralroca changed the title support to crop images in build-time support to crop/resize images in build-time Oct 7, 2024
@aralroca
Copy link
Collaborator

aralroca commented Oct 7, 2024

  • cropImage or resizeImage, if we crop de image, what would it be the strategy?

maybe something like that?

<img 
  src="..." 
  width={500} 
  height={500} 
  optimize={{ 
    strategy: 'crop',  // Strategy can be 'crop', 'resize' or 'none'
    quality: 50,       // Compression quality (1-100)
    ext: 'avif'        // Output format, e.g., 'avif', 'webp', 'jpeg', 'png'
  }} 
/>

Although we should do that we can do in all operating systems. It is necessary to test the Bun tool to run C in runtime, this way it could be put in a macro to use it in build-time, and use C without pre-compiled and it adapts according to the OS just at the moment. It is necessary to test it...

Thinking further, maybe this would be a nice idea for a Brisa plugin or part of a cloud adapter, instead of having it on Brisa core ??

The good thing about doing it in build-time (in prod only) is that it would work on all Cloud Providers.

@AlbertSabate
Copy link
Collaborator

Sounds good. Missing width || height || srcset. srcset generate defined sizes.
props.optimize === undefined => no optimisations by Brisa ??

@aralroca
Copy link
Collaborator

aralroca commented Oct 7, 2024

Exactly. However, we need to take into account how to handle these cases:

<picture>
  <source media="(prefers-color-scheme: dark)" srcSet={dark} />
  <source media="(prefers-color-scheme: light)" srcSet={light} />
  <img {...imgProps} />
</picture>

Apart from the picture and source that should also be managed, only if the src or srcSet is dynamic, it will be difficult to manage it in this way that we have proposed in build-time. Perhaps it is better to look for an alternative. I don't know how <Image /> of Next.js is managed when the src is a dynamic value and not static, it would be necessary to look at how it is done to inspire us, or to look for a better way.

@AlbertSabate
Copy link
Collaborator

NextJS for dynamic src is using Vercel image optimization https://vercel.com/docs/image-optimization which relies on the cloud provider. In my opinion Brisa should handle the statics, and enable to integrate cloud optimizers somehow. Being up to the user how to use it. Some devs may want to use a self hosted image service also...

@aralroca
Copy link
Collaborator

aralroca commented Oct 7, 2024

Ok, now I understand what you meant before.

Of course, here each Cloud Provider will have its own way of managing it: Cloudflare, Netlify, Vercel, etc...

For Brisa's part, using a Proxy to handle this puts too much load on the CPU. Besides that doing a resize in build-time makes sense to load the images faster, but if it is done in a runtime with a proxy I don't see the benefit, since it will take several milliseconds to do this optimization. That said, I would avoid having a proxy, as the images are always already in /public after the build.

@aralroca
Copy link
Collaborator

@JLarky
Copy link

JLarky commented Dec 29, 2024

Another option would be to make unpic integration for Brisa https://github.com/ascorbic/unpic-img ("un" stands for universal)

@aralroca
Copy link
Collaborator

aralroca commented Dec 29, 2024

Another option would be to make unpic integration for Brisa https://github.com/ascorbic/unpic-img ("un" stands for universal)

I like the idea, it looks very simple:

import { transformProps, type UnpicImageProps } from "@unpic/core";

export type ImageProps = UnpicImageProps<JSX.HTMLAttributes<HTMLImageElement>>;

export function Image(props: ImageProps) {
  return <img {...transformProps(props)} />;
}

This is currently working as Server Component to optimize CDN images, ex of usage:

import { Image } from "@/components/image";

export default function Homepage() {
  return (
    <>
      <Image
        src="https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&auto=format&fit=crop&w=900&q=60"
        layout="fullWidth"
        alt=""
        priority
        height={400}
        crossOrigin="anonymous"
        background="auto"
      ></Image>
      <Image
        src="https://cdn.shopify.com/static/sample-images/garnished.jpeg"
        layout="constrained"
        width={800}
        height={600}
        alt="Shopify"
      ></Image>
      <Image
        src="https://bunnyoptimizerdemo.b-cdn.net/bunny7.jpg?width=300"
        width={400}
        height={300}
        layout="fixed"
        alt="Bunny.net"
      ></Image>
    </>
  );
}

Result HTML:

<img alt="" crossorigin="anonymous" loading="eager" fetchpriority="high" role="presentation" style="object-fit:cover;background-image:url(https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=24&amp;q=60);background-size:cover;background-repeat:no-repeat;width:100%;height:400px;" srcset="https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=640&amp;q=60 640w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=750&amp;q=60 750w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=828&amp;q=60 828w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=960&amp;q=60 960w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=1080&amp;q=60 1080w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=1280&amp;q=60 1280w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=1668&amp;q=60 1668w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=1920&amp;q=60 1920w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=2048&amp;q=60 2048w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=2560&amp;q=60 2560w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=3200&amp;q=60 3200w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=3840&amp;q=60 3840w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=4480&amp;q=60 4480w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=5120&amp;q=60 5120w,
https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;w=6016&amp;q=60 6016w" src="https://images.unsplash.com/photo-1617718295766-0f839c2853e7?ixlib=rb-4.0.3&amp;ixid=MnwxMjA3fDB8MHxzZWFyY2h8MTczfHxyYWluZm9yZXN0JTIwYmVhY2h8ZW58MHx8MHx8&amp;auto=format&amp;fit=crop&amp;q=60&amp;h=400">

<img alt="Shopify" loading="lazy" decoding="async" sizes="(min-width: 800px) 800px, 100vw" style="object-fit:cover;max-width:800px;max-height:600px;aspect-ratio:1.3333333333333333;width:100%;" srcset="https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=640&amp;height=480 640w,
https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=750&amp;height=563 750w,
https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=800&amp;height=600 800w,
https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=828&amp;height=621 828w,
https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=960&amp;height=720 960w,
https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=1080&amp;height=810 1080w,
https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=1280&amp;height=960 1280w,
https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=1600&amp;height=1200 1600w" src="https://cdn.shopify.com/static/sample-images/garnished.jpeg?width=800&amp;height=600">

<img alt="Bunny.net" loading="lazy" decoding="async" sizes="400px" style="object-fit:cover;width:400px;height:300px;" srcset="https://bunnyoptimizerdemo.b-cdn.net/bunny7.jpg?width=400&amp;aspect_ratio=400%3A300 400w,
https://bunnyoptimizerdemo.b-cdn.net/bunny7.jpg?width=800&amp;aspect_ratio=800%3A600 800w" src="https://bunnyoptimizerdemo.b-cdn.net/bunny7.jpg?width=400&amp;aspect_ratio=400%3A300" width="400" height="300">

Related ascorbic/unpic-img#753

@losnappas @AlbertSabate meanwhile probably you can use this.

@aralroca
Copy link
Collaborator

aralroca commented Jan 1, 2025

Probably we can integrate it to work it correctly with assetPrefix

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

No branches or pull requests

4 participants