Simple. Fast. Customizable. Upload images from your web app directly to Amazon S3.
The Shubox mission is to take the tedium and boilerplate out of the web-based file storage and image manipulation story. After creating your Shubox account, and setting our js library up on your website(s), you will be able to:
- Upload to S3, "the internet's ftp server", directly from the web browser.
- Manipulate and transform images and videos after they are uploaded - cropping, optimizing, changing formats.
- Using front-end technologies you are comfortable with, Javascript and CSS, you can create the user experience YOU want using our Javascript library. We have some demos at Codepen!
Quite frankly I was tired of setting up the file attachment library du jour that was uploading files to an app server then to S3. It was slow, it hogged compute resources, it was repetitive and rote. In addition to that, the solutions that existed that did this sort of thing were ugly, hamfisted, and didn't provide for ways to customize the user experience that I wanted. I built Shubox so that I had some way to quickly and elegantly build the file upload processes that lived in my mind's eye.
Visit shubox.io to learn more. Contact the Shubox team directly via email, or the "Send a Message" chat box on bottom right of the Shubox website (thanks smallchat!).
Follow @shuboxio on Twitter for important announcements, or the occasional pithy tweet.
- Installation
- Set Up Your Own S3 Bucket
- Examples & Ideas
- Library Documentation
- Development Notes
- Code of Conduct
To get things working as fast as possible we'll operate under the understanding that we'll use a sandbox bucket that we've already set up. It's an ephemeral S3 bucket with limitations that you can use to upload and test against.
There's a lot under the hood (bucket setup, signatures, CORS policies) that needs to happen in order to get things working just right. We handle all of that stuff via the Shubox service.
- Sign up for an account at Shubox
- Obtain your "Sandbox" key from the Shubox dashboard.. You'll need this for your Javascript code when you initialize your first Shubox object.
You can use npm or yarn to install the library from npm:
$ npm install shubox
$ yarn add shubox
Require the Shubox library in your JS, via ES module:
import Shubox from 'shubox'
Download and embed the shubox js file directly in your html
$ curl -L https://unpkg.com/shubox -o public/shubox.umd.js
In your HTML:
<script src="shubox.umd.js"></script>
For this (very contrived) example let's say you want your users to upload an
avatar or profile photo. You have an HTML element with the ID "#avatar"
.
And your provided sandbox key is "abcde-qwerty-12345
.
new Shubox('#avatar', { key: "abcde-qwerty-12345" })
That's it! When you click that element you will see a file dialog pop up where you can select your image. Once that image is selected it will be uploaded to the sandbox S3 bucket. Your code works! Sure, it uploads to a temporary S3 bucket, but the code works! (More info soon on how to set up your own bucket)
Instructions coming soon!
Between the common and not-so-common use-cases, we've gathered some up and will be adding to them down below for the times you might be looking for a quick copypasta, or ideas as to what you could do with the Shubox library.
NOTE: All of the following are in ES6 syntax and assume that you have already installed the shubox script into your JS bundle, or have embedded and executed the standalone shubox.js script.
<div id="avatar"></div>
const shuboxKey = "[copied from Shubox dashboard]"
const avatar = new Shubox('#avatar', {
key: window.shuboxKey,
// prevents from inserting the base64 preview image
previewsContainer: false,
success: function(file) { // passed a ShuboxFile object
// create new image element
let img = new Image()
// once loaded, insert it inside div#avatar
img.onload = function() {
let el = document.getElementById('avatar')
el.insertAdjacentHTML('beforeend', '<img src="' + file.s3url + '">')
}
// assign the src and let it load
img.src = file.s3url
}
})
<div class="multiple-files-container">
<div class="shubox--upload-target" id="shubox--multiple-files"></div>
<div class="shubox--upload-preview" id="shubox--multiple-files-preview"></div>
</div>
.multiple-files-container {
padding: 40px;
width: 500px;
text-align: left;
margin: 0 auto;
}
.shubox--upload-target:hover {
border-color: silver;
}
.shubox--upload-target:before {
content: 'Drag && Drop || Click';
font-size: .6rem;
color: silver;
display: block;
font-family: courier;
margin-top: .4rem;
line-height: 1rem;
}
.shubox--upload-preview {
overflow: hidden;
}
.shubox--upload-preview > div {
display: inline-block;
position: relative;
width: 4rem;
height: 4rem;
margin-right: 1rem;
overflow: hidden;
}
.dz-progress {
display: block;
position: absolute;
bottom: 5px;
left: 5px;
right: 5px;
}
.dz-upload {
border-radius: 3px;
height: 5px;
background: #bce29e;
display: block;
clear: both;
}
.dz-upload[style="width: 100%;"] { display: none; }
.dz-error-message,
.dz-filename,
.dz-size,
.dz-success-mark,
.dz-error-mark { display: none; }
.dz-complete.dz-success .dz-success-mark,
.dz-complete.dz-error .dz-error-mark {
display: block;
position: absolute;
right: 5px;
bottom: 0px;
text-shadow: 1px 1px white;
}
const shuboxKey = "[copied from Shubox dashboard]"
const multfiles = new Shubox("#shubox--multiple-files", {
key: window.shuboxKey,
previewsContainer: '#shubox--multiple-files-preview',
})
"Transforms" are variants of uploaded images that you define as an option in
the Shubox
initializer. For example: you may want a 100x100
sized image
generated after an 800x600 photo is uploaded, or an image to be converted to an
avif
.
In the JS library you can define a corresponding callback that will fire once
that version of the image is generated, and HTTP request and response
successfully executed. For example, if you define a "144x144#"
transform,
that will intelligently resize and crop all uploaded images to that exact pixel
size -- 144 pixels wide by 144 pixels tall. To run a callback once that image
exists, the following options will add an image tag with that version of the
image's URL.
<div id="avatar"></div>
const shuboxKey = "[copied from Shubox dashboard]"
const avatar = new Shubox('#avatar', {
key: window.shuboxKey,
previewsContainer: false,
// image transforms represented by an array of ImageMagick Geometries
// see: https://legacy.imagemagick.org/Usage/resize/#resize
transforms: {
// resize and crop to 144x144
'144x144#': function(shuboxFile) {
// once image is found, insert an `img`
// tag with that url as the src
let el = document.getElementById('avatar')
el.insertAdjacentHTML(
'beforeend',
`<img src='${shuboxFile.transform.s3url}'>`
)
}
}
})
<textarea placeholder="Leave a comment or drag and drop some images."
class="shubox--textarea"
id="shubox--textarea"></textarea>
<div id="shubox--click-to-upload" class="shubox--click-to-upload">
Attach files by dragging & dropping, <strong>selecting them</strong>,
or pasting from the clipboard.
</div>
const shuboxKey = "[copied from Shubox dashboard]"
const githubForm = new Shubox('#shubox--textarea', {
key: window.shuboxKey,
// clicking on the element corresponding to the `clickable` selector
// will trigger the file dialog
clickable: '#shubox--click-to-upload',
// Once the file starts uploading the string in `uploadingTemplate` will be
// interpolated with the file's name and added to the textarea
uploadingTemplate: '![Uploading {{name}}...]()',
// Once the file completes uploading the string in `successTemplate` will be
// interpolated with the file's name and S3 url, then placed in the textarea
successTemplate: '![{{name}}]({{s3url}})'
})
<textarea placeholder="Leave a comment or drag and drop some images."
class="shubox--textarea shubox--textarea--no-click-bar"
id="shubox--textarea--cursor">Click to place cursor and upload.</textarea>
const shuboxKey = "[copied from Shubox dashboard]"
const atCursor = new Shubox('#shubox--textarea--cursor', {
key: window.shuboxKey,
// when inserting text into an input or textarea the `insertAtCursor` value
// tells shubox to add the S3 url wherever the cursor is currently placed
textBehavior: 'insertAtCursor',
// the text inserted into the form element is the final S3 url with a space
// before and after
successTemplate: ' {{s3url}} '
})
<textarea
tabindex="1"
placeholder="Leave a comment or drag and drop some images."
class="shubox--textarea shubox--textarea--no-click-bar"
id="shubox--textarea--replace">Drag & drop to replace this text</textarea>
const replace = new Shubox('#shubox--textarea--replace', {
key: window.shuboxSandboxKey,
s3urlTemplate: 'Replaced with: {{s3url}} ',
textBehavior: 'replace',
})
<textarea
tabindex="1"
placeholder="Leave a comment or drag and drop some images."
class="shubox--textarea shubox--textarea--no-click-bar"
id="shubox--textarea--append">
Dragging & dropping here will append after 👉
</textarea>
const append = new Shubox('#shubox--textarea--append', {
key: window.shuboxSandboxKey,
successTemplate: ' See? Told you. Right after --> {{s3url}}',
textBehavior: 'append',
})
<div id="webcam-photo" class="webcam"></div>
const webcamPhoto = new Shubox('#webcam-photo', {
key: window.shuboxSandboxKey,
webcam: 'photo',
success: function(file) {
console.log(`File ${file.name} successfully uploaded!`)
console.log(file.s3url)
},
})
<div id="webcam-with-options" class="webcam"></div>
<ul>
<li><a href="#" id="webcam-start">Start Camera đź“·</a></li>
<li><a href="#" id="webcam-stop">Stop Camera đźš«</a></li>
<li><a href="#" id="webcam-capture">Take Photo ✨</a></li>
</ul>
const webcamOptions = new Shubox('#webcam-with-options', {
key: window.shuboxSandboxKey,
webcam: {
type: 'photo',
startCamera: '#webcam-start',
stopCamera: '#webcam-stop',
takePhoto: '#webcam-capture'
},
success: function(file) {
console.log(`File ${file.name} successfully uploaded!`)
console.log(file.s3url)
},
})
<div id="webcam-video" class="webcam"></div>
const video = new Shubox('#webcam-video', {
key: window.shuboxSandboxKey,
webcam: 'video',
success: function(file) {
console.log(`File ${file.name} successfully uploaded!`)
console.log(file.s3url)
},
})
<div id="webcam-video" class="webcam"></div>
<ul>
<li><a href="#" id="video-start">Start Camera đź“ą</a></li>
<li><a href="#" id="video-stop">Stop Camera đźš«</a></li>
<li><a href="#" id="video-record-start">Start Recording đź”´</a></li>
<li><a href="#" id="video-record-stop">Stop Recording ⏹ </a></li>
<li><select class="shubox-audioinput">
<option>Select Mic</option>
</select></li>
<li><select class="shubox-videoinput">
<option>Select Camera</option>
</select></li>
</ul>
const videoWithOptions = new Shubox('#webcam-video', {
key: window.shuboxSandboxKey,
webcam: {
type: 'video',
startCamera: '#video-start',
stopCamera: '#video-stop',
startRecording: '#video-record-start',
stopRecording: '#video-record-stop',
audioInput: '.shubox-audioinput',
videoInput: '.shubox-videoinput',
cameraStarted: (_webcam) => { console.log("camera started") },
cameraStopped: (_webcam) => { console.log("camera stopped") },
recordingStarted: (_webcam) => { console.log("recording started") },
recordingStopped: (_webcam, _file) => { console.log("recording stopped") }
},
success: file => {
console.log(`File ${file.name} successfully uploaded!`)
console.log(file.s3url)
},
})
There are some cases where you might have a file object programmatically
generated for you and all you want to do is upload it to S3 yourself. Using any
combination of the options we provide to you, you can accomplish this by
explicitly calling upload
on the instantiated Shubox object. Example:
const shuboxObject = new Shubox('#element', { key: window.shuboxSandboxKey })
const file = new File(["contents"], "file.png", { type: "image/png" })
shuboxObject.upload(file)
The following section outlines what the Shubox specific options are for the Shubox object instantiation. You might ask - "Isn't it all Shubox specific?" Well, the answer is "No" because under the hood is a wonderful little library called Dropzone.js. Shubox does a tremendous amount of heavy lifting in addition to, instead of, on top of, Dropzone.js. You can pass any Dropzone.js options to the Shubox instantiation and they will be passed along into Dropzone.js in addition to Shubox.
Without further ado, here's what you get out of the box (no pun intended)
with Shubox.
These are all piggybacking on the Dropzone events and are also documented there but should be called out here nevertheless. These are the big-ticket events that would be most often used during the lifecycle of uploaded files.
The sending
callback is called immediately before each file is sent. It
receives file
[1], xhr
, and formData
objects. This
allows you to modify any or all of these object before they go in flight to
your endpoint (to add headers, etc).
sending: function(file, xhr, formData) {}
Assign a function to the success key that accepts a file
parameter which will
be run after files are successfully uploaded. More information about the File
type passed into this function can be found below.
success: function(file) {}
Assign a function to the error key, accepting a file object and error string. This method will be called when errors are incurred with a file upload, or during the S3 signature generation process
error: function(file, message) {}
The queuecomplete callback will be called when all files are finished uploading.
queuecomplete: function() {}
Upon successful upload of an image the Shubox library will pass a file object to all JavaScript callbacks. The format of this file object follows:
{
accepted: true,
custom_status: "ready",
name: "my-upload.jpg", // filename w/ext
width: 281, // in pixels
height: 500, // in pixels
size: 15596, // in bytes
lastModified: 1446064073000,
lastModifiedDate: Sun Jan 1 2016 00:00:01 ..., // Date Object
postData: {
AWSAccessKeyId: "...", // AWS Key
Content-Type: "",
acl: "public-read",
key: "path/to/file/in/bucket/filename.jpg",
policy: "...", // policy string
signature: "...", // signature string
success_action_status: "201" // HTTP response code
},
processing: true,
s3: "path/to/file/in/bucket/filename.jpg",
s3Url: "https://bucket.s3.amazonaws.com/path/to/file...",
transforms: {
"variantName": {
s3Url: "https://s3.amazonaws.com/bucket/variantName_file.jpg"
},
// ...
},
status: "success",
type: "image/jpeg",
upload: {
bytesSent: 999,
total: 999,
progress: 100
}
}
CDN's are ubiquitous and almost a requirement these days. To that end, putting
newly uploaded images behind a CDN, or a hostname that lives between a web
browser and the S3 bucket, instead of linking directly to the S3 object is
something you can with the cdn
option.
cdn: 'https://cdn.example.com' // will replace "https://s3.amazonaws.com/bucketname"
(NOTE: the s3Key
option is not permitted when using the
sandbox)
Do you want any/all files uploaded through one of your Shubox uploaders to have
an exact S3 key? The default behavior is for Shubox to send files up to your
bucket with the key /[random string]/filename.ext
so that you will not
overwrite previously uploaded files.
Setting the s3Key would be useful if you know that you are not risking an
overwrite unless you deliberately mean to. For example, you're logged in as
"Sam" and the Shubox uploader for your avatar
shuttles all images up to
/users/avatars/sam.jpg
. Similarly, if you have a resource/record that needs a
single photo associated with it, like
/dealership/:dealership_id/cars/:id/photo.jpg
.
If you are looking to customize part of the s3Key, but want to use some values that are determined at the time a file is uploaded -- like the filename -- then you can interpolate some values with representations in the string like:
{{domainSlug}}
- the domain slug{{uuid}}
- uuid (the random generated string){{filename}}
- file name (duh){{fileExtension}}
- file extension (the "jpg" part of "hello.jpg"){{baseFilename}}
- base filename (the "hello" part of "hello.jpg")
s3Key: '/[random letters and numbers]/filename.extension' // default
s3Key: '/users/avatars/sam.jpg'
s3Key: '/client-name/project-name/{{filename}}'
Warning
transformKey
and transformCallbacks
are deprecated and will be removed
from the library in version 1.2
. They have been replaced by the
transforms
option, which acts as a combination/unification of the two.
To view previous documentation around these two options visit a previous version of this README.
transforms: null // default
transforms: { string: function }
Assigning an object/hash to transforms
with keys that contain a ImageMagick
compatible geometry string, and a default of null
will tell Shubox to
generate a variant of that image. For example, the following will tell Shubox
to resize and generate an image with the width of 200px, while maintaining the
aspect ratio with a variable height.
transforms: {
'200x': ()=>{}
}
While the following will resize and crop (to fit) a 300x300 image, and
convert it to the webp
format. Additionally, it will execute the provided
callback function once that file's URL returns a successful HTTP HEAD request.
transforms: {
'300x300#.webp': function(shuboxFile) {
// once image is found, insert an `img`
// tag with that url as the src
let el = document.getElementById('shuboxElement')
el.insertAdjacentHTML(
'beforeend',
`<img src='${shuboxFile.transform.s3url}'>`
)
}
}
Image resize and file format variants
Here are the variants that you can watch for.
The file formats you can convert to:
'.webp'
'.webm'
'.mp4'
'.jp2'
The imagemagick geometries/sizes (more here):
'200'
, width - Width given, height automagically selected to preserve aspect ratio.'x200'
, xheight - Height given, width automagically selected to preserve aspect ratio.'200x200'
, widthxheight - Maximum values of height and width given, aspect ratio preserved.'200x200^'
, widthxheight^ - Minimum values of width and height given, aspect ratio preserved.'200x200!'
, widthxheight! - Width and height given, original aspect ratio ignored.'200x200#'
, widthxheight# Width and height explicitly given, cropped to fill.
The extracted animated GIF frame:
'frame'
transformCallbacks: null // default
transformCallbacks: {
// for the 200x200 image, trigger this
// method when it's available
'200x200': function(shuboxFile) {
// the `shuboxFile` has a property, `transforms`
// that is a hash with the versions of the file
// you are waiting for.
console.log(shuboxFile.transforms["200x200"].s3url);
},
// if you ask to convert images to WEBP format, this
// will run when it is available
'.webp': function(shuboxFile) {
console.log("a JPG converted to WEBP");
console.log(shuboxFile.transforms[".webp"].s3url);
},
// ... or combine the two
'200x200.webp': function(shuboxFile) {
console.log("a JPG resized AND converted to WEBP");
console.log(shuboxFile.transforms["200x200.webp"].s3url);
}
}
NOTE:
- You can have more than one variant callback, like in the above example.
- You can combine a resized variant with a new file format variant (
200x200
+webp
). - The resulting variant images/files need to have publicly readable permissions.
- If there are many variations in your transform pipeline it may take a long time to get through them, and therefore a long time (or not at all) before your callback is triggered.
NOTE: Formerly s3urlTemplate
. Will be deprecated with version 1.0.
When uploading to a form element, a string containing the URL to your S3
resource will be placed in your form element's value. By default this is a
result of a template having placeholder values interpolated before being
placed in your form element. That template is merely a handlebars-like value of
'{{s3Url}}'
to start out with but has several more values that can be used.
This can be changed to any string. For example, to instead insert markdown's
image pseudo code you would change this to '![{{name}}]({{s3url}})'
.
Other placeholders you may use:
height
- image heightwidth
- image widthname
- filenames3
- path of s3 objects3url
- url to filesize
- file size in bytestype
- file type
s3urlTemplate: '{{s3Url}}' // just the url
s3urlTemplate: '![{{name}}]({{s3url}})' // markdown image tag
s3urlTemplate: '<img src="{{s3Url}} width="{{width}}" height="{{height}}">' // img tag
Similar to successTemplate
- this is a string that gets interpolated with the
following values, however this is added while a file is being uploaded.
height
- image heightwidth
- image widthname
- filenames3
- path of s3 objects3url
- url to filesize
- file size in bytestype
- file type
uploadingTemplate: 'uploading {{size}} byte file: {{name}}'
uploadingTemplate: '![Uploading {{name}}...]()' // "temp" markdown tag a-la GH
When uploading through a form element (<input>
, <textarea>
, etc) the
behavior, by default, will be to 'replace'
the contents of the form element
value with the URL to the newly uploaded file.
The value 'append'
will append the resulting value from the newly uploaded
file to the form element.
'insertAtCursor'
will, as I am sure you are shocked to hear, insert the text
wherever your cursor is placed at the time of the upload.
textBehavior: 'replace' // default value
textBehavior: 'append'
textBehavior: 'insertAtCursor'
Shubox provides a mechanism with which to post custom data via a webhook
to an address of your choosing whenever files are uploaded. This will
allow you to share any information available during your users' session. The
information within the extraParams
hash will be sent to your webhook
endpoint along with the data from the uploaded file.
As an example, you may want to send data about the uploaded file(s) with a user's ID or email. It's not uncommon to want to know who is uploading a particular file.
extraParams: {} // default
extraParams: { // override with whatever you want
userID: 123,
userEmail: '[email protected]',
reecesPiecesOrPBCups: 'cups, obviously'
}
If you only want certain file types allowed to be uploaded, you may provide a list of mime types or extensions. The contents of the option may include a comma separated list of mime types or file extensions. Eg.:
acceptedFiles: "image/*" // default value
acceptedFiles: "image/*,application/pdf,.psd" // image, pdfs, psd
The webcam option(s) for the library support two modes of capture: "photo"
,
and "video"
.
You can capture a photo or record a video via your webcam and have it sent
right up to S3. The most straightforward approach is to initialize Shubox by
pointing to an element (usually a div) sized to how large you would like the
video element to be. The webcam
key accepts a string "photo"
or "video"
,
or an object with a required key of type
, and optional keys that are detailed
below. To start:
// let shubox handle everything in the context of your target el
// for a photo, set the root-level key of "webcam" to "photo"
webcam: 'photo',
// for a video, set the root-level key of "webcam" to "video"
webcam: 'video',
The above will wire up the video elements to trigger the steps in the lifecyle
by clicking on the element. Eg: start camera -> take photo
, or
start camera -> start record -> stop record
.
The options you can pass to the nested hash in webcam:
are as follows:
webcam: {
type: 'photo',
startCamera: '#webcam-start', // selector to element to start camera
stopCamera: '#webcam-stop', // selector to element that stops camera
takePhoto: '#webcam-take', // selector to element that takes the photo
// startCapture: '#webcam-take', // deprecated. same as `takePhoto
// the template that gets inserted into the shubox element div. You may
// customize this but do make sure the video, canvas, and img are in there
photoTemplate: `
<video class="shubox-video" muted autoplay></video>
<canvas style="display: none"></canvas>
<img style="display: none">
`,
// selector to a <select> element that will be populated with all camera
// devices. When changed that will change the video input stream
videoInput: '#select-camera',
// Callbacks that get triggered during the photo-taking lifecycle. `webcam`
// is an instance of the Webcam class. `file` is of type `Blob`.
cameraStarted: (webcam) => ();
cameraStopped: (webcam) => ();
photoTaken: (webcam, file) => ();
}
The options you can pass to the nested hash in webcam:
are as follows:
webcam: {
type: 'video',
startCamera: '#webcam-start', // selector to element to start camera
stopCamera: '#webcam-stop', // selector to element that stops camera
startRecording: '#video-start', // selector to element that starts recording
stopRecording: '#video-stop', // selector to element that stops recording
timeLimit: 5, // default: undefined. video recording time limit
portraitMode: false, // default: undefined. set width and height to fit portrait mode
// the template that gets inserted into the shubox element div. You may
// customize this but do make sure the video is in there.
// the following is the default:
videoTemplate: `<video class="shubox-video" muted autoplay></video>`,
// selector to <select> elements that will be populated with all camera and
// and microphone devices. When changed, each will change the video or audio
// input stream
videoInput: '#select-camera',
audioInput: '#select-mic',
// Callbacks that get triggered during the photo-taking lifecycle. `webcam`
// is an instance of the Webcam class. `file` is of type `Blob`.
cameraStarted: (webcam) => ();
cameraStopped: (webcam) => ();
recordingStarted: (webcam) => ();
recordingStopped: (webcam, file) => ();
}
Clone this repo:
git clone https://github.com/shuboxio/shubox.js.git shubox.js
cd ./shubox.js
Set everything up
./bin/setup
This will install dependencies, build the demo html files, and copy local demo config files.
Grab your "Sandbox" key from the Shubox dashboard.
open http://dashboard.shubox.io/v2/sandbox.txt
Place your key into shubox_config.js as a global variable. This will allow
your local dev/example server to use your sandbox key. _This is done within
./bin/setup
but, just in case:
cp demo/javascript/shubox_config_sample.js \
demo/javascript/shubox_config.js
echo "var shuboxSandboxKey = '[SANDBOX KEY GOES HERE]';" > \
demo/javascript/shubox_config.js
Run local example server
npm run dev
# ... then open up http://localhost:3000/
We believe in safe, open, and inclusive environments to collaborate in. As such this project adheres to the Contributor Covenant code of conduct. By participating, you are expected to uphold this code. Please report unacceptable behavior to [email protected].