timeline view minor enhancements
* support dates < 1900 by using logilab.common.date.ustrftime
* make the timeline a public member of the TimeLineWidget
* let buildWidget return the created widget, so that we can manipulate it later
/** * Functions dedicated to widgets. * * :organization: Logilab * :copyright: 2003-2010 LOGILAB S.A. (Paris, FRANCE), all rights reserved. * :contact: http://www.logilab.fr/ -- mailto:contact@logilab.fr * * */// widget namespaceWidgets={};/** * .. function:: buildWidget(wdgnode) * * this function takes a DOM node defining a widget and * instantiates / builds the appropriate widget class */functionbuildWidget(wdgnode){varwdgclass=Widgets[wdgnode.getAttribute('cubicweb:wdgtype')];if(wdgclass){returnnewwdgclass(wdgnode);}returnnull;}/** * .. function:: buildWidgets(root) * * This function is called on load and is in charge to build * JS widgets according to DOM nodes found in the page */functionbuildWidgets(root){root=root||document;jQuery(root).find('.widget').each(function(){if(this.getAttribute('cubicweb:loadtype')=='auto'){buildWidget(this);}});}// we need to differenciate cases where initFacetBoxEvents is called// with one argument or without any argument. If we use `initFacetBoxEvents`// as the direct callback on the jQuery.ready event, jQuery will pass some argument// of his, so we use this small anonymous function instead.jQuery(document).ready(function(){buildWidgets();});functionpostJSON(url,data,callback){returnjQuery.post(url,data,callback,'json');}functiongetJSON(url,data,callback){returnjQuery.get(url,data,callback,'json');}Widgets.SuggestField=defclass('SuggestField',null,{__init__:function(node,options){varmulti=node.getAttribute('cubicweb:multi')||"no";options=options||{};options.multiple=(multi=="yes")?true:false;vardataurl=node.getAttribute('cubicweb:dataurl');varmethod=postJSON;if(options.method=='get'){method=function(url,data,callback){// We can't rely on jQuery.getJSON because the server// might set the Content-Type's response header to 'text/plain'jQuery.get(url,data,function(response){callback(cw.evalJSON(response));});};}varself=this;// closuremethod(dataurl,null,function(data){// in case we received a list of couple, we assume that the first// element is the real value to be sent, and the second one is the// value to be displayedif(data.length&&data[0].length==2){options.formatItem=function(row){returnrow[1];};self.hideRealValue(node);self.setCurrentValue(node,data);}jQuery(node).autocomplete(data,options);});},hideRealValue:function(node){varhidden=INPUT({'type':"hidden",'name':node.name,'value':node.value});node.parentNode.appendChild(hidden);// remove 'name' attribute from visible input so that it is not submitted// and set correct value in the corresponding hidden fieldjQuery(node).removeAttr('name').bind('result',function(_,row,_){hidden.value=row[0];});},setCurrentValue:function(node,data){// called when the data is loaded to reset the correct displayed// value in the visible input field (typically replacing an eid// by a displayable value)varcurvalue=node.value;if(!node.value){return;}for(vari=0,length=data.length;i<length;i++){varrow=data[i];if(row[0]==curvalue){node.value=row[1];return;}}}});Widgets.StaticFileSuggestField=defclass('StaticSuggestField',[Widgets.SuggestField],{__init__:function(node){Widgets.SuggestField.__init__(this,node,{method:'get'});}});Widgets.RestrictedSuggestField=defclass('RestrictedSuggestField',[Widgets.SuggestField],{__init__:function(node){Widgets.SuggestField.__init__(this,node,{mustMatch:true});}});//remote version of RestrictedSuggestFieldWidgets.LazySuggestField=defclass('LazySuggestField',[Widgets.SuggestField],{__init__:function(node,options){varself=this;varmulti="no";options=options||{};options.max=50;options.delay=50;options.cacheLength=0;options.mustMatch=true;// multiple selection not supported yet (still need to formalize correctly// initial values / display values)varinitialvalue=cw.evalJSON(node.getAttribute('cubicweb:initialvalue')||'null');if(!initialvalue){initialvalue=node.value;}options=jQuery.extend({dataType:'json',multiple:(multi=="yes")?true:false,parse:this.parseResult},options);vardataurl=node.getAttribute('cubicweb:dataurl');// remove 'name' from original input and add the hidden one that will// store the actual valuevarhidden=INPUT({'type':"hidden",'name':node.name,'value':initialvalue});node.parentNode.appendChild(hidden);jQuery(node).bind('result',{hinput:hidden,input:node},self.hideRealValue).removeAttr('name').autocomplete(dataurl,options);},hideRealValue:function(evt,data,value){if(!value){value="";}evt.data.hinput.value=value;},/* * @param data: a list of couple (value, label) to fill the suggestion list, * (returned by CW through AJAX) */parseResult:function(data){varparsed=[];for(vari=0;i<data.length;i++){varvalue=''+data[i][0];// a string is required later by jquery.autocomplete.jsvarlabel=data[i][1];parsed[parsed.length]={data:[label],value:value,result:label};};returnparsed;}});/** * .. class:: Widgets.SuggestForm * * suggestform displays a suggest field and associated validate / cancel buttons * constructor's argumemts are the same that BaseSuggestField widget */Widgets.SuggestForm=defclass("SuggestForm",null,{__init__:function(inputid,initfunc,varargs,validatefunc,options){this.validatefunc=validatefunc||noop;this.sgfield=newWidgets.BaseSuggestField(inputid,initfunc,varargs,options);this.oklabel=options.oklabel||'ok';this.cancellabel=options.cancellabel||'cancel';bindMethods(this);connect(this.sgfield,'validate',this,this.entryValidated);},show:function(parentnode){varsgnode=this.sgfield.builddom();varbuttons=DIV({'class':"sgformbuttons"},[A({'href':"javascript: noop();",'onclick':this.onValidateClicked},this.oklabel),' / ',A({'href':"javascript: noop();",'onclick':this.destroy},escapeHTML(this.cancellabel))]);varformnode=DIV({'class':"sgform"},[sgnode,buttons]);appendChildNodes(parentnode,formnode);this.sgfield.textinput.focus();this.formnode=formnode;returnformnode;},destroy:function(){signal(this,'destroy');this.sgfield.destroy();removeElement(this.formnode);},onValidateClicked:function(){this.validatefunc(this,this.sgfield.taglist());},/* just an indirection to pass the form instead of the sgfield as first parameter */entryValidated:function(sgfield,taglist){this.validatefunc(this,taglist);}});/** * .. function:: toggleTree(event) * * called when the use clicks on a tree node * - if the node has a `cubicweb:loadurl` attribute, replace the content of the node * by the url's content. * - else, there's nothing to do, let the jquery plugin handle it. */functiontoggleTree(event){varlinode=jQuery(this);varurl=linode.attr('cubicweb:loadurl');if(url){linode.find('ul.placeholder').remove();linode.loadxhtml(url,{callback:function(domnode){linode.removeAttr('cubicweb:loadurl');jQuery(domnode).treeview({toggle:toggleTree,prerendered:true});returnnull;}},'post','append');}}/** * .. class:: Widgets.TimelineWidget * * widget based on SIMILE's timeline widget * http://code.google.com/p/simile-widgets/ * * Beware not to mess with SIMILE's Timeline JS namepsace ! */Widgets.TimelineWidget=defclass("TimelineWidget",null,{__init__:function(wdgnode){vartldiv=DIV({id:"tl",style:'height: 200px; border: 1px solid #ccc;'});wdgnode.appendChild(tldiv);vartlunit=wdgnode.getAttribute('cubicweb:tlunit')||'YEAR';vareventSource=newTimeline.DefaultEventSource();varbandData={eventPainter:Timeline.CubicWebEventPainter,eventSource:eventSource,width:"100%",intervalUnit:Timeline.DateTime[tlunit.toUpperCase()],intervalPixels:100};varbandInfos=[Timeline.createBandInfo(bandData)];this.tl=Timeline.create(tldiv,bandInfos);varloadurl=wdgnode.getAttribute('cubicweb:loadurl');Timeline.loadJSON(loadurl,function(json,url){eventSource.loadJSON(json,url);});}});Widgets.TemplateTextField=defclass("TemplateTextField",null,{__init__:function(wdgnode){this.variables=jQuery(wdgnode).attr('cubicweb:variables').split(',');this.options={name:wdgnode.getAttribute('cubicweb:inputid'),rows:wdgnode.getAttribute('cubicweb:rows')||40,cols:wdgnode.getAttribute('cubicweb:cols')||80};// this.variableRegexp = /%\((\w+)\)s/;this.errorField=DIV({'class':"errorMessage"});this.textField=TEXTAREA(this.options);jQuery(this.textField).bind('keyup',{'self':this},this.highlightInvalidVariables);jQuery('#substitutions').prepend(this.errorField);jQuery('#substitutions .errorMessage').hide();wdgnode.appendChild(this.textField);},/* signal callbacks */highlightInvalidVariables:function(event){varself=event.data.self;vartext=self.textField.value;varunknownVariables=[];varit=0;vargroup=null;varvariableRegexp=/%\((\w+)\)s/g;// emulates rgx.findAll()while(group=variableRegexp.exec(text)){if(!$.inArray(group[1],self.variables)){unknownVariables.push(group[1]);}it++;if(it>5){break;}}varerrText='';if(unknownVariables.length){errText="Detected invalid variables : "+unknownVariables.join(', ');jQuery('#substitutions .errorMessage').show();}else{jQuery('#substitutions .errorMessage').hide();}self.errorField.innerHTML=errText;}});cw.widgets={/** * .. function:: insertText(text, areaId) * * inspects textarea with id `areaId` and replaces the current selected text * with `text`. Cursor is then set at the end of the inserted text. */insertText:function(text,areaId){vartextarea=jQuery('#'+areaId);if(document.selection){// IEvarselLength;textarea.focus();varsel=document.selection.createRange();selLength=sel.text.length;sel.text=text;sel.moveStart('character',selLength-text.length);sel.select();}elseif(textarea.selectionStart||textarea.selectionStart=='0'){// mozillavarstartPos=textarea.selectionStart;varendPos=textarea.selectionEnd;// insert text so that it replaces the [startPos, endPos] parttextarea.value=textarea.value.substring(0,startPos)+text+textarea.value.substring(endPos,textarea.value.length);// set cursor pos at the end of the inserted texttextarea.selectionStart=textarea.selectionEnd=startPos+text.length;textarea.focus();}else{// safety belt for other browserstextarea.value+=text;}}};