• 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

    • You still can, its just under the Transform flyout for WordArt now
    • Likely nothing will be done in corporate America, there have been countless Tesla self-driving incidents. Then again, there have also been countless human operated incidents. It's literally daily news here in Canada, to the extent that it's now odd if we get a day where a collision doesn't get announced on the radio throughout the day...
    • SKG Hand Massager with Heat OS500 hands on by Steven Parker I was offered the chance to test out the SKG Hand Massager with Heat OS500, and full disclosure, they let me keep it regardless of my findings. Anyway, I jumped at the chance due to my long hours sitting at my desk, mousing around. Apologies for the knife cut across the top of the box; that was my doing, being a bit too heavy-handed with opening up the outer packaging. First up, what's in the box: SKG Hand Massager with Heat OS500 1x Type-C charging cable User Manual 1-Year Warranty (card) In short, everything you need to get started. According to the official Amazon listing, here are the key features: Full-Hand Air Compression: OS500 wraps your fingers, palm, and wrist with multi-chamber air compression for a complete hand relaxation experience. The extended massage chamber helps cover more of the hand and wrist area than standard palm-only hand massagers Palm Kneading with 6 Modes & 6 Intensities: Built-in palm kneading rollers add a hands-on massage feel, while 6 preset modes and 6 pressure levels let you choose the comfort level that fits your day—from gentle relaxation to a firmer full-hand massage 3 Heat Levels with Cooling Fan: Choose from 104°F, 113°F or 122°F warmth to suit different seasons and comfort preferences. The built-in cooling fan helps reduce stuffiness during heated sessions, keeping your hand feeling fresh and comfortable Easy Visual Display & Smart Timer: The digital image display clearly shows massage area, mode, intensity, heat level, and remaining time at a glance. Select 10, 15, or 20-minute sessions for quick office breaks, evening relaxation, or everyday hand care Rechargeable, Cordless & Comfortable: A 3000mAh battery supports over 90 minutes of full-function use on a full charge, with convenient USB-C charging. The soft inner lining, smooth ABS/PU finish, and premium black-gold design make OS500 ideal for home, office, or gifting With all that out of the way, here are my own findings. I gave it a try on both left and right hands, and as you can maybe see from the above YouTube Short, (sorry for the shaky video), my whole hand fits in, but my wrist barely enters the Hand Massager. I was able to push through a bit more with my fingertips extending out the other end to get a bit of massaging on the start of my wrist. Usage For some reason, there is a strap that is very difficult to fasten to my wrist with one hand. I am not sure what function it has, and it isn't mentioned in the user manual. The only thing I could find was in the product images that claimed "wrist precision". Unlike the Bob and Brad Hand Massager, this device does not massage the wrist anyway, even though a "wrist mode" is mentioned, which must be for smaller hands than I have, as it is mainly intended for the hand and fingers. In addition, for its steeper price, there are no disposable gloves provided in the box, which is a bit of an issue considering the internal cover (which appears to be elasticated nylon) cannot be removed for washing; so you are left with only one choice: always thoroughly wash your hands before using it. I can imagine this thing getting a bit grimy after a period of use, and that is a bit of a shame. With that said, the buttons on the device, from left to right, do the following: Heat button: 3-level heat control at 104°F, 113°F, or 122°F Mode button: Auto mode Circular mode Soothing mode Relax mode Palm and fingers mode Palm and wrist mode Intensity button: from (First-time users) 15Ka, 25Ka, 35Ka, 45Ka, 55Ka, 60Ka (Intensive relief) Knead button: on or off (6 pressure levels) Power button: Long-press to turn on or off Cooling button: turn on or off the cooling fan Also, in the product imagery, it states there are 36 "custom modes," but nowhere is it listed what these modes are. I can only imagine that they mean a combination of all of the above settings in different intensity levels. The device itself seems to rely on a single "kneading" mechanism located at the palm area of the hand, which spins when in use, and the other massage features are mainly utilized through the air sacs, increasing and decreasing at various levels on the hand and fingers. I am not sure it offered too much relief for someone who is typing and operating a mouse for hours at a time; further testing may be required. It does feel nice, though. Finally, you may be wondering how this fits into the scope of a tech website? Well, let me tell you something: sometimes I sit for up to 15 hours working on Neowin, and although I take breaks in between, it takes a toll on my body. I think in the immediate absence of a partner to apply relief, a good massager like this Hand Massager can shed the strains of the day in just a couple of 15-minute bursts. On the official website, this has an MSRP of $99.99, but luckily for our readers, it is selling at $10 off for just $89.99 right now on Amazon. SKG Hand Massager with Heat OS500 for $89.99 (with $10 off coupon), $99.99 MSRP For me, this gets a thumbs hands(?) down. However, it could be improved by making it so that the protective covering could be removed and thrown into the washing machine, or get yourself some disposable gloves to use with it. As an Amazon Associate, we earn from qualifying purchases.
    • Thanks for the info, but I'm still not sure if I need this....
    • We check out the SKG PS700 Neck Massager by Steven Parker I was offered the chance to test out the SKG PS700 Neck Massager, and full disclosure, they let me keep it regardless of my findings. Anyway, I jumped at the chance due to my long hours sitting at my desk; I figured it could offer some neck pain relief. What's in the box: SKG PS700-2 Neck Massager Rechargeable Battery (inside massager) Type-C USB cable User Manual Quick Start guide 1-Year Warranty In short, everything you need to get started. According to the official listing, here are the key features: Biomimetic Kneading & High Torque Motor: Designed with innovative biomimetic kneading heads that perfectly simulate the touch of human hands. Powered by a high-torque motor, this massager delivers powerful and precise deep tissue relief to effectively target stiff neck muscles and release built-up tension Soothing Heat & Integrated Sound Relaxation: Experience the ultimate Relaxationation with our dual-action approach. The soothing heat function gently warms your neck, while the built-in sound Relaxation provides calming audio tracks, helping you achieve a state of mindfulness and mental tranquility during your physical massage Cordless Convenience & Travel-Ready & Father's Day Gifts: Crafted for maximum portability and ease of use. Its lightweight, cordless design allows you to enjoy a premium massage anywhere without the hassle of tangled wires-whether you're taking a quick break at your desk or winding down at home Versatile Relief for Home & Office: An essential wellness companion for office workers, gamers, frequent travelers, or anyone looking to integrate mindfulness into their daily routine. It seamlessly fits into your lifestyle, providing instant neck relief whenever and wherever you need it Safe & Premium Materials: Manufactured with high-quality, skin-friendly materials to ensure a safe and comfortable experience without irritation. SKG backs this device with dedicated customer service, making it a thoughtful tech-health gift for family and friends App & Bluetooth Music Control: Connect via Bluetooth to control your massage settings through the dedicated app and enjoy your favorite music during your massage session for a fully customizable and immersive relaxation experience Red Light Warmth Technology: Features advanced red light warmth technology that penetrates deep into neck muscles to enhance blood circulation and provide soothing comfort while relieving muscle tension and stiffness Design With all that out of the way, here are my own findings. SKG does not say what materials are used to make the neck massager. However, on the product website, it mentions "soft-touch silicone" with what looks like PU leather cushioning, with the rest being mostly made up of plastics. On the inside of the massager, there are two "biomimetic kneading heads" that are motorized for the different styles of massage, which are not actually listed at all in the paper user manual, but the standard included modes are: De-stress mode, Mediation mode, Relax mode, Shiatsu mode. The massager looks quite premium and is actually very comfortable to wear. This massager is small and light enough to go anywhere, as it doesn't get in the way of anything, so I was able to use it in the chair while writing this review. Unlike the back massager, SKG does not warn in the user guide not to use it for more than 30 minutes a day (or two 15-minute sessions). However, there is a long laundry list of important safeguards to consider before and during the use of the device, and it is warned that the neck massager is not waterproof. It also includes a 1,400mAh battery with a rated power of 14W and input of 5V, which is the standard for up to USB 3.0 power (although the Amperage is not mentioned at all). SKG does not say how long it takes to charge, but a quick calculation at 2A (if that is what it is) would mean it would take roughly 1.5 hrs to charge from empty. In any case, the light around the button changes from orange to green on a full charge. In addition, it is not possible to use the device while it is charging. On the right of the neck massager is the On/Off and modes button, which also acts as a joystick. You can operate all the modes directly from the power button, as well as the app, which I'll get into a bit later: Push up: Short press to adjust Heat levels On/Off button: long press Mode Switching: Short press (while in operation) ➕ Push left: increase Music volume ➖ push right: decrease Music volume Push down: Short-press to turn Music on or off The massager defaults to De-stress mode, and it is not stated anywhere if the neck massager has overheat protection. This time around, regarding heat, the only detail I could find is that it has "triple action soothing heat." The temperature stages are not listed anywhere in the paper manual, Amazon listing, or official website. The heat levels can be adjusted through the app or directly on the device using the joystick button. Usage There's also the SKG Health app, which makes using the massager far easier than feeling around for the button on the side of your neck. If the app is stopped, you are required to log in with a verification code over email, which I am not too pleased with, as this means it will only work that way for however long SKG decides to support it through said app. However, I was not able to get the app to connect to the OS500, which I have reported back to my contact. Bluetooth appeared to be working on the neck massager as it became available to pair with my phone, but the SKG app failed to discover it. Before I forget, there's also a switch next to the USB charging port to deactivate and activate the Voice Prompt, which, when enabled, audibly tells the user when switching intensities, modes, or connecting to the app and informs when the massages start and are completed. That said, on to my likes and dislikes, which are listed below. What I didn't like Unable to connect the Neck Massager to the app Use through the mobile app relies on continued support from SKG What I liked Can be used without the app Cordless use Light and comfortable to wear Heat is also quite comfortable Where to buy: According to the official website, this has an MSRP of $249.99, but is currently $50 (on Amazon). To sweeten the deal a bit more, there's also an in-page coupon that knocks a further $20 off the price. SKG PS700-2 Neck Massager for $179.99 on Amazon (was $199.99) Apply the in-page $20 off coupon for the final price of $179.99 Just like the back massager, this gets a confused thumbs up (due to the cost). However, I cannot rate it through app usage as it failed to connect. As an Amazon Associate, we earn from qualifying purchases.
  • Recent Achievements

    • Dedicated
      Almohandis earned a badge
      Dedicated
    • Dedicated
      JuvenileDelinquent earned a badge
      Dedicated
    • First Post
      DrWankel earned a badge
      First Post
    • Reacting Well
      DrWankel earned a badge
      Reacting Well
    • Week One Done
      Supreme Spray LV earned a badge
      Week One Done
  • Popular Contributors

    1. 1
      +primortal
      505
    2. 2
      +Edouard
      184
    3. 3
      PsYcHoKiLLa
      85
    4. 4
      Michael Scrip
      78
    5. 5
      Steven P.
      76
  • Tell a friend

    Love Neowin? Tell a friend!