logo
Published on

Create a JavaScript plugin to highlight and tweet text

Authors
  • avatar
    Name
    Alberto Montalesi
    Twitter

Create a plugin to highlight and tweet text

Many times I found myself reading a passage on a website and thinking I wanted to share that with my friends or random people of the internet, but few are the times where the website facilitated that process for me.

Usually, I would have to copy the passage I wanted to share, go on Twitter and tweet it but we all know that most people are lazy so, if you want your user to tweet content from your articles/blogs, etc... in this tutorial, I will teach you how to create a simple plugin to do that.

Alternatively, if you don't want to bother with coding it yourself, I've already packaged it for you and you can find it on npm at highlight-to-tweet.

Credits for most of the idea and the code go to this user on WebFlow. I mostly transformed the JQuery code into a native JavaScript code and published it as an npm package.

The plugin we are going to build will work in the following way:

  • A user highlight some text on your page
  • They are prompted to tweet it with a button
  • They click the button and they can tweet it, all without leaving your site

 

Create the tweetHighlighted function

Now that you are ready to start, let's open your JavaScript file and start with:

document.addEventListener('DOMContentLoaded', () => {
  window.tweetHighlighted = () => {
    // our function will be built here
  }

  // we call it once the page loads
  window.tweetHighlighted()
})

The first thing we want to do know is to define some properties that will come in handy later on. Inside of the window.tweetHighlighted = () => { function, we want to add this code:

const settings = {
  node: "<a href='#'>Tweet</a>",
  maxLength: 280,
  extra: null,
  via: null,
  popupArgs: 'width=400,height=400,toolbar=0,location=0',
}

These are the basic settings of our plugin:

  • node defines the element that will be contained in the button that pops up
  • maxLength is the max length allowed for the tweet, here I've set it as 280 because that's the max allowed for a tweet, but if you want to set it lower you can do it
  • extra is simply some extra text that you want to include in your tweets
  • via defines the handler for twitter, if you want to be tagged whenever a user tweets something from your site
  • popupArgs is used to define the twitter popup

Our next step will be to create a button, add it to the DOM and give an event listener to it.

Continuing after the code we just wrote, add this:

let url = ''
let text = ''
const shareButton = document.createElement('div')
shareButton.style.display = 'none'
shareButton.addEventListener('click', (e) => {
  e.preventDefault()
  e.stopPropagation()
  window.open(url, '_blank', settings.popupArgs)
})
const body = document.querySelector('body')
body.appendChild(shareButton)

Firstly, we defined two empty variables that will hold our url and text for the tweet.

Secondly, we created a new div element and assign it to our variable shareButton.

We then hid it with display:'none' and added an event listener to it to listen for click events.

When a user clicks our button we use stopPropagation to avoid the click event to bubble down to the a tag that we assigned above to the node in our settings and we also call window.open passing the url that will be populated later and our arguments to define the size of the window to open.

After that, we are simply adding this newly created element to the body of our page.

One thing you may be wondering is how we are going to identify when the user highlights texts on the page.

The answer to that is simple, we will use the mouseup event and check if the user selected something.

 

Add event listeners to the function

Continuing from where we left, let's add this code:

document.addEventListener('mouseup', (e) => {
  text = getSelectedText()
  const btnExists = shareButton.style.display !== 'none'
  if (btnExists || !text.length) {
    return
  } else {
    appendShareButton(e)
  }
})

const getSelectedText = () => {
  if (window.getSelection) {
    return window.getSelection().toString()
  } else if (document.selection) {
    return document.selection.createRange().text
  }
  return ''
}

Awesome, these two functions will ensure that when the user triggers the mouseup event, if there is some text highlighted, we will capture it, store it in our text variable and call the function appendShareButton to add that button to the DOM.

Let's continue and define that function:

const appendShareButton = (e) => {
  url = getTweetURL(text, settings.extra, settings.via)

  shareButton.innerHTML = ''
  shareButton.innerHTML += settings.node
  shareButton.innerHTML += tweetIcon
  if (settings.cssClassess && settings.cssClassess.length) {
    shareButton.classList.add(settings.cssClassess)
  }
  shareButton.style.top = `${e.pageY}px`
  shareButton.style.left = `${e.pageX}px`
  shareButton.style.position = 'absolute'
  shareButton.style.cursor = 'pointer'
  shareButton.style.display = 'flex'
  shareButton.style.justifyContent = 'space-between'
  shareButton.style.alignContent = 'center'
  shareButton.style.alignItems = 'center'
  shareButton.style.opacity = 0

  shareButton.style.fontFamily = 'Arial, Helvetica, sans-serif'
  shareButton.style.backgroundColor = '#3898EC'
  shareButton.style.padding = '10px 15px'
  shareButton.style.width = '100px'
  shareButton.style.borderRadius = '5px'
  shareButton.style.color = 'white'
  shareButton.firstChild.style.color = 'white'
  shareButton.firstChild.style.textDecoration = 'none'
  shareButton.lastChild.style.fill = 'white'
}

Wow, that is a long function but don't worry, it does not do much. Most of those lines are just styling.

Firstly, we get the create the complete url by calling yet another function that we haven't defined, called getTweetURL, passing it the highlighted text, the extra text and the via handler.

We then define a list of styles on our shareButton to display it and position it appropriately.

The only important ones are the one relative to the positioning of the button, everything else is purely styling.

It may be gross to define each style on a different line like this but I figured it would be clearer for you to see. If you want, you can define styles this way too:

shareButton.style.cssText = `
/* your list of style here, same as how you would add it in a CSS file*/
`

You may have noticed this line: shareButton.innerHTML += tweetIcon;

We haven't defined yet a tweetIcon so let's go outside this function and copy this code:

const tweetIcon = `<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" viewBox="0 0 24 24"><path d="M24 4.557c-.883.392-1.832.656-2.828.775 1.017-.609 1.798-1.574 2.165-2.724-.951.564-2.005.974-3.127 1.195-.897-.957-2.178-1.555-3.594-1.555-3.179 0-5.515 2.966-4.797 6.045-4.091-.205-7.719-2.165-10.148-5.144-1.29 2.213-.669 5.108 1.523 6.574-.806-.026-1.566-.247-2.229-.616-.054 2.281 1.581 4.415 3.949 4.89-.693.188-1.452.232-2.224.084.626 1.956 2.444 3.379 4.6 3.419-2.07 1.623-4.678 2.348-7.29 2.04 2.179 1.397 4.768 2.212 7.548 2.212 9.142 0 14.307-7.721 13.995-14.646.962-.695 1.797-1.562 2.457-2.549z"/></svg>`

This is the tweet icon, it's a simple SVG I've found online, this will allow us to don't have to call it from a CDN or somewhere else.

Awesome, now we have a function to append the button to the DOM, we only need to define that function getTweetURL.

Let's do that now:

 

Construct the URL for the tweet

const getTweetURL = function (text, extra, via) {
  let url = 'https://twitter.com/intent/tweet?text='
  // trim the text to fit in the max allowed 280 characters
  const viaUrl = `&via=${via}`
  const maxLength = settings.maxLength > 280 ? 280 : settings.maxLength
  const maxAllowedLength = maxLength - viaUrl.length - extra.length
  let textToTweet = text
  if (text.length > maxAllowedLength) {
    textToTweet = text.substring(0, maxAllowedLength - 1)
  }
  url += encodeURIComponent(textToTweet)
  if (extra) url += encodeURIComponent(' ' + extra)

  if (via) url += viaUrl

  return url
}

Let's break down this function:

  • first, we define the base URL that we need
  • then we create the portion of URL that contains our via handler
  • after that, we calculate how many characters in our text allowed to take, based on the maxLength we defined (or the max of 280), the via handler and the extra text
  • if the text is too long, we use substring to take only a portion of it
  • we then stick everything together and use encodeURIComponent to ensure all characters are escaped properly

There you have it, this function returns a perfect URL to share the tweet.

Now you can try this code on your webpage and you will see that everything should work fine.

If you want a more customizable plugin you can check out the one i've published on Github.