Wiki source code of Live Table component

Version 130.1 by Oana Florea on 2010/04/14

Hide last authors
Vincent Massol 125.2 1 {{box cssClass="floatinginfobox" title="**Contents**"}}{{toc start="2" depth="6"/}}{{/box}}
2
ElenaOanaTabaranu 58.1 3 = Live Table component =
fkorteby 27.1 4
Vincent Massol 125.2 5 {{warning}}This document is a draft.{{/warning}}
fkorteby 27.1 6
ElenaOanaTabaranu 58.1 7 == Summary ==
fkorteby 27.1 8
Thibaut Camberlin 127.1 9 The **"Live Table"** component is a dynamic table loading data lazily using ajax requests as the user browse the table, in order to scale easily the display of very large amounts of data. Users can browse the table thanks to a pagination system. Filters on columns are available to search for specific entries. Columns can be made sortable in both direction. For a demonstration of the main capabilities, check out the [[XWiki live table component video>>http://vimeo.com/4474332]].
ElenaOanaTabaranu 65.1 10
ElenaOanaTabaranu 58.1 11 == How to use ==
fkorteby 27.1 12
Vincent Massol 125.2 13 The Live Table component is made available in several ways to [[Applications>>code:Applications.WebHome]] developers and developers of the [[XWiki platform and products>>dev:Main.WebHome]], depending on their specific needs. The simplest, and most of the times preferred, way to embed a Live Table component is to use the ##livetable## velocity macro, available in all pages of your wiki. With only one call to this macro, you will be able to display a live table of the data of your choice! For more specific needs, in the cases you need more control on the table behavior, or to build another component on top of the live table, it is possible to instantiate the live table from JavaScript. In this case, you, as opposed to the velocity macro, you will need to construct the HTML elements the JavaScript component expects yourself, either from JavaScript, either writing the [[HTML directly in your wiki page>>platform:Main.XWikiSyntax]]
fkorteby 27.1 14
ElenaOanaTabaranu 72.1 15 === Using the Velocity livetable macro ===
fkorteby 27.1 16
Thibaut Camberlin 128.1 17 Please refer to [[XWiki Live Table Macro>>code:Macros.LiveTableMacro]] reference.
fkorteby 27.1 18
ElenaOanaTabaranu 58.1 19 === HTML + JavaScript ===
fkorteby 27.1 20
ElenaOanaTabaranu 115.1 21 This is a dynamic component which must me used carefully because of performance issues. The HTML content of the Live Table is automatically updated using Javascript and [[JSON>>http://www.json.org]] code generated with the [[Live Table Macro>>code:Macros.LiveTableMacro]].
Oana Florea 130.1 22 The Live Table component is build using the Prototype library as a class representing an AJAX-populated live table:
23 * The default variable name generated in XWiki Enterprise is **ta** and you can use it, for example to manipulate the display of the component's rows by calling ##ta.showRows()##.
24 * Take a look at the [[code snippet>>code:Snippets.RefreshLiveTablewithchangedURLSnippet]] for refreshing the live table.
25 * For more details, check the Javascript file **livetable.js** on your wiki at **{{{http://localhost:8080/xwiki/bin/skin/resources/js/xwiki/table/livetable.js}}}**.
fkorteby 27.1 26
ElenaOanaTabaranu 58.1 27 === Example ===
Oana Florea 123.1 28
Vincent Massol 125.2 29 {{warning}}You need the Ratings plugin in your wiki to make the most of the example below.{{/warning}}
ElenaOanaTabaranu 103.1 30
ElenaOanaTabaranu 30.1 31 The Live Table component could be extended with a special column containing the ratings of some documents from the wiki.
ElenaOanaTabaranu 58.1 32 The first step would be to generate the JSON code of the column we want to add. In order to achieve this goal we need to modify the result page of the velocity live table macro with our custom page **TutorialsCode.LiveTableRatings**:
ElenaOanaTabaranu 32.1 33
Vincent Massol 125.2 34 {{code language="none"}}
ElenaOanaTabaranu 55.1 35 #set($collist = ["doc.name", "_ratings", "doc.date", "doc.author", "_actions"])
ElenaOanaTabaranu 57.1 36 #set($colprops = {
ElenaOanaTabaranu 32.1 37 "doc.name" : { "type" : "text" , "size" : 30, "sortable":true, "filterable":true},
38 "_ratings" : { "sortable":false},
39 "doc.date" : { "type" : "date" },
40 "doc.author" : { "type" : "text", "link" : "author"},
ElenaOanaTabaranu 51.1 41 "_actions" : { "actions": ["copy","delete","rename","rights"]}
ElenaOanaTabaranu 57.1 42 })
43 #set($options = { "resultPage":"TutorialsCode.LiveTableRatings",
44 "tagCloud" : true,
45 "translationPrefix" : "xe.index.",
46 "rowCount": 10 })
47 #set($ok = $xwiki.ssx.use("TutorialsCode.LiveTableRatings"))
48 #set($ok = $xwiki.jsx.use("TutorialsCode.LiveTableRatings"))
Oana Florea 122.1 49 #livetable("alldocs" $collist $colprops $options)
ElenaOanaTabaranu 58.1 50 {{/code}}
ElenaOanaTabaranu 32.1 51
Vincent Massol 125.2 52 If we want to display the rated documents from a certain space, e.g. **Tutorials**, the last parameter of the ###gridresultwithfilter## must contain the query we need.
ElenaOanaTabaranu 34.1 53
Vincent Massol 125.2 54 {{code language="none"}}
ElenaOanaTabaranu 34.1 55 #includeMacros("TutorialsCode.LiveTableResultsMacros")
ElenaOanaTabaranu 59.1 56
Raluca Stavro 126.1 57 #gridresultwithfilter("" $request.collist.split(",") "" "and doc.space='Tutorials' and doc.name<>'WebHome' and doc.name<>'WebPreferences'")
ElenaOanaTabaranu 60.1 58 {{/code}}
ElenaOanaTabaranu 73.1 59
ElenaOanaTabaranu 106.1 60 The page at **TutorialsCode.LiveTableResultsMacros** should contain a copy of the default code (the page is located at **XWiki.LiveTableResultsMacro**) for the live table result page. The next step implies extracting the rating information from the wiki pages located the **Tutorials** space. We add a new velocity macro which will test the presence of the rating object with class **XWiki.AverageRatingsClass** and extract the rating value.
ElenaOanaTabaranu 90.1 61
Vincent Massol 125.2 62 {{code language="none"}}
ElenaOanaTabaranu 81.1 63 ##
64 ## list ratings
65 ##
ElenaOanaTabaranu 73.1 66 #macro(grid_ratings $udoc)
67 #set($ratings = "")
68 #set($ratingObj = "")
69 #set($ratingObj = $udoc.getObject("XWiki.AverageRatingsClass"))
70 #if("$!ratingObj" != "")
71 #set($ratings = $ratingObj.get("averagevote"))
72 #end
73 #end
ElenaOanaTabaranu 74.1 74 {{/code}}
ElenaOanaTabaranu 90.1 75
Vincent Massol 125.2 76 The continue our extension of this component, we need to add a test for the presence of our custom column to call the ##grid_ratings## macro when generating the JSON:
ElenaOanaTabaranu 93.1 77
Vincent Massol 125.2 78 {{code language="none"}}
ElenaOanaTabaranu 75.1 79 ...
ElenaOanaTabaranu 79.1 80 #elseif($colname=="_images") ,
81 #grid_photolist($udoc)
82 "${colname}" : "${photolist}"
ElenaOanaTabaranu 75.1 83 #elseif($colname=="_ratings"),
84 #grid_ratings($udoc)
85 "${colname}" : "${ratings}"
ElenaOanaTabaranu 79.1 86 #else ,
ElenaOanaTabaranu 75.1 87 ...
88 {{/code}}
ElenaOanaTabaranu 87.1 89
Vincent Massol 125.2 90 {{info}}Firebug can be a useful tool when testing the JSON code generated.{{/info}}
ElenaOanaTabaranu 87.1 91
ElenaOanaTabaranu 107.1 92 Transforming the rating information to a nice star display requires using Javascript to replace the content of that column with the proper HTML. Attach the image below to your **TutorialsCode.LiveTableRatings** page to change the default white background when hovering on the stars.
ElenaOanaTabaranu 75.1 93
ElenaOanaTabaranu 109.1 94 [[image:star-table.gif||style="margin-right: 1em;"]]
ElenaOanaTabaranu 108.1 95
ElenaOanaTabaranu 107.1 96 Add an on-demand Javascript Extension with a content which can be parsed and use the code below to enrich your component:
97
Vincent Massol 125.2 98 {{code language="javascript"}}
ElenaOanaTabaranu 98.1 99 /** Display ratings in the live table on row event */
100 document.observe("xwiki:livetable:newrow", function(ev) {
101 $$('._ratings').each(function (el) {
102 // update content
103 var avgvote = 0.0;
104 if (el.innerHTML !== "") {
105 avgvote = parseFloat(el.innerHTML);
106 }
107 var wstyle = 0;
108 if (avgvote > 0) {
109 wstyle = avgvote * 20;
110 el.innerHTML="<div class='avg-rating'><div class='rating-stars'><ul class='small-star-rating'><li style='width: " + wstyle +"%;' class='current-rating'/><li><a class='one-star' href='#' onclick='return false;'>1</a></li><li><a class='two-stars' href='#' onclick='return false;'>2</a></li><li><a class='three-stars' href='#' onclick='return false;'>3</a></li><li><a class='four-stars' href='#' onclick='return false;'>4</a></li><li><a class='five-stars' href='#' onclick='return false;'>5</a></li></ul></div></div>";
111 } else if(el.innerHTML === ""){
112 el.innerHTML="<div class='avg-rating'><div class='rating-stars'><ul class='small-star-rating'><li style='width: 0%;' class='current-rating'/><li><a class='one-star' href='#' onclick='return false;'>1</a></li> <li><a class='two-stars' href='#' onclick='return false;'>2</a></li><li><a class='three-stars' href='#' onclick='return false;'>3</a></li><li><a class='four-stars' href='#' onclick='return false;'>4</a></li><li><a class='five-stars' href='#' onclick='return false;'>5</a></li></ul></div></div>";
113 }
114 });
115 document.fire("xwiki:livetable:ready", {
116 });
117 });
118 /** Update ratings images on hover event aka mouse over and mouse out.*/
119 document.observe("xwiki:livetable:ready", function(levent) {
120 $$('.xwiki-livetable-display-body tr').each(function (elem) {
121 Event.observe(elem, 'mouseover', function (ev) {
122 var ratings = elem.down('._ratings');
123 if(ratings) {
124 // update default white stars background
125 var elt = ratings.down(2);
126 elt.setStyle({
127 backgroundImage: 'url($xwiki.getDocument("TutorialsCode.LiveTableRatings").getAttachmentURL("star-table.gif"))',
128 backgroundPosition: '0% 0%'
129 });
130 // update votes background
131 elt = ratings.down(3);
132 elt.setStyle({
133 backgroundImage: 'url($xwiki.getDocument("TutorialsCode.LiveTableRatings").getAttachmentURL("star-table.gif"))',
134 backgroundPosition: '0% 50%'
135 });
136 }
137 });
138 Event.observe(elem, 'mouseout', function (ev) {
139 var ratings = elem.down('._ratings');
140 if(ratings) {
141 // restore default white stars background
142 var elt = ratings.down(2);
143 elt.setStyle({
144 backgroundImage: 'url($xwiki.getDocument("XWiki.Ratings").getAttachmentURL("star.gif"))',
145 backgroundPosition: '0% 0%'
146 });
147 // restore votes background
148 elt = elem.down('._ratings').down(3);
149 elt.setStyle({
150 backgroundImage: 'url($xwiki.getDocument("XWiki.Ratings").getAttachmentURL("star.gif"))',
151 backgroundPosition: '0% 50%'
152 });
153 }
154 });
ElenaOanaTabaranu 75.1 155
ElenaOanaTabaranu 98.1 156 });
157 });
158 {{/code}}
159
ElenaOanaTabaranu 110.1 160 Also, add some CSS in an on-demand Stylesheet extension with a content which can be parsed to polish the custom column display:
Vincent Massol 121.2 161
Vincent Massol 125.2 162 {{code language="none"}}
ElenaOanaTabaranu 98.1 163 /* ratings for live table */
164 .small-star-rating,
165 .small-star-rating .current-rating {
166 background: url($xwiki.getDocument("XWiki.Ratings").getAttachmentURL("star.gif")) left -1000px repeat-x;
167 }
168 .small-star-rating{
169 position:relative;
170 width:125px;
171 height:25px;
172 overflow:hidden;
173 list-style:none;
174 margin:0px !important;
175 padding:0px !important;
176 background-position: left top;
177 }
178 .small-star-rating li{
179 display: inline;
180 }
181 .small-star-rating a,
182 .small-star-rating .current-rating{
183 position:absolute;
184 top:0;
185 left:0;
186 text-indent:-1000em;
187 height:25px;
188 line-height:25px;
189 outline:none;
190 overflow:hidden;
191 border: none;
192 }
193 .small-star-rating .current-rating{
194 z-index:1;
195 background-position: left center;
196 }
197 .small-star-rating:hover,
198 .small-star-rating:active,
199 .small-star-rating:focus {
200 background-image: url($xwiki.getDocument("TutorialsCode.LiveTableRatings").getAttachmentURL("star-table.gif"));
201 background-repeat: repeat-x;
202 background-position: top left;
203 }
204 .small-star-rating a.one-star{
205 width:20%;
206 z-index:6;
207 }
208 .small-star-rating a.two-stars{
209 width:40%;
210 z-index:5;
211 }
212 .small-star-rating a.three-stars{
213 width:60%;
214 z-index:4;
215 }
216 .small-star-rating a.four-stars{
217 width:80%;
218 z-index:3;
219 }
220 .small-star-rating a.five-stars{
221 width:100%;
222 z-index:2;
223 }
224 .rating-stars {
225 display: inline;
226 float: left;
227 clear: none;
228 }
229 {{/code}}
ElenaOanaTabaranu 111.1 230
231 The final result should look like this:
ElenaOanaTabaranu 114.1 232
ElenaOanaTabaranu 113.1 233 [[image:livetabel.png||style="margin-right: 1em;"]]

Get Connected