Get an Absolute URL with JavaScript

By  on  

Dealing with URL formats can be a real nightmare.  Think of how just a few characters can effect a URL's absolute endpoint:

  • starting or not starting with /
  • starting with //
  • starting with ?
  • starting with #
  • ...and so on

What if you want an absolute URL though?  One that starts with http or https?  You can use an A element to get that absolute URL!

The JavaScript

I'm going to use a function that returns a function so that only one A element is ever created:

var getAbsoluteUrl = (function() {
	var a;

	return function(url) {
		if(!a) a = document.createElement('a');
		a.href = url;

		return a.href;
	};
})();

No matter how you pass in the URL string, the URL will come out absolute.  Of course strings like `javascript:;` wont come out any different, but real qualified URLs will come out as absolute!

Recent Features

  • By
    39 Shirts – Leaving Mozilla

    In 2001 I had just graduated from a small town high school and headed off to a small town college. I found myself in the quaint computer lab where the substandard computers featured two browsers: Internet Explorer and Mozilla. It was this lab where I fell...

  • By
    5 More HTML5 APIs You Didn’t Know Existed

    The HTML5 revolution has provided us some awesome JavaScript and HTML APIs.  Some are APIs we knew we've needed for years, others are cutting edge mobile and desktop helpers.  Regardless of API strength or purpose, anything to help us better do our job is a...

Incredible Demos

  • By
    CSS @supports

    Feature detection via JavaScript is a client side best practice and for all the right reasons, but unfortunately that same functionality hasn't been available within CSS.  What we end up doing is repeating the same properties multiple times with each browser prefix.  Yuck.  Another thing we...

  • By
    WordPress-Style Comment Controls Using MooTools or jQuery

    WordPress has a nice little effect on the Admin Dashboard where it shows and hides the comment control links when you mouseover and mouseout of the record's container. Here's how to achieve that effect using MooTools or jQuery. The XHTML Notice that we place the links into...

Discussion

  1. You can actually even remove the condition, changing the code to

    var getAbsoluteUrl = (function() {
    	var a = document.createElement('a');
    
    	return function(url) {
    		a.href = url;
    
    		return a.href;
    	};
    })();
    
    • True but you’re creating the element even if getAbsoluteUrl is never called.

    • Anees Iqbal

      How about

      var getAbsoluteUrl = (function() {
      	var a = null;
      	return function(url) {
      		a = a || document.createElement('a');
      		a.href = url;
      
      		return a.href;
      	};
      })();
      
    • Adam

      David,

      As you pointed out you create the object whether or not its called, so why not do it this way then:

      var getAbsoluteUrl = function(url) {
        var a = document.createElement('a');
        getAbsoluteUrl = function(url) {
          a.href=url;
          return a.href;
        }
        return getAbsoluteUrl(url);  
      }
      

      The benefit of this approach is that you don’t execute anything unless getAbsoluteUrl is called. Its not a huge improvement, unless you use Immediately Invoked Function Expressions a LOT within your code to create closures.

    • Sean

      I’m confused as to why the second function? Couldn’t it be removed to be:

      var getAbsoluteUrl = function(url) {
        var a = document.createElement('a');
        a.href=url;
        return a.href;
      }
      
    • Adam

      Sean,

      The second function lets you create the anchor once and reuse it. This saves time and memory.

    • Sean

      Thanks Adam, that makes sense :)

  2. Uhhh…
    You could also use the URL() object.
    https://developer.mozilla.org/en-US/docs/Web/API/URL

  3. Hi David. Great idea for a post! To make it better don’t forget that every piece of advice needs a context — so tell the reader where to paste that code — is it in their address bar, do they need to tweet it, or do they need domain specific pre-requisite knowledge to get this to work?

    I’m on a “dumb it down please” mission today :)

  4. So is flexbox, technically. And Selection, and XMLDocument and HTMLPictureElement.

    It’s there. It’s stable and it’s cross browser (IE 10). It also lets you resolve absolute URLs with different domains than the existing page.

  5. Andy Earnshaw

    Patrick, you’re wrong about IE 10 support. That refers to URL.createObjectURL(), not the URL constructor that implements the URLUtils interface. This method works all the way back beyond IE 6, and I’ve been using it for years to parse and resolve URLs. It’s a handy little technique that avoids complex regular expressions, give credit where it’s due!

    • Well, even better then. Nice to know.

    • Do you have a source for that statement? I just tested Patrick’s example in IE 6 and IE 8 and they crashed with “‘URL’ is undefined”.

  6. Yeah or…

    (function(a){
    
    String.implement('toURL', function(){
      a.href = this;
      return this;
    });
    
    })(document.createElement('a'));
    
    • MaxArt

      You should warn that you’re using MooTools, @Olmo.

    • I was being playful with David :P, but if you’d like:

      (function(a){
      
        String.prototype.toURL = function(){
          a.href = this;
          return this;
        };
      
      })(document.createElement('a'));
      
    • Extending the prototype… a big no-no

    • MooTools? In 2015??? Well I never…

    • alberto.c

      Why is bad idea to extend the prototype?

  7. This is super easy! worth trying.

  8. Cant this work?

    location.protocol + "/root/path"
  9. Dale

    None of these suggestions worked. I’m still getting the error 404 – file not found when referencing localhost:4200.

Wrap your code in <pre class="{language}"></pre> tags, link to a GitHub gist, JSFiddle fiddle, or CodePen pen to embed!