283 lines
7.2 KiB
JavaScript
283 lines
7.2 KiB
JavaScript
|
dojo.provide("dojox.data.FlickrStore");
|
||
|
|
||
|
dojo.require("dojo.data.util.simpleFetch");
|
||
|
dojo.require("dojo.io.script");
|
||
|
dojo.require("dojo.date.stamp");
|
||
|
dojo.require("dojo.AdapterRegistry");
|
||
|
|
||
|
(function(){
|
||
|
var d = dojo;
|
||
|
|
||
|
dojo.declare("dojox.data.FlickrStore", null, {
|
||
|
constructor: function(/*Object*/args){
|
||
|
// summary:
|
||
|
// Initializer for the FlickrStore store.
|
||
|
// description:
|
||
|
// The FlickrStore is a Datastore interface to one of the basic services
|
||
|
// of the Flickr service, the public photo feed. This does not provide
|
||
|
// access to all the services of Flickr.
|
||
|
// This store cannot do * and ? filtering as the flickr service
|
||
|
// provides no interface for wildcards.
|
||
|
if(args && args.label){
|
||
|
this.label = args.label;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
_storeRef: "_S",
|
||
|
|
||
|
label: "title",
|
||
|
|
||
|
_assertIsItem: function(/* item */ item){
|
||
|
// summary:
|
||
|
// This function tests whether the item passed in is indeed an item in the store.
|
||
|
// item:
|
||
|
// The item to test for being contained by the store.
|
||
|
if(!this.isItem(item)){
|
||
|
throw new Error("dojox.data.FlickrStore: a function was passed an item argument that was not an item");
|
||
|
}
|
||
|
},
|
||
|
|
||
|
_assertIsAttribute: function(/* attribute-name-string */ attribute){
|
||
|
// summary:
|
||
|
// This function tests whether the item passed in is indeed a valid 'attribute' like type for the store.
|
||
|
// attribute:
|
||
|
// The attribute to test for being contained by the store.
|
||
|
if(typeof attribute !== "string"){
|
||
|
throw new Error("dojox.data.FlickrStore: a function was passed an attribute argument that was not an attribute name string");
|
||
|
}
|
||
|
},
|
||
|
|
||
|
getFeatures: function(){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.getFeatures()
|
||
|
return {
|
||
|
'dojo.data.api.Read': true
|
||
|
};
|
||
|
},
|
||
|
|
||
|
getValue: function(item, attribute, defaultValue){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.getValue()
|
||
|
var values = this.getValues(item, attribute);
|
||
|
if(values && values.length > 0){
|
||
|
return values[0];
|
||
|
}
|
||
|
return defaultValue;
|
||
|
},
|
||
|
|
||
|
getAttributes: function(item){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.getAttributes()
|
||
|
return [
|
||
|
"title", "description", "author", "datePublished", "dateTaken",
|
||
|
"imageUrl", "imageUrlSmall", "imageUrlMedium", "tags", "link"
|
||
|
];
|
||
|
},
|
||
|
|
||
|
hasAttribute: function(item, attribute){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.hasAttributes()
|
||
|
if(this.getValue(item,attribute)){
|
||
|
return true;
|
||
|
}
|
||
|
return false;
|
||
|
},
|
||
|
|
||
|
isItemLoaded: function(item){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.isItemLoaded()
|
||
|
return this.isItem(item);
|
||
|
},
|
||
|
|
||
|
loadItem: function(keywordArgs){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.loadItem()
|
||
|
},
|
||
|
|
||
|
getLabel: function(item){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.getLabel()
|
||
|
return this.getValue(item,this.label);
|
||
|
},
|
||
|
|
||
|
getLabelAttributes: function(item){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.getLabelAttributes()
|
||
|
return [this.label];
|
||
|
},
|
||
|
|
||
|
containsValue: function(item, attribute, value){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.containsValue()
|
||
|
var values = this.getValues(item,attribute);
|
||
|
for(var i = 0; i < values.length; i++){
|
||
|
if(values[i] === value){
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
return false;
|
||
|
},
|
||
|
|
||
|
getValues: function(item, attribute){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.getValue()
|
||
|
|
||
|
this._assertIsItem(item);
|
||
|
this._assertIsAttribute(attribute);
|
||
|
var u = d.hitch(this, "_unescapeHtml");
|
||
|
var s = d.hitch(d.date.stamp, "fromISOString");
|
||
|
switch(attribute){
|
||
|
case "title":
|
||
|
return [ u(item.title) ];
|
||
|
case "author":
|
||
|
return [ u(item.author) ];
|
||
|
case "datePublished":
|
||
|
return [ s(item.published) ];
|
||
|
case "dateTaken":
|
||
|
return [ s(item.date_taken) ];
|
||
|
case "imageUrlSmall":
|
||
|
return [ item.media.m.replace(/_m\./, "_s.") ];
|
||
|
case "imageUrl":
|
||
|
return [ item.media.m.replace(/_m\./, ".") ];
|
||
|
case "imageUrlMedium":
|
||
|
return [ item.media.m ];
|
||
|
case "link":
|
||
|
return [ item.link ];
|
||
|
case "tags":
|
||
|
return item.tags.split(" ");
|
||
|
case "description":
|
||
|
return [ u(item.description) ];
|
||
|
default:
|
||
|
return [];
|
||
|
}
|
||
|
},
|
||
|
|
||
|
isItem: function(item){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.isItem()
|
||
|
if(item && item[this._storeRef] === this){
|
||
|
return true;
|
||
|
}
|
||
|
return false;
|
||
|
},
|
||
|
|
||
|
close: function(request){
|
||
|
// summary:
|
||
|
// See dojo.data.api.Read.close()
|
||
|
},
|
||
|
|
||
|
_fetchItems: function(request, fetchHandler, errorHandler){
|
||
|
// summary:
|
||
|
// Fetch flickr items that match to a query
|
||
|
// request:
|
||
|
// A request object
|
||
|
// fetchHandler:
|
||
|
// A function to call for fetched items
|
||
|
// errorHandler:
|
||
|
// A function to call on error
|
||
|
|
||
|
var rq = request.query = request.query||{};
|
||
|
|
||
|
//Build up the content to send the request for.
|
||
|
var content = {
|
||
|
format: "json",
|
||
|
tagmode:"any"
|
||
|
};
|
||
|
|
||
|
d.forEach(
|
||
|
[ "tags", "tagmode", "lang", "id", "ids" ],
|
||
|
function(i){
|
||
|
if(rq[i]){ content[i] = rq[i]; }
|
||
|
}
|
||
|
);
|
||
|
|
||
|
content.id = rq.id||rq.userid||rq.groupid;
|
||
|
|
||
|
if(rq.userids){
|
||
|
content.ids = rq.userids;
|
||
|
}
|
||
|
|
||
|
//Linking this up to Flickr is a PAIN!
|
||
|
var handle = null;
|
||
|
var getArgs = {
|
||
|
url: dojox.data.FlickrStore.urlRegistry.match(request),
|
||
|
preventCache: true,
|
||
|
content: content
|
||
|
};
|
||
|
var myHandler = d.hitch(this, function(data){
|
||
|
if(!!handle){
|
||
|
d.disconnect(handle);
|
||
|
}
|
||
|
|
||
|
//Process the items...
|
||
|
fetchHandler(this._processFlickrData(data), request);
|
||
|
});
|
||
|
handle = d.connect("jsonFlickrFeed", myHandler);
|
||
|
var deferred = d.io.script.get(getArgs);
|
||
|
|
||
|
//We only set up the errback, because the callback isn't ever really used because we have
|
||
|
//to link to the jsonFlickrFeed function....
|
||
|
deferred.addErrback(function(error){
|
||
|
d.disconnect(handle);
|
||
|
errorHandler(error, request);
|
||
|
});
|
||
|
},
|
||
|
|
||
|
_processFlickrData: function(data){
|
||
|
var items = [];
|
||
|
if(data.items){
|
||
|
items = data.items;
|
||
|
//Add on the store ref so that isItem can work.
|
||
|
for(var i = 0; i < data.items.length; i++){
|
||
|
var item = data.items[i];
|
||
|
item[this._storeRef] = this;
|
||
|
}
|
||
|
}
|
||
|
return items;
|
||
|
},
|
||
|
|
||
|
_unescapeHtml: function(/*String*/ str){
|
||
|
// summary:
|
||
|
// Utility function to un-escape XML special characters in an
|
||
|
// HTML string.
|
||
|
// str: String.
|
||
|
// The string to un-escape
|
||
|
// returns:
|
||
|
// HTML String converted back to the normal text (unescaped)
|
||
|
// characters (<,>,&, ", etc,).
|
||
|
|
||
|
//TODO:
|
||
|
// Check to see if theres already compatible escape() in
|
||
|
// dojo.string or dojo.html
|
||
|
return str.replace(/&/gm, "&").
|
||
|
replace(/</gm, "<").
|
||
|
replace(/>/gm, ">").
|
||
|
replace(/"/gm, "\"").
|
||
|
replace(/'/gm, "'");
|
||
|
}
|
||
|
});
|
||
|
|
||
|
dojo.extend(dojox.data.FlickrStore, dojo.data.util.simpleFetch);
|
||
|
|
||
|
var feedsUrl = "http://api.flickr.com/services/feeds/";
|
||
|
|
||
|
var reg = dojox.data.FlickrStore.urlRegistry = new d.AdapterRegistry(true);
|
||
|
|
||
|
reg.register("group pool",
|
||
|
function(request){ return !!request.query["groupid"]; },
|
||
|
feedsUrl+"groups_pool.gne"
|
||
|
);
|
||
|
|
||
|
reg.register("default",
|
||
|
function(request){ return true; },
|
||
|
feedsUrl+"photos_public.gne"
|
||
|
);
|
||
|
})();
|
||
|
|
||
|
|
||
|
//We have to define this because of how the Flickr API works.
|
||
|
//This somewhat stinks, but what can you do?
|
||
|
if(!jsonFlickrFeed){
|
||
|
var jsonFlickrFeed = function(data){};
|
||
|
}
|