HTML5: Network Information API

Introduction

One of the most discussed concepts in the world of the web design is responsive web design. Thousands of articles have been written on responsive web design and I therefore won’t discuss it in this article. However, responsive web design has an important limitation, responsive web design is for the most part based on the size of the browser’s viewport.

While this approach works well for serving up images of the right size and resolution, it isn’t ideal in all situations, video content being one example. What we really need in these cases is more information about the device’s network connection.

Imagine you’re visiting YouTube on your smartphone or tablet. You’re at home and connected over Wi-Fi. In such cases, you don’t care about the number of bytes that are being downloaded, you’re only interested in high quality video content. This isn’t true if you’re connected over a slow mobile connection. In that case, you want to see the video, the quality is secondary.

I should be clear that every developer who wants a website to be really good still has to optimize the assets it serves to allow pages to load as fast as possible. However, in the above example, serving up a high resolution video isn’t a waste of the user’s bandwidth, but an improvement of the user experience.

The Network Information API is exactly what we need to find out more about the network connection of the device.

1.

What is it?

The Network Information API provides access to the connection type the system is using to communicate with the network, cellular, Wi-Fi, Bluetooth, etc. It also provides a means for scripts to be notified if the connection type changes. This is to allow developers to make dynamic changes to the DOM and/or inform the user that the network connection type has changed.

The first release of the specification of the Network Information API was in 2011, but the API has changed several times since. As a proof of this, the current version is a mere editor’s draft, which means that it’s bound to change in the future.

Despite the changes, the use cases for this API are so interesting that it’s really worth exploring it. In this article, we’ll discuss the latest version of the specification, but we’ll also look at some of the deprecated properties and events for reasons I’ll explain later.

2.

Implementation

The current version of the Network Information API exposes a connection object that belongs to the window.navigator object. The connection object contains one property, type, which returns the user agent’s connection type. The type property can have one of the following values:

  • bluetooth
  • cellular
  • ethernet
  • none
  • wifi
  • other
  • unknown

Some of these values, such as bluetooth and wifi, are self-explanatory while others need a bit more explaining. The cellular type refers to a mobile connection, such as EDGE, 3G, 4G, etc. The other type means that the current connection type is not unknown, but it isn’t any of the other types either. The unknown type means that the user agent has established a network connection, but it is unable to determine what the connection type is.

In addition to type, the Network Information API exposes the ontypechange event. It is fired every time the type of the network connection changes.

Developers can use the Network Information API to change some features based on the current connection type. For example, we can slow down a process that takes up significant bandwidth if we detect the device is using a mobile connection. The API also lets us assign a specific class to the html element, for example high-bandwidth, in the same way Modernizr does. We can leverage CSS to change one or more properties of an element, such as the background image.

Now that we know what the Network Information API does and who we can benefit form it, let’s see which browsers support the API.

3.

Browser Support

At the time of writing, the Network Information API is only supported by Firefox, using its vendor prefix, and Chrome Canary. In Chrome Canary, we have to enable the experimental web platform features flag to use the API. You can find more information in this post by Paul Irish.

As if support for the Network Information API wasn’t already poor enough, Firefox up to version 30, the most recent version, supports the old API specification. Because of this and in case you want to play with the Network Information API right now, it’s important to take a look at the properties and events exposed by the previous specification of the API.

The old specification exposed two properties, bandwidth and metered. The bandwidth property is a double representing an estimation of the current bandwidth in megabytes per second (MB/s). The metered property is a boolean that specifies if the the network connection of the device is subject to any limitations. The previous specification also defined a onchange event to notify any listeners about changes of the aforementioned properties.

To give you an idea of the new and the old version of the specification, in the next section we’ll build a demo that uses both.

4.

Demo

So far, we’ve covered the properties and the events exposed by the API as well as the API’s use cases. In this section, we’re going to create a simple web page to see the API in action.

The demo consists of an HTML5 page that has an unordered list with three list items. Each item contains a text snippet to verify the value of the properties exposed by the old and the new specification of the Network Information API. The list items are hidden by default and are only shown if the corresponding properties are supported.

The demo also detects if the browser implements the old specification of the API (to target Firefox) and if the browser supports the Network Information API at all. In the first case, you’ll see the message Old API version supported, in the second case the message API not supported will be displayed.

Testing for support of the Network Information API is very simple as you can see below:

1

// Deal with vendor prefixes

2

var

connection

=

window

.

navigator

.

connection

||

3
                 

window

.

navigator

.

mozConnection

||

4
                 

null

;

5

if

(

connection

===

null

)

{

6
   

// API not supported :(

7

}

else

{

8
   

// API supported! Let's start the fun :)

9

}

To detect if the version implemented is the old specification, we can test for the presence of the metered property as shown below:

1

if

(

'

metered

'

in

connection

)

{

2
   

// Old version

3

}

else

{

4
   

// New version

5

}

Once we’ve tested for support of the Network Information API and we’ve figured out which version of the specification the browser supports, we can attach a handler to the correct event. Inside the handler we update the text of the corresponding list item, for example type for the new API specification.

You can find the complete code of the demo below and you can also play with it if you like.

1

<!DOCTYPE html>

2

<html>

3
   

<head>

4
      

<meta

charset=

"UTF-8"

>

5
      

<meta

name=

"viewport"

content=

"width=device-width, initial-scale=1.0"

/>

6
      

<meta

name=

"author"

content=

"Aurelio De Rosa"

>

7
      

<title>

Network Information API Demo by Aurelio De Rosa

</title>

8
      

<style>

9
         

*

10
         

{

11
            

-webkit-box-sizing

:

border-box

;

12
            

-moz-box-sizing

:

border-box

;

13
            

box-sizing

:

border-box

;

14
         

}

15

16
         

body

17
         

{

18
            

max-width

:

500px

;

19
            

margin

:

2em

auto

;

20
            

padding

:

0

0.5em

;

21
            

font-size

:

20px

;

22
         

}

23

24
         

h1

25
         

{

26
            

text-align

:

center

;

27
         

}

28

29
         

.api-support

30
         

{

31
            

display

:

block

;

32
         

}

33

34
         

.hidden

35
         

{

36
            

display

:

none

;

37
         

}

38

39
         

.value

40
         

{

41
            

font-weight

:

bold

;

42
         

}

43

44
         

.author

45
         

{

46
            

display

:

block

;

47
            

margin-top

:

1em

;

48
         

}

49
      

</style>

50
   

</head>

51
   

<body>

52
      

<h1>

Network Information API

</h1>

53

54
      

<span

id=

"ni-unsupported"

class=

"api-support hidden"

>

API not supported

</span>

55
      

<span

id=

"nio-supported"

class=

"api-support hidden"

>

Old API version supported

</span>

56

57
      

<ul>

58
         

<li

class=

"new-api hidden"

>

59
            The connection type is 

<span

id=

"t-value"

class=

"value"

>

undefined

</span>

.
60
         

</li>

61
         

<li

class=

"old-api hidden"

>

62
            The connection bandwidth is 

<span

id=

"b-value"

class=

"value"

>

undefined

</span>

.
63
         

</li>

64
         

<li

class=

"old-api hidden"

>

65
            The connection is 

<span

id=

"m-value"

class=

"value"

>

undefined

</span>

.
66
         

</li>

67
      

</ul>

68

69
      

<small

class=

"author"

>

70
         Demo created by 

<a

href=

"http://www.audero.it"

>

Aurelio De Rosa

</a>

71
         (

<a

href=

"https://twitter.com/AurelioDeRosa"

>

@AurelioDeRosa

</a>

).

<br

/>

72
         This demo is part of the 

<a

href=

"https://github.com/AurelioDeRosa/HTML5-API-demos"

>

HTML5 API demos repository

</a>

.
73
      

</small>

74

75
      

<script>

76
         

var

connection

=

window

.

navigator

.

connection

||

77
                          

window

.

navigator

.

mozConnection

||

78
                          

null

;

79
         

if

(

connection

===

null

)

{

80
            

document

.

getElementById

(

'

ni-unsupported

'

).

classList

.

remove

(

'

hidden

'

);

81
         

}

else

if

(

'

metered

'

in

connection

)

{

82
            

document

.

getElementById

(

'

nio-supported

'

).

classList

.

remove

(

'

hidden

'

);

83
            

[].

slice

.

call

(

document

.

getElementsByClassName

(

'

old-api

'

)).

forEach

(

function

(

element

)

{

84
               

element

.

classList

.

remove

(

'

hidden

'

);

85
            

});

86

87
            

var

bandwidthValue

=

document

.

getElementById

(

'

b-value

'

);

88
            

var

meteredValue

=

document

.

getElementById

(

'

m-value

'

);

89

90
            

connection

.

addEventListener

(

'

change

'

,

function

(

event

)

{

91
               

bandwidthValue

.

innerHTML

=

connection

.

bandwidth

;

92
               

meteredValue

.

innerHTML

=

(

connection

.

metered

?

''

:

'

not

'

)

+

'

metered

'

;

93
            

});

94
            

connection

.

dispatchEvent

(

new

Event

(

'

change

'

));

95
         

}

else

{

96
            

var

typeValue

=

document

.

getElementById

(

'

t-value

'

);

97
            

[].

slice

.

call

(

document

.

getElementsByClassName

(

'

new-api

'

)).

forEach

(

function

(

element

)

{

98
               

element

.

classList

.

remove

(

'

hidden

'

);

99
            

});

100

101
            

connection

.

addEventListener

(

'

typechange

'

,

function

(

event

)

{

102
               

typeValue

.

innerHTML

=

connection

.

type

;

103
            

});

104
            

connection

.

dispatchEvent

(

new

Event

(

'

typechange

'

));

105
         

}

106
      

</script>

107
   

</body>

108

</html>

Conclusion

In this article, I introduced you to the Network Information API. In the first part of this article, we’ve discussed what the API is and what it can do for us. We also learned what properties and events the Network Information API exposes. As I mentioned in Browser Support, the API is currently poorly supported, but this is in part due to several changes of the API’s specification.

The Network Information API is very simple to use and there are no excuses to not take advantage of the information it offers once it’s supported by more browsers. What do you think of this API? Will you use it when it’s supported by more browsers?