Écrit par Neil Deakin.
Traduit par René-Luc D'Hont (13/06/2005).
Page originale :
http://www.xulplanet.com/tutorials/mozsdk/rdfsources.php
Cette section décrit les datasources (sources de données) RDF dans Mozilla.
Mozilla stocke les informations RDF dans quelque chose appelé un datasource. Chaque datasource contient un ensemble de triplets RDF. Vous pouvez questionner ces datasources pour des triplets RDF en utilisant diverses méthodes. Ceci vous permet de naviguer dans le graphique RDF ou de déterminer les données que le datasource contient. Quelques datasources sont modifiables ainsi on peut ajouter ou enlever des triplets du datasource. Vous pouvez employer les datasources indépendamment les uns des autres, ou vous pouvez les combiner et ainsi former ce qui s'appelle un datasource composite (composite-datasource). Le datasource composite est un groupe de datasources RDF.
Habituellement, chaque datasource contient des données qui sont reliées de différente manière. Par exemple, Mozilla emploie un datasource de signets (bookmarks) qui contient des informations sur les signets de l'utilisateur. Mozilla emploie un certain nombre de datasources contenant diverses informations donc si vous créez une extention de Mozilla, vous pourrez les utiliser. Cependant, vous pouvez également créer vos propres datasources. Mozilla supporte également l'analyse (le parsing) de fichier RDF/XML dans des datasources.
Tous les datasources RDF mettent en application l'interface nsIRDFDataSource. Cette interface fournit des méthodes pour interroger et modifier l'information dans le datasource. Voici les datasources les plus communs que vous emploierez, celles fournis par Mozilla :
Datasource | Description |
---|---|
in-memory-datasource | Ce datasource contient des triplets RDF en mémoire. Plusieurs des autres datasources enveloppent ce datasource. Ce datasource peut être modifié. |
XML-datasource | Ce datasource contient les triplets lus à partir d'un fichier RDF/XML. Il peut maintenir le fichier à partir du quel il a été chargé et il peut sauver des modifications de nouveau dans ce dossier. Ce datasource peut également contenir un RDF/XML chargé à partir d'une URL distante, bien que ceux-ci ne puissent pas être modifiés ou sauvés. Tous les XML-datasources mettent en application l'interface nsIRDFRemoteDataSource, même ceux qui sont chargées à partir des fichiers locaux. |
composite-datasource | Contient une collection d'autres datasources. Quand vous questionnez ce datasource, il questionnera chaque datasource à leur tour. De même, les modifications sont proposées à chaque datasource jusqu'à ce qu'un d'eux accepte le changement. Ces datasources mettent également en application l'interface nsIRDFCompositeDataSource. |
Il y a également nombreux autres datasources, mais ceux-ci contiennent des données spécifiques à l'application de Mozilla.
Mozilla emploie un service RDF qui est responsable de l'obtention des
datasources RDF. Le service renvoie également d'autres objets RDF
relationnels tels que les objets de type ressource et literal. Les
datasources sont identifiés par une URI. Si vous employez une URI, ceci
aura habituellement comme conséquence la création d'un
xml-datasource
qui chargera le RDF/XML de cette URI.
Mozilla fournit un certain nombre de datasources additionnels qui peuvent
être récupérés en employant une URI qui commence par 'rdf:'. Par exemple,
l'Uri rdf:history
renverra le datasource d'historique qui
stocke les données liées à l'historique de la navigation de l'utilisateur.
Quand vous demandez un datasource de cette forme, c.-à-d., un datasource
avec une URI qui commence par 'rdf:', Mozilla recherchera un composant qui
manipule ce datasource. Il trouve ce composant en prenant la partie de l'Uri
après le préfixe et l'appose à la chaîne de caractère
@mozilla.org/rdf/datasource;1?name=
pour former un nom composé.
Par exemple, l'Uri rdf:bookmarks
, donnera le composant
@mozilla.org/rdf/datasource;1?name=bookmarks
.
Tous les datasources fonctionnent de cette façon. Si vous créez un composant
fait sur commande en utilisant cette convention, vous pourrez rechercher le
datasource en utilisant le service RDF. Vous pouvez également employer un
datasource spécifique dans une template (calibre) en plaçant l'Uri sur
l'attribut datasources
d'un élément. Les
datasources 'rdf:' ont habituellement des méthodes de manipulation du
datasource spéciale, ou stocke les données d'une façon particulière.
Par exemple, le datasource d'historique stocke des données dans un format
fait sur mesure et emploie du code pour envelopper les données de manière
qu'elles puissent être interrogé en utilisant les APIs de datasource RDF.
Le datasource des signets stocke les données sur le disque dans un format
fait sur mesure, mais utilise un in-memory-datasource
pour
stocker les données dans la mémoire.
Le service RDF a deux méthodes pour rechercher des datasources. La première
méthode, GetDataSource
est employée pour charger
un datasource de façon asynchrone. Cette fonction peut retourner une réponse
avant que le datasource ait été chargé, sachant que les datasources qui sont
déjà chargés sont disponibles tout de suite. Les datasources 'rdf:' sont
habituellement disponibles immédiatement, car le navigateur se servira déjà
d'eux. Encore que les datasources 'rdf:' font habituellement n'importe quelle
initialisation tout de suite. Ceci signifie que la méthode
GetDataSource
convient au chargement de ce genre
de datasources. Vous devriez également employer cette méthode pour charger
les fichiers RDF/XML distants de de sorte qu'ils puissent être recherchés en
tache de fond. Une méthode sera expliquée plus tard pour déterminer quand le
chargement est complet.
La deuxième méthode, GetDataSourceBlocking
obtient un datasource et attend qu'il soit chargé. Cette méthode peut être
employée pour les fichiers RDF/XML locaux. Cette méthode retournera une
réponse une fois que le datasource aura été chargé et analysé. Cette méthode
ne fonctionne pas actuellement pour les fichiers distants.
Les deux méthodes prennent un argument simple, l'Uri absolu du datasource
à charger, et toutes les deux elles renvoient le datasource. Dans le cas de
la fonction GetDataSource
, le datasource est
retourné mais il peut encore ne contenir aucune données. Voici quelques
exemples des deux fonctions.
var rdfService = Components.classes["@mozilla.org/rdf/rdf-service;1"].
getService(Components.interfaces.nsIRDFService);
var historyDS = rdfService.GetDataSource("rdf:history");
var fileDS = rdfService.GetDataSourceBlocking("file:///somedir/somefile.rdf");
D'abord, nous obtenons le service RDF. Puisque c'est un service, nous employons getService au lieu de createInstance. La ligne suivante recupère le datasource d'historique en utilisant la fonction GetDataSource. En conclusion, nous recupérons un fichier RDF/XML en utilisant la fonction GetDataSourceBlocking, qui peut être interogée et modifiée immédiatement. Chacun des datasources retournés implémente l'interface nsIRDFDataSource.
Pour les datasources 'rdf:', les fonctions
GetDataSource
et
GetDataSourceBlocking
créent le nouveau datasource
comme un service. Cela signifie que seulement une copie du datasource existe
à la fois. Faites attention à ceci si vous exploitez vos propres datasources.
Si vous voulez qu'un datasource différent soit créé à chaque fois, vous devrez
créer le composant de la manière habituelle de XPCOM, avec la méthode
createInstance
. Par exemple, pour créer un
nouveau in-memory-datasource
, faites ce qui suit :
var inmemds = Components.classes["@mozilla.org/rdf/datasource;1?name=in-memory-datasource"]
.createInstance(Components.interfaces.nsIRDFDataSource);
C'est nécessaire à partir du moment où vous ne voulez pas employer les mêmes données en mémoire à chaque fois.
Le service RDF met en cache les datasources qui ont été chargés. Cela
signifie que quand vous essayez de rechercher le même URI en employant encore
les fonctions GetDataSource
ou
GetDataSourceBlocking
, le service RDF renverra
le même objet. Ceci évite de créer et charger le même datasource à plusieurs
reprises, à chaque fois qu'il est mentionné. En outre, cela signifie que vous
pouvez appeler avec le service RDF le même datasource plusieurs fois et
récupérer le même objet à chaque fois.
Cette description, au-dessus, n'est cependant pas techniquement exact sur
le fonctionnement du cache. Quand la fonction
GetDataSource
ou
GetDataSourceBlocking
est appellé, ces fonctions
regardent dans le cache du service RDF les datasources déjà utilisés. Si
l'Uri est trouvée dans le cache, ce datasource est retourné. Si l'Uri n'est
pas trouvée, un nouveau datasource du type approprié est créé, comme décrit
ci-dessus. Le service RDF n'ajoute cependant pas le datasource nouvellement
créé au cache. C'est de la responsabilité du datasource lui-même, en appelant
la méthode RegisterDataSource
du service RDF.
Cette méthode ajoutera le datasource au cache du service RDF. Si le datasource
n'appelle jamais cette méthode, le datasource n'apparaîtra jamais dans le
cache.
Ceci créera un nouveau datasource à chaque fois. D'autres datasources
appelleront RegisterDataSource
et ainsi un
nouvel objet n'est pas créé à chaque fois. Quand
RegisterDataSource
est appellé, le service RDF
interroge le datasource sur son URI et la stocke dans le cache. Notez qu'il
est possible qu'un datasource renvoie une URI différente de celle employée
pour le créer, ce qui offre quelques possibilités intéressantes. Le datasource
xml-datasource
appelle la méthode
RegisterDataSource
, mais l'URL sera l'URL du
fichier RDF/XML. Un datasource peut appeler la méthode
RegisterDataSource
à tout moment, ainsi il est
possible de mettre en cache un datasource à l'avance.
La méthode correspondante à UnregisterDataSource
du service RDF est employée pour enlever des datasources du cache. Celle-ci
normalement sera appelée par le destructeur des datasource, c.-à-d., quand le
datasource est supprimé. Un datasource sera supprimé quand désormais plus
aucune référence à lui sera faîte. Quand il est supprimé, le datasource peut
être désenregistrer avec le service RDF, et il sera enlevé du cache. La
prochaine fois que l'URL est demandé, le datasource devra être créé une nouvelle
fois. Tandis qu'il est possible d'enlever un datasource du cache du service
RDF à tout moment en employant UnregisterDataSource
,
le datasource et ses données continueront à exister jusqu'à ce qu'il n'y aient
plus de références faites à lui.
Les éléments XUL peuvent être associés à un datasource. Tous les éléments XUL
dans un document XUL peuvent avoir un datasource, bien qu'il soit aussi possible
à d'autres types d'éléments d'être liés à des datasources s'ils sont placés dans
un document XUL. Les éléments qui ne sont pas dans un document XUL ne peuvent
pas employer de template. Le datasource associé à un élément peut être récupéré
en obtenant la valeur de la propriété database
de l'élément. La base de données est toujours un composite-datasource
qui contient un certain nombre de datasources RDF. Elle peut contenir un grand
nombre de datasources et peut même n'en contenir aucun, et l'ensemble des
sources peuvent être modifiées en employant les méthodes de l'interface
nsIRDFCompositeDataSource.
La plupart des éléments n'auront pas de datasource associé ainsi la propriété
database
sera nulle. Vous pouvez indiquer que vous
voulez qu'un élément ait une base de données en ajoutant l'attribut datasources
à un élément. La valeur de cet attribut
est une liste d'URIs des datasources initiaux à ajouter à la base de données
séparés par un espace. Tandis que vous pouvez changer les datasources en
employant les méthodes de l'interface nsIRDFCompositeDataSource
,
l'attribut datasources
représente seulement les
datasources initiaux à employer. Changer la valeur de l'attribut ne change pas
les datasources dans la base de données.
Par exemple, dans l'exemple ci-dessous, deux datasources sont assignés comme
datasources initiaux à associer à l'élément tree
.
<tree datasources="rdf:bookmarks animals.rdf">
Ceci entrainera l'association d'un
composite-datasource
à l'arbre contenant les deux
datasources indiqués. Le premier datasource est rdf:bookmarks
qui est le datasource de Mozilla utilisés
pour stocker les signets de l'utilisateur. Le deuxième datasource est manié en
tant qu'URI d'un fichier RDF/XML, dans ce cas-ci relativement au fichier XUL.
Dans cet exemple, le datasource de signets (bookmarks) serait seulement employé si le code est privilégié, ce qui signifie habituellement qu'il appartient à une application chrome. Le code non privilégié ne pourra pas accéder au datasource de signets.
Une différence additionnelle entre le code privilégié et non privilégié est
que pour le code privilégié, le datasource
rdf:local-store
est toujours inclus dans la liste
des datasources indépendamment du fait qu'on l'ait indiqué ou pas. Ce datasource
est normalement employé pour contenir quelques informations diverses d'état tel
que les tailles de fenêtre, quel toolbar est visible, les tailles des colonnes
d'arbre, et ainsi de suite. Ceci signifie que la base de données dans l'exemple
ci-dessus aura réellement trois datasources, en premier le magasin local (local
store), en deuxième les signets et en troisième le fichier animals.rdf. Le code
non privilégié fonctionnant sur un site Web distant n'inclut pas ce datasource,
puisqu'il contient des information sur l'utilisateur.
Une fois qu'une base de données a été appliquée à un élément, elle n'est pas
enlevée jusqu'à ce que le document soit détruit, ce qui se produira lorsque
l'utilisateur activera une autre page ou fermera la fenêtre. Ainsi, en enlevant
un élément puis en l'ajoutant de nouveau au document la même base de données
se maintiendra. Si un élément avec un attribut
datasources
qui n'était pas dans le document avant
qu'il n'y soit inséré, alors une nouvelle base de données sera créé pour lui.
Il est normal d'employer un calibre ou template à l'intérieur de l'élément
associé à une base de données, bien que ce ne soit pas strictement nécessaire.
Le calibre doit soit être un élément template
enfant
direct de l'élément avec l'attribut datasources
,
soit y faire référence en utilisant un attribut
template
. Dans le dernier cas, la valeur de
l'attribut devra être la valeur de l'attribut id
d'un élément template
quelque part dans le document.
Ceci permet à un calibre simple d'être employé dans plusieurs endroits, bien
que cette utilisation ne soit pas commune.
Un constructeur de template sera employé pour construire le contenu réel du calibre. Il emploiera le template pour construire les nouveaux noeuds du DOM qui seront insérés au document. Si les données d'un datasource associées à un élément changent, le constructeur de template régénérera les données, ajoutant, enlevant ou changeant le contenu selon les besoins. Le constructeur de template manipule ceci par l'enregistrement dans un observateur du datasource, de sorte qu'il soit avisé des changements.
La reconstruction automatique du contenu du calibre se produit seulement dans deux situations. La première quand le datasource RDF fondamental change. Quand le RDF change, le constructeur de template sera avisé du changement. Le constructeur de template reconstruira les parties du contenu qui seront affectées par le changement, mais ne change pas les parties du contenu qui ne seront pas affectées. Cette reconstruction automatique se produit si une opération d'insertion, de suppression ou de changement se produit dans le datasource, mais pas sur une opération de mouvement. Le constructeur de calibre reconstruira également automatiquement après une opération en batch (en lot) quand la méthode onEndUpdateBatch est appelé.
La deuxième situation qui déclenchera une reconstruction automatique de
contenu du calibre est quand l'attribut ref
est
modifié sur l'élément externe associé à la base de données. Ceci entraîne la
reconstruction complète du contenu. Notez que la valeur doit être une valeur
différente. Le réglage de l'attribut ref
à la
même valeur n'entraîne pas de reconstruction. Faites attention à ceci, comme
précédemment Mozilla construit et fait reconstruire même quand la valeur est
simplement placée à la même valeur.
Toutes autres modifications ne déclenchent pas une reconstruction.
Spécifiquement, ajouter ou enlever des datasources, changer l'attribut
datasources
, ou modifier le contenu du calibre,
n'entraînent pas une reconstruction automatique. En cas de doute, faites une
reconstruction manuelle.
La reconstruction peut être exécutée manuellement en appelant la méthode
rebuild
du constructeur de template. Tous les
éléments XUL ont une propriété builder
qui est
une référence au constructeur de calibre lié à l'élément. S'il n'y a aucun
constructeur de template lié à un élément, la valeur de cette propriété sera
nulle. Le code suivant est employé pour reconstruire le contenu d'un calibre,
où element
est l'élément associé à la base de
données :
element.builder.rebuild();
Changer l'attribut ref
a pour conséquence
l'exécution du même code fondamental que pour la reconstruction manuelle,
sauf que le nouveau noeud racine est employé pour établir le nouveau contenu.
Le constructeur de calibre créera souvent paresseusement le contenu. Cela signifie qu'il créera seulement le contenu réel quand il sera nécessaire de le montrer. Par exemple, le contenu d'un menu n'est pas produit tant que le menu n'a pas été ouvert une fois. De même pour des noeuds enfant dans le contenu d'un arbre qui ne sont pas créés tant que le noeud parent n'a pas été ouvert. C'est quelque chose à prendre en compte quand vous examinerez le DOM de l'arbre résultant.