Après un mois d’absence, me voici de retour avec de nouvelles astuces pour Firefox OS :
- Quels sont les comptes Twitter à suivre ?
- Comment faire de l’OAuth dans une application Firefox OS ?
- WebAPI : WebActivities
Les évangélistes Firefox OS
Angelina Fabbro est une techno évangéliste pour Mozilla (Vidéo d’introduction). Elle écrit de nombreux articles sur Firefox OS sur MozHacks. Elle est spécialisée en JavaScript.
Follow @angelinamagnum
Frédéric Harper est un techno évangéliste pour Mozilla (Vidéo d’introduction).
Follow @fharper
Jason Weathersby est un techno évangéliste pour Mozilla (Vidéo d’introduction)
Follow @JasonWeathersby
Autre techno évangéliste chez Mozilla, Robert Nyman écrit aussi des articles sur la blog de MozHacks avec une expertise en HTML5.
Follow @robertnyman
Les modérateurs Firefox OS
Adora a.k.a. Lisa Brewster est une « app reviewer » pour le Firefox Marketplace.
Follow @Adora
@FirefoxOSFeed est un fil Twitter regroupant les dernières infos sur Firefox OS. Il fait partie des must-have.
Follow @FirefoxOSFeed
@AllFirefoxOS est un fil Twitter lié au site allfirefoxos.com.
Follow @AllFirefoxOS
@FirefoxApps est le compte Twitter officiel du Firefox Marketplace.
Follow @FirefoxApps
@TheFirefoxos est un fil Twitter lié au site thefirefoxos.com.
Follow @thefirefoxos
Pour vous montrer comment faire de l’OAuth avec une application Firefox OS, je vais prendre l’exemple d’une application Facebook.
Créer une application Facebook
Pour cela, il faut aller sur le site Developers FB. A ce niveau là, vous créez une nouvelle application avec le nom souhaité. Quand celle ci est créé, vous n’avez plus rien à modifier. Vous n’avez à récupérer que son Application ID.
Créer l’application Firefox OS
Le fichier manifest.webapp
Tout d’abord, il faut un fichier manifest.webapp :
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
| {
"name": "FacebookAPI",
"description": "FacebookAPI",
"launch_path": "/index.html",
"developer": {
"name": "Progi1984",
"url": "https://lefevre.dev"
},
"icons": {
"128": "/img/default.png"
},
"permissions": {
"systemXHR": { "description":"Used to send data to Facebook API." },
"browser": { "description":"Enables authentication to Facebook via OAuth." }
},
"type":"privileged"
}
|
On lui donne accès à deux permissions pour qu’il puisse lancer l’authentification vers Facebook dans un navigateur, et lancer des requêtes AJAX vers Facebook.
Le fichier index.html
Puis on lui associe un fichier index.html qui sera le coeur de notre application.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
| <html>
<body>
<input id="btnLogin" value="Login" type="button" />
<input id="btnLogout" value="Logout" type="button" />
<iframe mozbrowser remote id='browser' style="width:100%;height:100%;overflow:true;"></iframe>
<script>
var gAccessToken = 0;
/**
* Utility function to parse OAuth authorization token out of a URL.
*/
function parseTokens(psUrl) {
if(psUrl.lastIndexOf('?') == -1){
var psUrl = psUrl.slice(psUrl.lastIndexOf('#') + 1);
} else {
var psUrl = psUrl.slice(psUrl.lastIndexOf('?') + 1);
}
var result = {};
psUrl.split('&').forEach(function(parts) {
parts = parts.split('=');
console.log(parts);
result[parts[0]] = parts[1];
});
return result;
}
// INIT
document.getElementById('btnLogout').style.display = 'none';
document.getElementById('browser').style.display = 'none';
// LOGIN
function fnClick_btnLogin(){
console.log('fnClick_btnLogin');
sURLRedirect = 'https://www.facebook.com/connect/login_success.html';
sURLAuth = 'https://www.facebook.com/dialog/oauth'
+ '?client_id=XXXXXXXXXXXXXX' /* Remplacer la série de X par l'Application ID */
+ '&redirect_uri=' + sURLRedirect
+ '&response_type=token'
+ '&scope=email';
+ '&display=touch';
document.getElementById('browser').addEventListener('mozbrowserlocationchange', function(e) {
if (e.detail && (e.detail.indexOf(sURLRedirect) === 0)) {
var result = parseTokens(e.detail);
var tokens = JSON.stringify(result);
console.log('Facebook Access Token : ' + result['access_token']);
gAccessToken = result['access_token'];
document.getElementById('browser').style.display = 'none';
document.getElementById('btnLogout').style.display = 'block';
}
});
document.getElementById('browser').setAttribute('src', sURLAuth);
document.getElementById('browser').style.display = 'block';
document.getElementById('btnLogin').style.display = 'none';
}
document.getElementById('btnLogin').addEventListener('click', fnClick_btnLogin, false);
// LOGOUT
function fnClick_btnLogout(){
console.log('fnClick_btnLogout');
console.log(gAccessToken);
sURLRedirect = 'https://www.facebook.com/connect/login_success.html';
document.getElementById('browser').addEventListener('mozbrowserlocationchange', function(e) {
if (e.detail && (e.detail.indexOf(sURLRedirect) === 0)) {
var result = parseTokens(e.detail);
var tokens = JSON.stringify(result);
console.log('Facebook Access Token : ' + result['access_token']);
document.getElementById('browser').style.display = 'none';
document.getElementById('btnLogin').style.display = 'block';
}
});
document.getElementById('browser').setAttribute('src', 'https://www.facebook.com/logout.php?next='+sURLRedirect+'&access_token=' + gAccessToken);
document.getElementById('browser').style.display = 'block';
document.getElementById('btnLogout').style.display = 'none';
}
document.getElementById('btnLogout').addEventListener('click', fnClick_btnLogout, false);
</script>
</body>
</html>
|
Notre page web est composé de deux boutons : Login & Logout et d’une iframe.
Par défaut, je cache le bouton Logout et l’iframe.
Quand je clique sur le bouton Login, j’affiche l’iframe et lui définit comme adresse une URL d’authentification qui, après authentification de l’utilisateur, me retournera dans la nouvelle URL un paramètre access_token utilisé pour mes différents appels API.
Pour récupérer cette nouvelle URL, j’ajoute un listener ‘mozbrowserlocationchange’ à l’iframe. Celui-ci interceptera tous les changements d’URL pour cet élément HTML.
La technique utilisée pour faire de l’OAuth dans une application Firefox OS est identique à celle utilisée pour les applications de type Desktop ayant un accès au Web mais pas d’URL publique.
Cette technique est utilisée par la plupart des fournisseurs d’authentification OAuth :
WebAPI : WebActivities
Les WebActivities permettent de lier les différentes applications ensemble. Ainsi, l’application Twitter va offrir une WebActivity Share pour partager sur Twitter ou l’application Contacts va offrir une WebActivity Webcontacts/Contact pour créer rapidement un contact.
Lier son application à une WebActivity
Il faut d’abord définir dans le fichier manifest.webapp les WebActivities que fournira votre application.
Dans l’exemple ci-dessous, on voit que l’application permettra de visualiser les images de type GIF ou PNG et que ce sera le fichier image_view.html de l’application qui gérera.
1
2
3
4
5
6
7
8
9
| "activities": {
"view": {
"filters": {
type: ["image/png", "image/gif"],
},
"href": "image_view.html",
"disposition": "window"
}
}
|
Dans le fichier HTML, on a d’abord l’enregistrement du fournisseur de WebActivity :
1
2
3
4
5
6
7
8
9
10
11
| var oActivityHandler = navigator.mozRegisterActivityHandler({
name: "view",
disposition: "inline",
filters: {
type: ["image/png", "image/gif"]
}
});
oActivityHandler.onerror = function () {
console.log('Echec de l\'enregistrement du fournisseur de WebActivity view');
}
|
Ensuite, on gère la WebActivity :
1
2
3
4
5
| navigator.mozSetMessageHandler("activity", function (a) {
var img = getImageObject();
img.src = a.source.url;
// Appelle a.postResult() ou a.postError() si la WebActivity doit retourner une valeur
});
|
Et voilà, votre application est désormais capable de gérer la visualisation d’images pour l’ensemble des applications de votre téléphone.
Utiliser une WebActivity dans son application
De nombreuses WebActivities sont fournies par défaut. En voici quelques unes :
Appeler quelqu’un :
1
2
3
4
5
6
| var oMozActivity = new MozActivity({
name: "dial",
data: {
number: "+33600000000"
}
});
|
Envoyer un SMS :
1
2
3
4
5
6
7
| var oMozActivity = new MozActivity({
name: "new",
data: {
type: "websms/sms",
number: "+33600000000"
}
});
|
Enregistrer un contact :
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
| var oMozActivity = new MozActivity({
name: "new",
data: {
type: "webcontacts/contact",
params: {
giveName: "givenName",
familyName: "FamilyName",
tel: "+33600000000",
email: "email@fai.tld",
address: "Adress",
note: "Note",
company: "Company"
}
}
});
|
Visualiser une URL :
1
2
3
4
5
6
7
| var oMozActivity = new MozActivity({
name: "view",
data: {
type: "url",
url: "http://url.tld"
}
});
|
Enregistrer un favori :
1
2
3
4
5
6
7
8
9
| var oMozActivity = new MozActivity({
name: "save-bookmark",
data: {
type: "url",
url: "http://url.tld",
name: "Bookmark Name",
icon: "http://url.tld/favicon.png"
}
});
|
Choisir une image :
1
2
3
4
5
6
| var oMozActivity = new MozActivity({
name: "pick",
data: {
type: ["image/png", "image/jpg", "image/jpeg"]
}
});
|
D’autres WebActivities sont disponibles :
- configure
- costcontrol/balance
- costcontrol/data_usage
- costcontrol/telephony
- open
- record
- share
- test
Il faut savoir ensuite gérer le retour de cette WebActivity (est ce qu’il a bien retourné ce que je voulais ? est ce qu’il y a eu une erreur ?).
Nous allons prendre le cas de la WebActivity ‘pick’ :
1
2
3
4
5
6
7
8
9
10
11
12
13
14
| oMozActivity.onsuccess = function () {
// Créee une image et définit le blob retourné comme source de l'image
var img = document.createElement("img");
img.src = window.URL.createObjectURL(this.result.blob);
// Affiche l'image dans votre application
var imagePresenter = document.querySelector("#image-presenter");
imagePresenter.appendChild(img);
};
oMozActivity.onerror = function () {
// Si une erreur est apparu ou que l'utilisateur a annulé l'opération
alert('Erreur avec la MozActivity pick');
};
|
Liens :