Wiki source code of Live Table component

Version 129.1 by JeromeVelociter on 2010/01/19

Show last authors
1 {{box cssClass="floatinginfobox" title="**Contents**"}}{{toc start="2" depth="6"/}}{{/box}}
2
3 = Live Table component =
4
5 {{warning}}This document is a draft.{{/warning}}
6
7 == Summary ==
8
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]].
10
11 == How to use ==
12
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]]
14
15 === Using the Velocity livetable macro ===
16
17 Please refer to [[XWiki Live Table Macro>>code:Macros.LiveTableMacro]] reference.
18
19 === HTML + JavaScript ===
20
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]].
22 The Live Table component is build using the Prototype library as a class representing an AJAX-populated live table. 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()##.
23
24 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}}}**.
25
26 === Example ===
27
28 {{warning}}You need the Ratings plugin in your wiki to make the most of the example below.{{/warning}}
29
30 The Live Table component could be extended with a special column containing the ratings of some documents from the wiki.
31 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**:
32
33 {{code language="none"}}
34 #set($collist = ["doc.name", "_ratings", "doc.date", "doc.author", "_actions"])
35 #set($colprops = {
36 "doc.name" : { "type" : "text" , "size" : 30, "sortable":true, "filterable":true},
37 "_ratings" : { "sortable":false},
38 "doc.date" : { "type" : "date" },
39 "doc.author" : { "type" : "text", "link" : "author"},
40 "_actions" : { "actions": ["copy","delete","rename","rights"]}
41 })
42 #set($options = { "resultPage":"TutorialsCode.LiveTableRatings",
43 "tagCloud" : true,
44 "translationPrefix" : "xe.index.",
45 "rowCount": 10 })
46 #set($ok = $xwiki.ssx.use("TutorialsCode.LiveTableRatings"))
47 #set($ok = $xwiki.jsx.use("TutorialsCode.LiveTableRatings"))
48 #livetable("alldocs" $collist $colprops $options)
49 {{/code}}
50
51 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.
52
53 {{code language="none"}}
54 #includeMacros("TutorialsCode.LiveTableResultsMacros")
55
56 #gridresultwithfilter("" $request.collist.split(",") "" "and doc.space='Tutorials' and doc.name<>'WebHome' and doc.name<>'WebPreferences'")
57 {{/code}}
58
59 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.
60
61 {{code language="none"}}
62 ##
63 ## list ratings
64 ##
65 #macro(grid_ratings $udoc)
66 #set($ratings = "")
67 #set($ratingObj = "")
68 #set($ratingObj = $udoc.getObject("XWiki.AverageRatingsClass"))
69 #if("$!ratingObj" != "")
70 #set($ratings = $ratingObj.get("averagevote"))
71 #end
72 #end
73 {{/code}}
74
75 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:
76
77 {{code language="none"}}
78 ...
79 #elseif($colname=="_images") ,
80 #grid_photolist($udoc)
81 "${colname}" : "${photolist}"
82 #elseif($colname=="_ratings"),
83 #grid_ratings($udoc)
84 "${colname}" : "${ratings}"
85 #else ,
86 ...
87 {{/code}}
88
89 {{info}}Firebug can be a useful tool when testing the JSON code generated.{{/info}}
90
91 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.
92
93 [[image:star-table.gif||style="margin-right: 1em;"]]
94
95 Add an on-demand Javascript Extension with a content which can be parsed and use the code below to enrich your component:
96
97 {{code language="javascript"}}
98 /** Display ratings in the live table on row event */
99 document.observe("xwiki:livetable:newrow", function(ev) {
100 $$('._ratings').each(function (el) {
101 // update content
102 var avgvote = 0.0;
103 if (el.innerHTML !== "") {
104 avgvote = parseFloat(el.innerHTML);
105 }
106 var wstyle = 0;
107 if (avgvote > 0) {
108 wstyle = avgvote * 20;
109 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>";
110 } else if(el.innerHTML === ""){
111 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>";
112 }
113 });
114 document.fire("xwiki:livetable:ready", {
115 });
116 });
117 /** Update ratings images on hover event aka mouse over and mouse out.*/
118 document.observe("xwiki:livetable:ready", function(levent) {
119 $$('.xwiki-livetable-display-body tr').each(function (elem) {
120 Event.observe(elem, 'mouseover', function (ev) {
121 var ratings = elem.down('._ratings');
122 if(ratings) {
123 // update default white stars background
124 var elt = ratings.down(2);
125 elt.setStyle({
126 backgroundImage: 'url($xwiki.getDocument("TutorialsCode.LiveTableRatings").getAttachmentURL("star-table.gif"))',
127 backgroundPosition: '0% 0%'
128 });
129 // update votes background
130 elt = ratings.down(3);
131 elt.setStyle({
132 backgroundImage: 'url($xwiki.getDocument("TutorialsCode.LiveTableRatings").getAttachmentURL("star-table.gif"))',
133 backgroundPosition: '0% 50%'
134 });
135 }
136 });
137 Event.observe(elem, 'mouseout', function (ev) {
138 var ratings = elem.down('._ratings');
139 if(ratings) {
140 // restore default white stars background
141 var elt = ratings.down(2);
142 elt.setStyle({
143 backgroundImage: 'url($xwiki.getDocument("XWiki.Ratings").getAttachmentURL("star.gif"))',
144 backgroundPosition: '0% 0%'
145 });
146 // restore votes background
147 elt = elem.down('._ratings').down(3);
148 elt.setStyle({
149 backgroundImage: 'url($xwiki.getDocument("XWiki.Ratings").getAttachmentURL("star.gif"))',
150 backgroundPosition: '0% 50%'
151 });
152 }
153 });
154
155 });
156 });
157 {{/code}}
158
159 Also, add some CSS in an on-demand Stylesheet extension with a content which can be parsed to polish the custom column display:
160
161 {{code language="none"}}
162 /* ratings for live table */
163 .small-star-rating,
164 .small-star-rating .current-rating {
165 background: url($xwiki.getDocument("XWiki.Ratings").getAttachmentURL("star.gif")) left -1000px repeat-x;
166 }
167 .small-star-rating{
168 position:relative;
169 width:125px;
170 height:25px;
171 overflow:hidden;
172 list-style:none;
173 margin:0px !important;
174 padding:0px !important;
175 background-position: left top;
176 }
177 .small-star-rating li{
178 display: inline;
179 }
180 .small-star-rating a,
181 .small-star-rating .current-rating{
182 position:absolute;
183 top:0;
184 left:0;
185 text-indent:-1000em;
186 height:25px;
187 line-height:25px;
188 outline:none;
189 overflow:hidden;
190 border: none;
191 }
192 .small-star-rating .current-rating{
193 z-index:1;
194 background-position: left center;
195 }
196 .small-star-rating:hover,
197 .small-star-rating:active,
198 .small-star-rating:focus {
199 background-image: url($xwiki.getDocument("TutorialsCode.LiveTableRatings").getAttachmentURL("star-table.gif"));
200 background-repeat: repeat-x;
201 background-position: top left;
202 }
203 .small-star-rating a.one-star{
204 width:20%;
205 z-index:6;
206 }
207 .small-star-rating a.two-stars{
208 width:40%;
209 z-index:5;
210 }
211 .small-star-rating a.three-stars{
212 width:60%;
213 z-index:4;
214 }
215 .small-star-rating a.four-stars{
216 width:80%;
217 z-index:3;
218 }
219 .small-star-rating a.five-stars{
220 width:100%;
221 z-index:2;
222 }
223 .rating-stars {
224 display: inline;
225 float: left;
226 clear: none;
227 }
228 {{/code}}
229
230 The final result should look like this:
231
232 [[image:livetabel.png||style="margin-right: 1em;"]]

Get Connected