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

Add JavaScript tutorial to the docs #271

Merged
merged 54 commits into from
Sep 17, 2023
Merged
Show file tree
Hide file tree
Changes from 38 commits
Commits
Show all changes
54 commits
Select commit Hold shift + click to select a range
de482b9
Create javascript.md
perborgen Aug 31, 2023
908b35f
Add links to images
perborgen Aug 31, 2023
15ef3a2
Update image in JavaScript tutorial
perborgen Aug 31, 2023
ff1f150
Add JavaScript example to 3_examples.snippet
perborgen Aug 31, 2023
725471e
Cleanup formatting in JavaScript example in 3_examples.snippet
perborgen Aug 31, 2023
720fcea
Add JavaScript tutorial to _toctree.yml
perborgen Aug 31, 2023
9346982
Add JavaScript object detection example
perborgen Aug 31, 2023
e5ba807
Add scrim walk through to JavaScript tutorial
perborgen Aug 31, 2023
bfce054
Add semicolons to index.js
perborgen Aug 31, 2023
98810d3
Missing semicolon in index.js
perborgen Aug 31, 2023
c3d7167
Add semi colons to utils.js
perborgen Aug 31, 2023
9ffbabd
Fix indentation + title in index.html
perborgen Aug 31, 2023
f1ac272
Update javascript.md wrt tabs and jsx->js snippets
perborgen Aug 31, 2023
b202a14
Update docs/source/tutorials/javascript.md
perborgen Sep 10, 2023
eaccb2f
Update docs/snippets/3_examples.snippet
perborgen Sep 10, 2023
c348a1d
Update docs/source/tutorials/javascript.md
perborgen Sep 10, 2023
19ed8fd
Update examples/javascript-object-detection/index.js
perborgen Sep 10, 2023
c430bcb
formatting
xenova Sep 11, 2023
9515cda
Update title
xenova Sep 11, 2023
a288ebf
Update tutorial name
xenova Sep 11, 2023
8d4c2fc
Add comments
xenova Sep 11, 2023
98232aa
Move pipeline creation to outside `run` function
xenova Sep 11, 2023
c31fae8
Remove `removeElements` helper function
xenova Sep 11, 2023
7b718ad
Update index.html
xenova Sep 11, 2023
d8610aa
Cleanup
xenova Sep 11, 2023
c50de9d
Use anonymous function for loading file
xenova Sep 11, 2023
21756a3
Move upload-icon.png to HF Hub
xenova Sep 14, 2023
b2d5b4d
Fix typos
xenova Sep 14, 2023
729fafb
Cleanup
xenova Sep 14, 2023
f97f2f5
Update tutorial with refactored source code
perborgen Sep 15, 2023
31bd58c
Clean up code indentation in the tutorial
perborgen Sep 15, 2023
646f45d
Remove unused .upload-button class and move styles
perborgen Sep 15, 2023
6feca1a
Rename tutorial in sidebar
perborgen Sep 15, 2023
62a3739
Rename javascript.md to vanilla-js.md
perborgen Sep 15, 2023
48d9b1f
Rename javascript-object-detection to vanilla-js
perborgen Sep 15, 2023
ef04437
Delete examples/vanilla-js directory
perborgen Sep 15, 2023
1dcf1c9
Rename js example app
perborgen Sep 15, 2023
0ac6dbe
Fix wrong img link in vanilla-js.md
perborgen Sep 15, 2023
66916ae
Fix link to example
xenova Sep 15, 2023
b4813e7
Fix typo
xenova Sep 15, 2023
34a4909
Improve wording
xenova Sep 15, 2023
b468e24
Wording
xenova Sep 15, 2023
0f279f6
Update source code link
xenova Sep 15, 2023
a7bba83
Improve wording
xenova Sep 15, 2023
7e4a83a
Improvements
xenova Sep 15, 2023
d959d3d
Improve step 3
xenova Sep 16, 2023
8a7a99d
Change sentence about drag&drop html file
perborgen Sep 17, 2023
6a75c1f
Update vanilla js tutorial to mention server
perborgen Sep 17, 2023
c0f1fd1
Last bit of proof-reading + improvements
xenova Sep 17, 2023
49db2c4
Wording
xenova Sep 17, 2023
8625aaf
Link to new demo site
perborgen Sep 17, 2023
2df0457
Merge branch 'main' of https://github.com/perborgen/transformers.js i…
xenova Sep 17, 2023
02c8c55
Update vanilla JS example README
xenova Sep 17, 2023
58e84c6
Update demo app link
xenova Sep 17, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions docs/snippets/3_examples.snippet
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ Want to jump straight in? Get started with one of our sample applications/templa
| Doodle Dash | Real-time sketch-recognition game (see [blog](https://huggingface.co/blog/ml-web-games)) | [link](https://github.com/xenova/doodle-dash) |
| Code Playground | In-browser code completion website | [link](./examples/code-completion/) |
| Semantic Image Search | Search for images with text (Next.js + Supabase) | [link](./examples/semantic-image-search/) |
| Vanilla JavaScript | In-browser object detection | [link](./examples/vanilla-javascript/) |
| React | Multilingual translation website | [link](./examples/react-translator/) |
| Browser extension | Text classification extension | [link](./examples/extension/) |
| Electron | Text classification application | [link](./examples/electron/) |
Expand Down
2 changes: 2 additions & 0 deletions docs/source/_toctree.yml
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@
title: Custom usage
title: Get started
- sections:
- local: tutorials/vanilla-js
title: Building a Vanilla JS Application
- local: tutorials/react
title: Building a React Application
- local: tutorials/next
Expand Down
281 changes: 281 additions & 0 deletions docs/source/tutorials/vanilla-js.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,281 @@
# Building a Vanilla JavaScript Application

In this tutorial, you’ll build a simple JavaScript application that detects objects in images using Transformers.js. The app requires no server, external libraries, or build tools... all you need is a code editor and a browser!

Here's how it works: the user uploads an image to the browser via the “Upload image” button. The app then downloads an AI model, runs the image through it, and finally displays an overview of where the different objects are located, like this:

![Demo](https://huggingface.co/datasets/Xenova/transformers.js-docs/resolve/main/js-detection-interence-zebra.png)

**************************Useful links:**************************

- [Demo site](https://huggingface.co/spaces/Scrimba/javascript-object-detector)
- [Interactive code walk-through (scrim)](https://scrimba.com/scrim/cKm9bDAg)
- [Source code](https://huggingface.co/spaces/Scrimba/javascript-object-detector/tree/main)

## Step 1: HTML and CSS setup

The first thing we'll do is create some markup and styling. Create an in `index.html` file with a basic HTML skeleton, and add the following `<main>` tag to the `<body>`:

```html
<main class="container">
<label class="custom-file-upload">
<input id="file-upload" type="file" accept="image/*" />
<img class="upload-icon" src="https://huggingface.co/datasets/Xenova/transformers.js-docs/resolve/main/upload-icon.png" />
Upload image
</label>
<div id="image-container"></div>
<p id="status"></p>
</main>
```

Let’s break it down:

We’re adding an `<input>` element with `type="file"`. This gives the user the ability to pick an image from their local files. The default styling for this element looks pretty bad, so we are going to style it as a nice button instead. The easiest way to achieve this is to wrap the `<input>` element in a `<label>`, hide the input, and then style the label as a button.

We’re also adding an empty `<div>` container for displaying the image, plus an empty `<p>` tag that we'll use to give status updates to the user as we're downloading and running the model, as both of these operations take some time.

Next, add the following CSS rules in a `style.css` file and and link it to the HTML:

```css
html,
body {
font-family: Arial, Helvetica, sans-serif;
}

.container {
margin: 40px auto;
width: max(50vw, 400px);
display: flex;
flex-direction: column;
align-items: center;
}

.custom-file-upload {
display: flex;
align-items: center;
cursor: pointer;
gap: 10px;
border: 2px solid black;
padding: 8px 16px;
cursor: pointer;
border-radius: 6px;
}

#file-upload {
display: none;
}

.upload-icon {
width: 30px;
}

#image-container {
width: 100%;
margin-top: 20px;
position: relative;
}

#image-container>img {
width: 100%;
}
```

If you drag and drop the `index.html` file into a browser, you should see the following:
xenova marked this conversation as resolved.
Show resolved Hide resolved

![Demo](https://huggingface.co/datasets/Xenova/transformers.js-docs/resolve/main/js-detection-btn.png)
## Step 2: JavaScript setup

The next step is to link our HTML file to a script. Add the following tag at the end of the `<body>` in the HTML file:

```html
<script src="./index.js" type="module"></script>
```

The `type="module"` attribute is important, as it turns our file into a JavaScript module, meaning that we’ll be able to use imports and exports. This immediately comes in handy, as we’ll import the Transformers.js library from their CDN at the top of our `index.js` file:

```js
import { pipeline, env } from "https://cdn.jsdelivr.net/npm/@xenova/[email protected]";
```

Since we will download the model from the Hugging Face Hub, we skip checking if we have it available locally:

```js
env.allowLocalModels = false;
```

Next up, we’ll grab ahold of all the DOM elements we need to access via JavaScript:

```js
const fileUpload = document.getElementById("file-upload");
const imageContainer = document.getElementById("image-container");
const status = document.getElementById("status");
```

## Step 3: Download the model

We’re ready to download the model! As this can take some time, we first update the `status` paragraph so that the user knows that we’re about to run the model.

```js
status.textContent = "Loading model...";
```

This is important to do because the UI will freeze when we’re downloading the model, as JavaScript is single-threaded and can only do one thing at a time. If you want avoid the browser from freezing, you’ll need to use a web worker to download and run the model in the background.

However, we’re not going to do that in this tutorial, so let’s download the model by invoking the `pipeline()` function that we imported at the top of our file.

```js
const detector = await pipeline("object-detection", "Xenova/detr-resnet-50");
```

We’re passing two arguments into the `pipeline()` function. The first one tells Transformers.js what kind of task we want to perform. In our case, that is `object-detection`, but there are tons of other tasks you can perform as well. Here’s a few of them.

- `text-generation`
- `sentiment-analysis`
- `summarization`
- `automatic-speech-recognition`

…and many more. See a full overview [here.](https://huggingface.co/docs/transformers.js/pipelines#tasks)

The second argument in the `pipeline()` function specifies which underlying model we would like to use to solve the given task, as each task can usually be solved by many different models. We will use `'Xenova/detr-resnet-50'`, as it is a reasonable small (43MB) but powerful model for object detection in images.

After we’ve awaited this function to run we’ll tell the user that the app is ready to be used.

```js
status.textContent = "Ready";
```

## Step 4: Create the image uploader

The next step is to enable users to upload images. To achieve this, we will listen for "change" events from the `fileUpload` element. In the callback function, we first check that the user has actually uploaded a file, and returns the function if they haven’t.

If there’s actually an image there, we use a `FileReader()` to read the content of the image:

```js
fileUpload.addEventListener("change", function (e) {
const file = e.target.files[0];
if (!file) {
return;
}

const reader = new FileReader();

// Set up a callback when the file is loaded
reader.onload = function (e2) {
imageContainer.innerHTML = "";
const image = document.createElement("img");
image.src = e2.target.result;
imageContainer.appendChild(image);
// detect(image);
};
reader.readAsDataURL(file);
});
```

Once the image has been properly loaded into the browser, the `reader.onload` callback function will be invoked, which will give us access to all the necessary data about the image. In it, we append a new `<img>` element to the DOM so that we can display the image to the user.

Don’t worry about the `detect(image)` function call, which we’ve commented out and will explain later. Try to run the app and upload an image to the browser to make sure your code is working properly. You should see your image displayed underneath the button like this:

![Demo](https://huggingface.co/datasets/Xenova/transformers.js-docs/resolve/main/js-detection-btn-img.png)

## Step 5: Run the model

We’re finally ready to start interacting with Transformers.js! Let’s uncomment the `detect(image)` function call from the snippet above. Then we’ll define the function itself:

```js
async function detect(img) {
status.textContent = "Analysing...";
const output = await detector(img.src, {
threshold: 0.5,
percentage: true,
});
status.textContent = "";
console.log("output", output);
// ...
}
```

It needs to be asynchronous, as we’ll use the `await` keyword when we’re running the model.

Once we’ve updated the `status` paragraph, we’re ready to perform the so-called inference, which simply means to run the model with some data. This is done via the `detector()` function that was returned from `pipeline()`. The first argument we’re passing in is the image data (`img.src`).

The second argument is an options object. In it, we're setting the `threshold` property to 0.5. This means that we want the model to be 50% confident before claiming it has detected an object in the image. The lower you set this value, the more objects it'll detect, but the more mistakes it'll make as well.

We also specify `percentage: true`, which means that we want the coordinates for the objects in percentage values as opposed to pixels.

Try to run the app, upload an image, and open up the console. You should see the `output` variable logged out like this:

![Demo](https://huggingface.co/datasets/Xenova/transformers.js-docs/resolve/main/js-detection-console.png)

In the example above, we uploaded the image of the two elephants, so the `output` variable holds an array with two objects. Both objects have a key called `label`, which has the value “elephant”. The also have `box` objects that contains the coordinates for each of the animals.

Notice that the score is more than 0.99 for both, which means that the models is almost 100% sure it spotted two elephants. And indeed it did.

## Step 6: Render the boxes

The final step is to turn the `box` coordinates into rectangles around each of the elephants.

At the end of our `detect()` function, we’ll use loop through each of the objects in the `output` array with `.forEach()` and pass in `renderBox` as the callback.

```js
output.forEach(renderBox);
```

Here’s the code for the `renderBox()` function with comments to help you understand what’s going on:

```js
// Render a bounding box and label on the image
function renderBox({ box, label }) {
const { xmax, xmin, ymax, ymin } = box;

// Generate a random color for the box
const color =
"#" +
Math.floor(Math.random() * 0xffffff)
.toString(16)
.padStart(6, 0);

// Draw the box
const boxElement = document.createElement("div");
boxElement.className = "bounding-box";
Object.assign(boxElement.style, {
borderColor: color,
left: 100 * xmin + "%",
top: 100 * ymin + "%",
width: 100 * (xmax - xmin) + "%",
height: 100 * (ymax - ymin) + "%",
});

// Draw the label
const labelElement = document.createElement("span");
labelElement.textContent = label;
labelElement.className = "bounding-box-label";
labelElement.style.backgroundColor = color;

boxElement.appendChild(labelElement);
imageContainer.appendChild(boxElement);
}
```

The bounding box and label span also need some styling, so add the following to the `style.css` file:

```css
.bounding-box {
position: absolute;
box-sizing: border-box;
}

.bounding-box-label {
position: absolute;
color: white;
font-size: 12px;
}
```

And that’s it!

You have now built a fully functioning AI app that detects objects in images using nothing but a browser. No external server, no APIs, no build tools. Pretty cool!

![Demo](https://huggingface.co/datasets/Xenova/transformers.js-docs/resolve/main/js-detection-inference-elephant.png)


The app is live at the following URL: [https://huggingface.co/spaces/Scrimba/javascript-object-detector](https://huggingface.co/spaces/Scrimba/javascript-object-detector)
10 changes: 10 additions & 0 deletions examples/vanilla-js/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
# JavaScript Image Object Detection

This example contains a browser-based JavaScript application that detects objects in images using Transformers.js. The app requires no server, external libraries, or build tools. All you need is a code editor and a browser.

Here's how it works: the user uploads an image to the browser via the “Upload image” button. The app then downloads an AI model, runs the image through it, and finally displays an overview of where the different objects are located.

To learn more about how the code works, visit [this tutorial.](https://huggingface.co/docs/transformers.js/tutorials/vanilla-js)

## Getting Started
Download the repo and simply drag and drop the `index.html` file into a browser of your choice.
24 changes: 24 additions & 0 deletions examples/vanilla-js/index.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
<!DOCTYPE html>
<html lang="en">

<head>
<meta charset="UTF-8" />
<link rel="stylesheet" href="style.css" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Transformers.js - Object Detection demo</title>
</head>

<body>
<main class="container">
<label class="custom-file-upload">
<input id="file-upload" type="file" accept="image/*" />
<img class="upload-icon" src="https://huggingface.co/datasets/Xenova/transformers.js-docs/resolve/main/upload-icon.png" />
Upload image
</label>
<div id="image-container"></div>
<p id="status"></p>
</main>
<script src="index.js" type="module"></script>
</body>

</html>
Loading