• 0

Javascript - Determine length of text in pixels


Question

Hi all,

A while ago I had someone create a sliding bar effect for a website I was working on. If you check out the site linked below, you will see the sliding bar thingy that I am talking about where it says "Tona Boards" on a sort of green arrow, then it slides back and forth to reveal different text.

http://www.tonaboards.com/

Now there is just one small problem with it. There is always this unnecessary amount of extra space after the text if the line of text is quite long. This is a result of a poor calculation for the width of the arrow. Currently this is how the width of the arrow is calculated:

var width = titles[title_id][0].length * 15;

Basically, he is just multiplying the number of characters by 15 to get a close (but not accurate) pixel width that will fit the text.

Is there a better way to work out the exact amount of pixels a line of text takes up? I did some googling and it seems possible but I am a javascript noob and can't figure out how to incorporate any of the examples in to my particular code.

Here is the full javascript for the sliding arrow.

http://www.tonaboards.com/wp-content/themes/TonaLife/arrow.js

Cheers for any help.

Jordan

8 answers to this question

Recommended Posts

  • 0

You could try setting the tag holding the text as inline-block and check the width. Display block, float left is the other setting you could use to determine its width. I had to do this once to set the width on the fly for a menu that had to be centered at all times.

  • 0

Would I use something like this for determining the width?

var arrowlink = document.getElementByClass("arrow_link");

var width = arrowlink.style.width;

Like I said, I am a js noob. I tried this and it didn't work.

  • 0

Its easier to do it with jQuery. You could collect all the elements with a certain class loop through them, detect and store the width as below.

var $links = $('.arrow_links');

for(i=0; i<$links.length; i++){
       var $obj = $links.eq(i);
       var len = $obj.width();
       $obj.data('wd', len);
}

Now that the widths have been stored you can access at the time of animation. Like I mentioned earlier you will need to either set the display type to inline-block or a combo of display block and float left, whichever suits your circumstances.

  • 0

You could just wrap the text with a <span> and use jQuery's .width() method to get the width of that element.

JS:

$('h2 span').width());

HTML:

&lt;h2&gt;&lt;span&gt;Pellentesque habitant morbi tristique&lt;/span&gt;&lt;/h2&gt;

It should work :)

  • 0

The only problem I see with your implementation is that you need the width before adding the text in the DOM. Something like jQuery.width() will only work with elements inside the DOM, they need to be added on the page. Now, a solution to work around this would be to calculate all text widths beforehand, store the results and use those later. During the calculation, the texts can be appended to an element which is absolutely positioned outside the visible screen area. After all texts are measured, this element can be removed and the results can be used for the animations afterwards.

I see you're using MooTools for this script. I don't want to force you to convert to jQuery, but since you're working in a WordPress environment I believe you should choose jQuery for this. WordPress already loads jQuery for its own scripts (e.g. threaded comment reply form) and most WordPress plug-in developers tend to use jQuery as well. Also, it's best practice to stick with one JavaScript library and not mix them. (Although you could say WordPress is to blame here... perhaps this could be a valid exception.) If you're only going to use MooTools for its DOM capabilities and not use its OOP functionality, jQuery is the preferred library anyway.

Anyway, here's how you could implement text width calculation in jQuery:

jQuery(document).ready(function($) {
	// Title element structure: [ titleText, linkUrl, titleWidth ]
	var titles = [
		['Tona Boards', '', 0],
		['Check Out The New Driftwood Wakeskate', 'http://www.tonaboards.com/driftwood', 0],
		['Read The Driftwood Blog', 'http://www.tonaboards.com/blog', 0],
		['Learn More About TONA', 'http://www.tonaboards.com/130/wake-skate-project', 0]
	];

	// Create temporary element to test widths in
	var textdiv = $('&lt;div /&gt;').css({
		// position it off the page (in a land far, far away)
		position: absolute,
		left: -9999,
		top: -9999,
		// reset styles so that we only measure the text
                // if you need these to be added in your calculation, just remove these four lines and the comma on the previous line
		border: 0,
		margin: 0,
		padding: 0,
		outline: 0
	}).appendTo('#arrow_content'); // by appending to #arrow_content, we can inherit its font styling as well (yay!)
	// Loop through the titles array
	for(var i=0, len=titles.length; i &lt; len; ++i) {
		// Set text, retrieve width and store it (all on one line, thanks to chaining!)
		titles[i][2] = textdiv.html(titles[i][0]).width();
	}
	textdiv.remove(); // remove temporary element
	textdiv = null; // garbage collection for IE (don't know whether it really helps though...)

	/* ------------------------------ */

	// Now, whenever you need to animate your texts, use something like this:
	$('#arrow_content').width(0).animate({
		width: titles[i][2]	// make sure you got a valid i here!
	});
	// This would first set the width to 0px and then animate ('morph') to the calculated text width
});

Once again: if you prefer another library, feel free to replicate this in your desired library.

(On a side note, you're including MooTools twice in your <head> - better fix that :p)

  • 0

Thanks a lot Calculator. I would like to switch over to jQuery from MooTools (it was the guy who wrote the initial script that choose MooTools in the first place) so I will try out your code. Hopefully this works.

One question though, should your code work as is? I replaced my arrow.js with this and nothing is happening. Does the .animate part need to be expanded on before anything will happen?

  • 0

It looks like there's indeed a small error in my snippet above, the word absolute is meant to be a string value and should be wrapped in quotes, like so:

        // Create temporary element to test widths in
        var textdiv = $('&lt;div /&gt;').css({
                // position it off the page (in a land far, far away)
                position: 'absolute', // now with added quotes!
                left: -9999,
                top: -9999,

Basically, you want to start editing below the /* ----- */. Below that line, there's some example code of how you could start the animation, the reason why this doesn't work straight away is that the value of i is invalid there (after the loop, it will have exceeded the titles array index bounds). You can remove that and start recoding the rest of your script from there. First, you animate the arrow to the width of the first item in the titles array, then you set the link's text and href using the data from the title item and then you set a timeout to animate and set the next item after x seconds. This should be pretty straightforward in jQuery, have a look at the jQuery Docs and start experimenting! :D

This topic is now closed to further replies.
  • Recently Browsing   0 members

    • No registered users viewing this page.
  • Posts

    • PotPlayer 260622 by Razvan Serea PotPlayer is an extremely light-weight multimedia player for Windows. It feels like the KMPlayer, but is in active development. Supports almost every available video formats out there. PotPlayer contains internal codecs and there is no need to install codecs manually. Other key features include WebCam/Analog/Digital TV devices support, gapless video playback, DXVA, live broadcasting. Distinctive features of the player is a high quality playback, support for all modern video and audio formats and a built DXVA video codecs. A wide range of subtitles are supported and you are also able to capture audio, video, and screenshots. A comprehensive video and audio player, that also supports TV channels, subtitles and skins. Its been described on the Internet as The KMPlayer redux, and it pretty much is. Daum PotPlayer 260622 (1.7.22963) changelog: Removed Kakao TV Added pause function when navigating via the navigation bar Significantly improved internal stability Fixed an issue where colors appeared strange during RGB24 processing Improved playback for some HTTP streams Improved sync processing for the built-in audio renderer Fixed an issue where certain MP4 files behaved abnormally during playback Download: Daum PotPlayer (64-bit) | 54.7 MB (Freeware) Download: Daum PotPlayer (32-bit) | 61.1 MB View: Daum PotPlayer Home Page | Screenshot Get alerted to all of our Software updates on Twitter at @NeowinSoftware
    • Tixati 3.44 is out.
    • Speccy 1.34.084 by Razvan Serea Speccy will give you detailed statistics on every piece of hardware in your computer. Including CPU, Motherboard, RAM, Graphics Cards, Hard Disks, Optical Drives, Audio support. Additionally Speccy adds the temperatures of your different components, so you can easily see if there's a problem! Processor brand and model Hard drive size and speed Amount of memory (RAM) Graphics card Operating system At first glance, Speccy may seem like an application for system administrators and power users. It certainly is, but Speccy can also help normal users, in everyday computing life. If you need to add more memory to your system, for example, you can check how many memory slots your computer has and what memory's already installed. Then you can go out and buy the right type of memory to add on or replace what you've already got. Download: Speccy 1.34.084 | 20.5 MB (Freeware) View: Speccy Website | Screenshot Get alerted to all of our Software updates on Twitter at @NeowinSoftware
    • ImgDrive 2.2.7 by Razvan Serea ImgDrive is a CD/DVD/BD emulator - a tool that allows you to mount optical disc images by simply clicking on them in Windows Explorer. If you have downloaded an ISO image and want to use it without burning it to a blank disc, ImgDrive is the easiest way to do it. ImgDrive features: One-click mounting of iso, cue, nrg, mds/mdf, ccd, isz images Runs on 32-bit and 64-bit Windows versions Mount ape, flac, m4a, wav, wavpack, tta file as AUDIO CD (16-bit/44.1kHz) Mount a folder as DVD/BD Mount images in command line Does not require rebooting after installation Support up to 7 virtual drives at the same time Support multi session disc image (ccd/mds/nrg) A special portable version is available Translated to more than 10 languages Support File Type: .ccd - CloneCD image files .cue - Cue sheets files of ape/flac/m4a/tta/wav/wv/bin .iso - Standard ISO image files .isz - Compressed ISO image files .nrg - Nero image files .mds - Media descriptor image files ImgDrive 2.2.7 changelog: Added command line parameter to set number of drives Added AACS-Auth support for HD DVD Bumped kernel driver version to 2.2.7 Download: ImgDrive 2.2.7 | 692 KB (Freeware, paid upgrade available) Download: ImgDrive Portable 535 KB View: ImgDrive Home Page | Screenshot Get alerted to all of our Software updates on Twitter at @NeowinSoftware
    • AnyDesk 9.7.7 by Razvan Serea AnyDesk is a fast remote desktop system and enables users to access their data, images, videos and applications from anywhere and at any time, and also to share it with others. AnyDesk is the first remote desktop software that doesn't require you to think about what you can do. CAD, video editing or simply working comfortably with an office suite for hours are just a few examples. AnyDesk is designed for modern multi-core CPUs. Most of AnyDesk's image processing is done con­currently. This way, AnyDesk can utilize up to 90% of modern CPUs. AnyDesk works across multiple platforms and operating systems: Windows, Linux, Free BSD, Mac OS, iOS and Android. Just 7 megabytes - downloaded in a glimpse, sent via email, or fired up from your USB drive, AnyDesk will turn any desktop into your desktop in se­conds. No administrative privileges or installation needed. AnyDesk 9.7.7 fixes: Fixed an issue that prevented users from creating meetings without an active license Download: AnyDesk 9.7.7 | 8.0 MB (Free for private use, paid upgrade available) Links: AnyDesk Home Page | Other platforms | Release History | Screenshot Get alerted to all of our Software updates on Twitter at @NeowinSoftware
  • Recent Achievements

    • Dedicated
      tuben earned a badge
      Dedicated
    • Week One Done
      mnsgroup earned a badge
      Week One Done
    • Conversation Starter
      sumytbe earned a badge
      Conversation Starter
    • One Year In
      B4dM1k3 earned a badge
      One Year In
    • One Year In
      DarkWun earned a badge
      One Year In
  • Popular Contributors

    1. 1
      +primortal
      523
    2. 2
      +Edouard
      199
    3. 3
      PsYcHoKiLLa
      94
    4. 4
      Michael Scrip
      82
    5. 5
      Steven P.
      67
  • Tell a friend

    Love Neowin? Tell a friend!