Making a photo editor that is as least as capable as Microsoft Paint is not hard. Developers have done the hard work for building photo editor widgets for JavaScript. One example is the Toast UI Image Editor, located at https://ui.toast.com/tui-image-editor.
It is a very capable photo editor. You can style it with themes and users can apply effects that they like to their image easily. The list of features for the editor include loading images, undo and redo, crop, flip, rotate, draw, add shapes, write text, and add mask and image filters.
You can retrieve the image easily and save it to disk since the editor loads the image into a HTML canvas element.
In this article, we will build a photo editor with React using the React version of the Toast UI Image Editor. We will style it and add a download button to save the image to disk.
To start, we create the app with Create React App. Run npx create-react-app photo-editor-app
to create the project and install the packages.
Next we add our own packages to the app. We need @toast-ui/react-image-editor
, which the React version of the Toast UI Image Editor, Bootstrap and React Bootstrap for styling, React Router for routing, and Download.js for downloading the image to disk.
To do this, run npm i @toast-ui/react-image-editor bootstrap downloadjs react-bootstrap react-router-dom
. This will install all the libraries listed above.
With all the packages installed, we can start writing code. First we start with modifying App.js
. We replace the existing code with:
import React from "react";
import "./App.css";
import TopBar from "./TopBar";
import { Router, Route, Link } from "react-router-dom";
import { createBrowserHistory as createHistory } from "history";
import HomePage from "./HomePage";
const history = createHistory();
function App() {
return (
<div className="App">
<Router history={history}>
<TopBar />
<Route
path="/"
exact
component={HomePage}
/>
</Router>
</div>
);
}
export default App;
We will display the top bar which we will create and add a route to the home page so we can access it after we build it.
Next in App.css
we replace the existing code with:
.center {
text-align: center;
}
so that we can center our titles and buttons later.
Next we add a home page. Create a file called HomePage.js
in the src
folder and add the following:
import React, { useState, useEffect } from "react";
import "./HomePage.css";
import "tui-image-editor/dist/tui-image-editor.css";
import ImageEditor from "@toast-ui/react-image-editor";
import Button from "react-bootstrap/Button";
const icona = require("tui-image-editor/dist/svg/icon-a.svg");
const iconb = require("tui-image-editor/dist/svg/icon-b.svg");
const iconc = require("tui-image-editor/dist/svg/icon-c.svg");
const icond = require("tui-image-editor/dist/svg/icon-d.svg");
const download = require("downloadjs");
const myTheme = {
"menu.backgroundColor": "white",
"common.backgroundColor": "#151515",
"downloadButton.backgroundColor": "white",
"downloadButton.borderColor": "white",
"downloadButton.color": "black",
"menu.normalIcon.path": icond,
"menu.activeIcon.path": iconb,
"menu.disabledIcon.path": icona,
"menu.hoverIcon.path": iconc,
};
function HomePage() {
const [imageSrc, setImageSrc] = useState("");
const imageEditor = React.createRef();
const saveImageToDisk = () => {
const imageEditorInst = imageEditor.current.imageEditorInst;
const data = imageEditorInst.toDataURL();
if (data) {
const mimeType = data.split(";")[0];
const extension = data.split(";")[0].split("/")[1];
download(data, `image.${extension}`, mimeType);
}
};
return (
<div className="home-page">
<div className="center">
<h1>Photo Editor</h1>
<Button className='button' onClick={saveImageToDisk}>Save Image to Disk</Button>
</div>
<ImageEditor
includeUI={{
loadImage: {
path: imageSrc,
name: "image",
},
theme: myTheme,
menu: ["crop", "flip", "rotate", "draw", "shape", "text", "filter"],
initMenu: "",
uiSize: {
height: `calc(100vh - 160px)`,
},
menuBarPosition: "bottom",
}}
cssMaxHeight={window.innerHeight}
cssMaxWidth={window.innerWidth}
selectionStyle={{
cornerSize: 20,
rotatingPointOffset: 70,
}}
usageStatistics={true}
ref={imageEditor}
/>
</div>
);
}
export default HomePage;
There are a few tricks to note to get the image editor working properly. First we need to include the icons ourselves by adding:
const icona = require("tui-image-editor/dist/svg/icon-a.svg");
const iconb = require("tui-image-editor/dist/svg/icon-b.svg");
const iconc = require("tui-image-editor/dist/svg/icon-c.svg");
const icond = require("tui-image-editor/dist/svg/icon-d.svg");
This is not in the documentation so if you just follow the instructions, you won’t see the icons in the toolbar.
To change the colors of the image editor, we have:
const myTheme = {
"menu.backgroundColor": "white",
"common.backgroundColor": "#151515",
"downloadButton.backgroundColor": "white",
"downloadButton.borderColor": "white",
"downloadButton.color": "black",
"menu.normalIcon.path": icond,
"menu.activeIcon.path": iconb,
"menu.disabledIcon.path": icona,
"menu.hoverIcon.path": iconc,
};
This is fully documented in http://nhn.github.io/tui.image-editor/latest/themeConfig. The code above just changes the image editor to a black background and change the color of the download button to black with a white background.
Next we changed the size of the image editor by adding:
cssMaxHeight={window.innerHeight}
cssMaxWidth={window.innerWidth}
and we added the toolbar functionality by adding:
menu: ["crop", "flip", "rotate", "draw", "shape", "text", "filter"]
We set initMenu
to an empty string so that it won’t show any dialog when user load the image.
To let user save the image to disk, we add a button with a handler to do this. We have:
<Button className='button' onClick={saveImageToDisk}>Save Image to Disk</Button>
The saveImageToDisk
handler saves the image to disk and we get the image’s base64 text by adding:
const imageEditorInst = imageEditor.current.imageEditorInst;
const data = imageEditorInst.toDataURL();
into the saveImageToDisk
function.
Since we set the ref
prop in the ImageEditor
component by adding const imageEditor = React.createRef();
, we can get the image with those 2 lines above.
If data
is defined, then we can download the image. We add:
const mimeType = data.split(";")[0];
const extension = data.split(";")[0].split("/")[1];
download(data, `image.${extension}`, mimeType);
to get the file type and the extension of the image.
Base64 data is string that starts with some file type metadata, for example:

Notice that we have data:image/png
at the beginning. That tells us that the image is in png
format, so we can extract those with:
const mimeType = data.split(";")[0];
const extension = data.split(";")[0].split("/")[1];
and pass them as arguments to the download
function.
Then create a file called HomePage.css
, and add:
.button {
margin-bottom: 10px;
}
to add some margin to the ‘Save file to disk’ button.
Next we build the top bar by creating TopBar.js
in the src
folder and add the following code:
import React from "react";
import Navbar from "react-bootstrap/Navbar";
import Nav from "react-bootstrap/Nav";
import "./TopBar.css";
import { withRouter } from "react-router-dom";
function TopBar({ location }) {
const { pathname } = location;
return (
<Navbar expand="lg" variant="dark">
<Navbar.Brand href="#home">Photo Editor App</Navbar.Brand>
<Navbar.Toggle aria-controls="basic-navbar-nav" />
<Navbar.Collapse id="basic-navbar-nav">
<Nav className="mr-auto">
<Nav.Link href="/" active={pathname == "/"}>
Home
</Nav.Link>
</Nav>
</Navbar.Collapse>
</Navbar>
);
}
export default withRouter(TopBar);
The Navbar
component is provided by React Boostrap. We wrap the TopBar
component with the withRouter
function so that we get the location
prop passed in, which has the pathname
property. We use pathname
to check for the path and see if we set the active
propr of Nav.Link
to true
by seeing the path match the URL of the link.
Next create a file called TopBar.css
and add:
nav {
background-color: green!important;
}
to change the color of the Navbar
.
Finally, in index.html
, we replace the existing code with:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="theme-color" content="#000000" />
<meta
name="description"
content="Web site created using create-react-app"
/>
<link rel="apple-touch-icon" href="logo192.png" />
<!--
manifest.json provides metadata used when your web app is installed on a
user's mobile device or desktop. See https://developers.google.com/web/fundamentals/web-app-manifest/
-->
<link rel="manifest" href="%PUBLIC_URL%/manifest.json" />
<!--
Notice the use of %PUBLIC_URL% in the tags above.
It will be replaced with the URL of the `public` folder during the build.
Only files inside the `public` folder can be referenced from the HTML.
Unlike "/favicon.ico" or "favicon.ico", "%PUBLIC_URL%/favicon.ico" will
work correctly both with client-side routing and a non-root public URL.
Learn how to configure a non-root public URL by running `npm run build`.
-->
<title>Photo Editor App</title>
<link
rel="stylesheet"
href="https://maxcdn.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css"
integrity="sha384-ggOyR0iXCbMQv3Xipma34MD+dH/1fQ784/j6cY/iJTQUOhcWr7x9JvoRxT2MZw1T"
crossorigin="anonymous"
/>
</head>
<body>
<noscript>You need to enable JavaScript to run this app.</noscript>
<div id="root"></div>
<!--
This HTML file is a template.
If you open it directly in the browser, you will see an empty page.
You can add webfonts, meta tags, or analytics to this file.
The build step will place the bundled scripts into the <body> tag.
To begin the development, run `npm start` or `yarn start`.
To create a production bundle, use `npm run build` or `yarn build`.
-->
</body>
</html>
We changed the title of the app and added the Bootstrap CSS to the head
tag.