Categories
Vue

How to Make a Photo Editor with React

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.

Categories
Vue Answers

How ow to Download a File in the Browser with Vue.js?

Sometimes, we want to download a file in the browser with Vue.js.

In this article, we’ll look at how to download a file in the browser with Vue.js.

Download a File in the Browser with Vue.js

To download a file in the browser with Vue.js, we can make a GET request to get the file response data.

Then we can create a link from it and click on it programmatically to download the file.

For instance, we can write:

<template>
  <div id="app">
    <a
      href="#"
      @click.prevent="
        downloadItem({
          url:
            'https://test.cors.workers.dev/?https://www.w3.org/WAI/ER/tests/xhtml/testfiles/resources/pdf/dummy.pdf',
          label: 'example.pdf',
        })
      "
    >
      download
    </a>
  </div>
</template>
<script>
import axios from "axios";

export default {
  name: "App",
  methods: {
    async downloadItem({ url, label }) {
      const response = await axios.get(url, { responseType: "blob" });
      const blob = new Blob([response.data], { type: "application/pdf" });
      const link = document.createElement("a");
      link.href = URL.createObjectURL(blob);
      link.download = label;
      link.click();
      URL.revokeObjectURL(link.href);
    },
  },
};
</script>

We have the downloadItem method that takes an object with the file url to download, and the label property which has the file name.

In the method, we call axios.get to get the data from the url .

We set responseType to 'blob' to let us download the data.

Next, we create a Blob instance with the response.data which has the file contents.

type is set to the MIME type of the file.

Next, we use document.createElement to create the a element which we click to download the file.

We set its href to the URL created by URL.createObjectURL with the blob .

And then we set the download property to the label file name.

Then we call click to click on the a element to do the download.

And finally, we call URL.revokeObjectURL to clear the URL resources.

Conclusion

To download a file in the browser with Vue.js, we can make a GET request to get the file response data.

Categories
Vue Answers

How to Listen for Right Clicks with Vue.js?

Sometimes, we want to listen for right clicks and do something when the user right-clicks on an element with Vue.js.

In this article, we’ll look at how to listen for right clicks and do something when the user right-clicks on an element with Vue.js.

Listen for Right Clicks with Vue.js

To listen for right clicks and do something when the user right-clicks on an element with Vue.js, we can use the @contextmenu directive to listen for contextmenu events, which are emitted on right-click.

For instance, we can write:

<template>  
  <div id="app">  
    <button @contextmenu.prevent="onRightClick">right click me</button>  
  </div>  
</template>  
<script>  
export default {  
  name: "App",  
  methods: {  
    onRightClick() {  
      console.log("right clicked");  
    },  
  },  
};  
</script>

We add the prevent modifier to prevent the default right-click menu from showing, and instead the onRightClick method will be run when we right click on the button.

Now when we right-click on the button, we see 'right clicked' logged.

Conclusion

To listen for right clicks and do something when the user right-clicks on an element with Vue.js, we can use the @contextmenu directive to listen for contextmenu events, which are emitted on right-click.

Categories
Vue Answers

How to Get an Element’s Height with Vue.js?

Sometimes, we want to get an element’s height with Vue.js.

In this article, we’ll look at how to get an element’s height with Vue.js.

Get an Element’s Height with Vue.js

To get an element’s height with Vue.js, we can assign a ref to the element we want to get the height for.

Then we can get the height with the clientHeight property of the element that’s been assigned the ref.

For instance, we can write:

<template>  
  <div id="app">  
    <div ref="infoBox">hello world</div>  
  </div>  
</template>  
<script>  
export default {  
  name: "App",  
  mounted() {  
    console.log(this.$refs.infoBox.clientHeight);  
  },  
};  
</script>

We assigned the ref with the ref prop set to a name.

Then we can use the this.$refs property to get the element with thos.$refs.infoBox which returns the div element.

And then we can use clientHeight to get the div’s height.

Conclusion

To get an element’s height with Vue.js, we can assign a ref to the element we want to get the height for.

Categories
Vue Answers

How to Output a Comma-Separated Array with Vue.js?

Sometimes, we want to output a comma-separated array with Vue.js.

In this article, we’ll look at how to output a comma-separated array with Vue.js.

Output a Comma-Separated Array with Vue.js

To output a comma-separated array with Vue.js, we can use the v-for directive.

For instance, we can write:

<template>
  <div id="app">
    <span v-for="(list, index) in lists" :key="list">
      <span>{{ list }}</span>
      <span v-if="index + 1 < lists.length">, </span>
    </span>
  </div>
</template>
<script>
export default {
  name: "App",
  data() {
    return {
      lists: ["Vue", "Angular", "React"],
    };
  },
};
</script>

We use the v-for directive to render the list array into a comma-separated list.

We add a comma only when index + 1 < lists.length so we only add the commas between the words.

A shorter way to render the words in the array into a comma-separated list is to use the array join method.

For example, we can write:

<template>
  <div id="app">
    <span>{{ lists.join(", ") }}</span>
  </div>
</template>
<script>
export default {
  name: "App",
  data() {
    return {
      lists: ["Vue", "Angular", "React"],
    };
  },
};
</script>

to combine the words with the join method in the template.

Either way, we get:

Vue, Angular, React

rendered on the screen.

Conclusion

To output a comma-separated array with Vue.js, we can use the v-for directive.

A shorter way to render the words in the array into a comma-separated list is to use the array join method.