How to compress image before uploading to Google Cloud Storage/S3?

I want to have my image compressed before they are uploaded to Google Cloud Storage or S3. How can I do that?

You could perhaps use something like the following in a "Run JS Code" query to rescale the image before uploading it:

function getImage(dataUrl)
{
return new Promise((resolve, reject) => {
const image = new Image();
image.src = dataUrl;
image.onload = () => {
resolve(image);
};
image.onerror = (el, err) => {
reject(err.error);
};
});
}

async function downscaleImage(
dataUrl,
imageType, // e.g. 'image/jpeg'
resolution, // max width/height in pixels
quality // e.g. 0.9 = 90% quality
) {

// Create a temporary image so that we can compute the height of the image.
const image = await getImage(dataUrl);
const oldWidth = image.naturalWidth;
const oldHeight = image.naturalHeight;
console.log('dims', oldWidth, oldHeight);

const longestDimension = oldWidth > oldHeight ? 'width' : 'height';
const currentRes = longestDimension == 'width' ? oldWidth : oldHeight;
console.log('longest dim', longestDimension, currentRes);

if (currentRes > resolution) {
console.log('need to resize...');

// Calculate new dimensions
const newSize = longestDimension == 'width' ? Math.floor(oldHeight / oldWidth * resolution) : Math.floor(oldWidth / oldHeight * resolution);
const newWidth = longestDimension == 'width' ? resolution : newSize;
const newHeight = longestDimension == 'height' ? resolution : newSize;
console.log('new width / height', newWidth, newHeight);

// Create a temporary canvas to draw the downscaled image on.
const canvas = document.createElement('canvas');
canvas.width = newWidth;
canvas.height = newHeight;

// Draw the downscaled image on the canvas and return the new data URL.
const ctx = canvas.getContext('2d');
ctx.drawImage(image, 0, 0, newWidth, newHeight);
const newDataUrl = canvas.toDataURL(imageType, quality);
return newDataUrl;
}
else {
return dataUrl;
}

}

let dataUrl = 'data:'+fileButton1.files[0].type+';base64,'+fileButton1.value[0];
let imageType = fileButton1.files[0].type
let resolution = 200
let quality = .8


let ds = downscaleImage(
dataUrl,
imageType, // e.g. 'image/jpeg'
resolution, // max width/height in pixels
quality // e.g. 0.9 = 90% quality
)

return ds

I also found the following library that seems to work in Retool, which could work for you as well: https://cdnjs.cloudflare.com/ajax/libs/jic/2.0.2/JIC.min.js

2 Likes

How to use this ds data to rest form/data upload api?

Hey @rony! If you're returning ds from a JS query, you can access the return from the query like {{query_name.data}} pretty much anywhere in Retool, including in a REST query.

How you set up that REST query depends on your endpoint setup and the action you're trying to perform.

Let me know if I can help answer any other questions :blush:

@victoria @mark How would I modify this to output the result in binary?

1 Like

Hey @nroeder! What format are you starting with?

Two scenarios, using the camera component in the mobile app, which gives a png, second is a file input which could be png, jpg, etc.

I can upload directly to azure using the {{fileInput1.value['0']}} body binary.

however if I use the compression JavaScript above I get a base64 output and when I upload the file and try to open I get a cannot ready file.

This is the output from the compression

""

Ah, got it! It looks like it's returning some additional info and the base64, so we can do something like this to just return the base64 (you can put the code directly in your image source field)

your_output_string.split("base64,")[1]

Let me know if this works for you!

it works perfectly for jpeg but when I am using png instead of decreasing the size its actually increases the size

Hi,

I cannot get this code to work. The js is still 'thinking' five minutes after providing a jpg via the fileButton. I have set the parameters as follows:

async function downscaleImage(
dataUrl,
imageType = "image/jpeg", // e.g. 'image/jpeg'
resolution = 600/800, // max width/height in pixels
quality = 0.4 // e.g. 0.9 = 90% quality
)

let ds = downscaleImage(
dataUrl,
imageType = "image/jpeg", // e.g. 'image/jpeg'
resolution = 600/800, // max width/height in pixels
quality = 0.4 // e.g. 0.9 = 90% quality
)

It doesn't throw an error, the little blue wheel just keeps a-turnin'.

Hi @patient_but_frustrated, are you still working on this? I noticed the original solution is from a few years back so it's possible things have been updated since then. Would you mind walking me through the problem and what you're goal end result is?

Hi I'm a different person but I'm trying to compress/resize images in Retool and am struggling with the above code as well. The code above just pinwheels for me.

I have imported a library that compresses photos but it requires a File or Blob object which I can't seem to get from the upload component. Can you help with this?

Hi @eslinz,

Happy to help. What library are you using to compress images?