]> git.wh0rd.org - tt-rss.git/blame - js/FeedTree.js
add tooltip
[tt-rss.git] / js / FeedTree.js
CommitLineData
05f224a3
AD
1dojo.provide("fox.FeedTree");
2dojo.provide("fox.FeedStoreModel");
3
4dojo.require("dijit.Tree");
997429c2 5dojo.require("dijit.Menu");
05f224a3
AD
6
7dojo.declare("fox.FeedStoreModel", dijit.tree.ForestStoreModel, {
1beea800
AD
8 getItemsInCategory: function (id) {
9 if (!this.store._itemsByIdentity) return undefined;
10
11 cat = this.store._itemsByIdentity['CAT:' + id];
12
13 if (cat && cat.items)
14 return cat.items;
15 else
16 return undefined;
17
18 },
05f224a3
AD
19 getItemById: function(id) {
20 return this.store._itemsByIdentity[id];
21 },
452e75cc 22 getFeedValue: function(feed, is_cat, key) {
b8aa9ca7
AD
23 if (!this.store._itemsByIdentity) return undefined;
24
452e75cc 25 if (is_cat)
05f224a3
AD
26 treeItem = this.store._itemsByIdentity['CAT:' + feed];
27 else
28 treeItem = this.store._itemsByIdentity['FEED:' + feed];
29
30 if (treeItem)
31 return this.store.getValue(treeItem, key);
32 },
452e75cc 33 getFeedName: function(feed, is_cat) {
05f224a3
AD
34 return this.getFeedValue(feed, is_cat, 'name');
35 },
36 getFeedUnread: function(feed, is_cat) {
37 var unread = parseInt(this.getFeedValue(feed, is_cat, 'unread'));
38 return (isNaN(unread)) ? 0 : unread;
39 },
40 setFeedUnread: function(feed, is_cat, unread) {
41 return this.setFeedValue(feed, is_cat, 'unread', parseInt(unread));
42 },
43 setFeedValue: function(feed, is_cat, key, value) {
44 if (!value) value = '';
b8aa9ca7 45 if (!this.store._itemsByIdentity) return undefined;
05f224a3 46
452e75cc 47 if (is_cat)
05f224a3
AD
48 treeItem = this.store._itemsByIdentity['CAT:' + feed];
49 else
50 treeItem = this.store._itemsByIdentity['FEED:' + feed];
51
52 if (treeItem)
53 return this.store.setValue(treeItem, key, value);
54 },
692de159 55 getNextUnreadFeed: function (feed, is_cat) {
6a0906d1
AD
56 if (!this.store._itemsByIdentity)
57 return null;
58
692de159
AD
59 if (is_cat) {
60 treeItem = this.store._itemsByIdentity['CAT:' + feed];
61 items = this.store._arrayOfTopLevelItems;
62 } else {
63 treeItem = this.store._itemsByIdentity['FEED:' + feed];
64 items = this.store._arrayOfAllItems;
65 }
66
67 for (var i = 0; i < items.length; i++) {
68 if (items[i] == treeItem) {
69
e331188f 70 for (var j = i+1; j < items.length; j++) {
692de159
AD
71 var unread = this.store.getValue(items[j], 'unread');
72 var id = this.store.getValue(items[j], 'id');
73
74 if (unread > 0 && (is_cat || id.match("FEED:"))) return items[j];
75 }
76
e331188f 77 for (var j = 0; j < i; j++) {
692de159
AD
78 var unread = this.store.getValue(items[j], 'unread');
79 var id = this.store.getValue(items[j], 'id');
80
81 if (unread > 0 && (is_cat || id.match("FEED:"))) return items[j];
82 }
83 }
84 }
452e75cc 85
692de159
AD
86 return null;
87 },
88 hasCats: function() {
89 if (this.store && this.store._itemsByIdentity)
90 return this.store._itemsByIdentity['CAT:-1'] != undefined;
91 else
92 return false;
93 },
05f224a3
AD
94});
95
96dojo.declare("fox.FeedTree", dijit.Tree, {
71bb56d9
AD
97 _onKeyPress: function(/* Event */ e) {
98 return; // Stop dijit.Tree from interpreting keystrokes
99 },
05f224a3
AD
100 _createTreeNode: function(args) {
101 var tnode = new dijit._TreeNode(args);
452e75cc 102
89473cb5 103 if (args.item.icon && args.item.icon[0])
05f224a3
AD
104 tnode.iconNode.src = args.item.icon[0];
105
997429c2
AD
106 var id = args.item.id[0];
107 var bare_id = parseInt(id.substr(id.indexOf(':')+1));
108
f822a8e5 109 if (bare_id < _label_base_index) {
9fe80bcd
AD
110 var span = dojo.doc.createElement('span');
111 var fg_color = args.item.fg_color[0];
112 var bg_color = args.item.bg_color[0];
113
114 span.innerHTML = "&alpha;";
115 span.className = 'labelColorIndicator';
116 span.setStyle({
117 color: fg_color,
118 backgroundColor: bg_color});
119
120 dojo.place(span, tnode.iconNode, 'replace');
121 }
122
10afc7c6 123 if (id.match("FEED:")) {
997429c2
AD
124 var menu = new dijit.Menu();
125 menu.row_id = bare_id;
126
e3d2c029
AD
127 menu.addChild(new dijit.MenuItem({
128 label: __("Mark as read"),
129 onClick: function() {
130 catchupFeed(this.getParent().row_id);
131 }}));
132
10afc7c6
AD
133 if (bare_id > 0) {
134 menu.addChild(new dijit.MenuItem({
135 label: __("Edit feed"),
136 onClick: function() {
137 editFeed(this.getParent().row_id, false);
138 }}));
139
c0c2abba 140 /* menu.addChild(new dijit.MenuItem({
10afc7c6
AD
141 label: __("Update feed"),
142 onClick: function() {
c0c2abba
AD
143 heduleFeedUpdate(this.getParent().row_id, false);
144 }})); */
10afc7c6 145 }
997429c2
AD
146
147 menu.bindDomNode(tnode.domNode);
148 tnode._menu = menu;
149 }
150
057cc813 151 if (id.match("CAT:") && bare_id >= 0) {
e3d2c029
AD
152 var menu = new dijit.Menu();
153 menu.row_id = bare_id;
154
155 menu.addChild(new dijit.MenuItem({
156 label: __("Mark as read"),
157 onClick: function() {
158 catchupFeed(this.getParent().row_id, true);
159 }}));
160
161 menu.bindDomNode(tnode.domNode);
162 tnode._menu = menu;
163 }
164
6cf60c88
AD
165 if (id.match("CAT:")) {
166 loading = dojo.doc.createElement('img');
167 loading.className = 'loadingNode';
84b48db4 168 loading.src = 'images/blank_icon.gif';
6cf60c88
AD
169 dojo.place(loading, tnode.labelNode, 'after');
170 tnode.loadingNode = loading;
171 }
172
ec6e5155
AD
173 if (id.match("CAT:") && bare_id == -1) {
174 var menu = new dijit.Menu();
175 menu.row_id = bare_id;
176
177 menu.addChild(new dijit.MenuItem({
178 label: __("Mark all feeds as read"),
179 onClick: function() {
180 catchupAllFeeds();
181 }}));
182
183 menu.bindDomNode(tnode.domNode);
184 tnode._menu = menu;
185 }
186
66af65f1
AD
187 ctr = dojo.doc.createElement('span');
188 ctr.className = 'counterNode';
189 ctr.innerHTML = '0';
190 dojo.place(ctr, tnode.labelNode, 'after');
191 tnode.counterNode = ctr;
ec6e5155 192
947daa10 193 //tnode.labelNode.innerHTML = args.label;
05f224a3
AD
194 return tnode;
195 },
69970d5b
AD
196 getTooltip: function (item) {
197 if (item.updated)
198 return item.updated;
199 else
200 return "";
201 },
05f224a3
AD
202 getIconClass: function (item, opened) {
203 return (!item || this.model.mayHaveChildren(item)) ? (opened ? "dijitFolderOpened" : "dijitFolderClosed") : "feedIcon";
204 },
205 getLabelClass: function (item, opened) {
206 return (item.unread == 0) ? "dijitTreeLabel" : "dijitTreeLabel Unread";
207 },
208 getRowClass: function (item, opened) {
66af65f1 209 var rc = (!item.error || item.error == '') ? "dijitTreeRow" :
05f224a3 210 "dijitTreeRow Error";
66af65f1
AD
211
212 if (item.unread > 0) rc += " Unread";
213
214 return rc;
05f224a3
AD
215 },
216 getLabel: function(item) {
cb2c7faa
AD
217 var name = String(item.name);
218
219 /* Horrible */
906fc5b7
AD
220 name = name.replace(/&quot;/g, "\"");
221 name = name.replace(/&amp;/g, "&");
222 name = name.replace(/&mdash;/g, "-");
223 name = name.replace(/&lt;/g, "<");
224 name = name.replace(/&gt;/g, ">");
cb2c7faa 225
66af65f1 226 /* var label;
2c5f231e 227
05f224a3 228 if (item.unread > 0) {
2c5f231e 229 label = name + " (" + item.unread + ")";
05f224a3 230 } else {
2c5f231e 231 label = name;
66af65f1 232 } */
2c5f231e 233
66af65f1 234 return name;
05f224a3 235 },
2196b517
AD
236 expandParentNodes: function(feed, is_cat, list) {
237 try {
238 for (var i = 0; i < list.length; i++) {
239 var id = String(list[i].id);
240 var item = this._itemNodesMap[id];
241
242 if (item) {
243 item = item[0];
244 this._expandNode(item);
245 }
246 }
247 } catch (e) {
248 exception_error("expandParentNodes", e);
249 }
250 },
251 findNodeParentsAndExpandThem: function(feed, is_cat, root, parents) {
252 // expands all parents of specified feed to properly mark it as active
253 // my fav thing about frameworks is doing everything myself
254 try {
255 var test_id = is_cat ? 'CAT:' + feed : 'FEED:' + feed;
256
257 if (!root) {
258 if (!this.model || !this.model.store) return false;
259
260 var items = this.model.store._arrayOfTopLevelItems;
261
262 for (var i = 0; i < items.length; i++) {
263 if (String(items[i].id) == test_id) {
264 this.expandParentNodes(feed, is_cat, parents);
265 } else {
266 this.findNodeParentsAndExpandThem(feed, is_cat, items[i], []);
267 }
268 }
269 } else {
270 if (root.items) {
271 parents.push(root);
272
273 for (var i = 0; i < root.items.length; i++) {
274 if (String(root.items[i].id) == test_id) {
275 this.expandParentNodes(feed, is_cat, parents);
276 } else {
4429463d 277 this.findNodeParentsAndExpandThem(feed, is_cat, root.items[i], parents.slice(0));
2196b517
AD
278 }
279 }
280 } else {
281 if (String(root.id) == test_id) {
4429463d 282 this.expandParentNodes(feed, is_cat, parents.slice(0));
2196b517
AD
283 }
284 }
285 }
286 } catch (e) {
287 exception_error("findNodeParentsAndExpandThem", e);
288 }
289 },
05f224a3 290 selectFeed: function(feed, is_cat) {
2196b517
AD
291 this.findNodeParentsAndExpandThem(feed, is_cat, false, false);
292
452e75cc 293 if (is_cat)
05f224a3
AD
294 treeNode = this._itemNodesMap['CAT:' + feed];
295 else
296 treeNode = this._itemNodesMap['FEED:' + feed];
297
298 if (treeNode) {
299 treeNode = treeNode[0];
300 if (!is_cat) this._expandNode(treeNode);
81bea17a 301 this.set("selectedNodes", [treeNode]);
05f224a3
AD
302 }
303 },
304 setFeedIcon: function(feed, is_cat, src) {
452e75cc 305 if (is_cat)
05f224a3
AD
306 treeNode = this._itemNodesMap['CAT:' + feed];
307 else
308 treeNode = this._itemNodesMap['FEED:' + feed];
309
310 if (treeNode) {
311 treeNode = treeNode[0];
312 treeNode.iconNode.src = src;
313 return true;
314 }
315 return false;
316 },
317 setFeedExpandoIcon: function(feed, is_cat, src) {
452e75cc 318 if (is_cat)
05f224a3
AD
319 treeNode = this._itemNodesMap['CAT:' + feed];
320 else
321 treeNode = this._itemNodesMap['FEED:' + feed];
322
323 if (treeNode) {
324 treeNode = treeNode[0];
6cf60c88
AD
325 if (is_cat) {
326 if (treeNode.loadingNode) {
327 treeNode.loadingNode.src = src;
328 return true;
329 }
330 } else {
331 treeNode.expandoNode.src = src;
332 return true;
333 }
05f224a3
AD
334 }
335
336 return false;
337 },
338 hasCats: function() {
692de159 339 return this.model.hasCats();
05f224a3 340 },
5712a2fe 341 hideReadCat: function (cat, hide, show_special) {
05f224a3 342 if (this.hasCats()) {
05f224a3 343 var tree = this;
452e75cc 344
5712a2fe 345 if (cat && cat.items) {
05f224a3 346 var cat_unread = tree.hideReadFeeds(cat.items, hide, show_special);
452e75cc 347
05f224a3
AD
348 var id = String(cat.id);
349 var node = tree._itemNodesMap[id];
350 var bare_id = parseInt(id.substr(id.indexOf(":")+1));
452e75cc 351
05f224a3
AD
352 if (node) {
353 var check_unread = tree.model.getFeedUnread(bare_id, true);
452e75cc 354
0ef32f48 355 if (hide && cat_unread == 0 && check_unread == 0) {
452e75cc 356 Effect.Fade(node[0].rowNode, {duration : 0.3,
05f224a3
AD
357 queue: { position: 'end', scope: 'FFADE-' + id, limit: 1 }});
358 } else {
359 Element.show(node[0].rowNode);
360 ++cat_unread;
361 }
452e75cc 362 }
5712a2fe
AD
363 }
364 }
365 },
366 hideRead: function (hide, show_special) {
367 if (this.hasCats()) {
368
369 var tree = this;
370 var cats = this.model.store._arrayOfTopLevelItems;
371
372 cats.each(function(cat) {
373 tree.hideReadCat(cat, hide, show_special);
05f224a3 374 });
05f224a3
AD
375
376 } else {
452e75cc 377 this.hideReadFeeds(this.model.store._arrayOfTopLevelItems, hide,
05f224a3
AD
378 show_special);
379 }
380 },
381 hideReadFeeds: function (items, hide, show_special) {
382 var tree = this;
383 var cat_unread = 0;
384
385 items.each(function(feed) {
386 var id = String(feed.id);
6cdc4576 387
1a8a6239
AD
388 // it's a subcategory
389 if (feed.items) {
6e67a7c4 390 tree.hideReadCat(feed, hide, show_special);
1a8a6239 391 } else { // it's a feed
6cdc4576
AD
392 var bare_id = parseInt(feed.bare_id);;
393
394 var unread = feed.unread[0];
395 var node = tree._itemNodesMap[id];
396
397 if (node) {
f822a8e5 398 if (hide && unread == 0 && (bare_id > 0 || bare_id < _label_base_index || !show_special)) {
6cdc4576
AD
399 Effect.Fade(node[0].rowNode, {duration : 0.3,
400 queue: { position: 'end', scope: 'FFADE-' + id, limit: 1 }});
401 } else {
402 Element.show(node[0].rowNode);
403 ++cat_unread;
404 }
05f224a3
AD
405 }
406 }
407 });
452e75cc 408
05f224a3
AD
409 return cat_unread;
410 },
7613280a
AD
411 collapseCat: function(id) {
412 if (!this.model.hasCats()) return;
413
414 var tree = this;
415
416 var node = tree._itemNodesMap['CAT:' + id][0];
417 var item = tree.model.store._itemsByIdentity['CAT:' + id];
418
419 if (node && item) {
88918ca6 420 if (!node.isExpanded)
7613280a
AD
421 tree._expandNode(node);
422 else
423 tree._collapseNode(node);
424
7613280a
AD
425 }
426 },
452e75cc
AD
427 getVisibleUnreadFeeds: function() {
428 var items = this.model.store._arrayOfAllItems;
429 var rv = [];
430
431 for (var i = 0; i < items.length; i++) {
432 var id = String(items[i].id);
433 var box = this._itemNodesMap[id];
434
435 if (box) {
436 var row = box[0].rowNode;
437 var cat = false;
438
439 try {
440 cat = box[0].rowNode.parentNode.parentNode;
441 } catch (e) { }
442
443 if (row) {
444 if (Element.visible(row) && (!cat || Element.visible(cat))) {
445 var feed_id = String(items[i].bare_id);
446 var is_cat = !id.match('FEED:');
447 var unread = this.model.getFeedUnread(feed_id, is_cat);
448
449 if (unread > 0)
450 rv.push([feed_id, is_cat]);
451
452 }
453 }
454 }
455 }
456
457 return rv;
458 },
d026d372
AD
459 getNextFeed: function (feed, is_cat) {
460 if (is_cat) {
461 treeItem = this.model.store._itemsByIdentity['CAT:' + feed];
462 } else {
463 treeItem = this.model.store._itemsByIdentity['FEED:' + feed];
464 }
465
466 items = this.model.store._arrayOfAllItems;
467 var item = items[0];
468
469 for (var i = 0; i < items.length; i++) {
470 if (items[i] == treeItem) {
471
e331188f 472 for (var j = i+1; j < items.length; j++) {
452e75cc 473 var id = String(items[j].id);
d026d372
AD
474 var box = this._itemNodesMap[id];
475
476 if (box) {
c2b2869c
AD
477 var row = box[0].rowNode;
478 var cat = box[0].rowNode.parentNode.parentNode;
d026d372 479
c2b2869c 480 if (Element.visible(cat) && Element.visible(row)) {
d026d372
AD
481 item = items[j];
482 break;
483 }
484 }
485 }
486 break;
487 }
488 }
489
490 if (item) {
452e75cc 491 return [this.model.store.getValue(item, 'bare_id'),
d026d372
AD
492 !this.model.store.getValue(item, 'id').match('FEED:')];
493 } else {
494 return false;
495 }
496 },
497 getPreviousFeed: function (feed, is_cat) {
498 if (is_cat) {
499 treeItem = this.model.store._itemsByIdentity['CAT:' + feed];
500 } else {
501 treeItem = this.model.store._itemsByIdentity['FEED:' + feed];
502 }
503
504 items = this.model.store._arrayOfAllItems;
505 var item = items[0];
506
507 for (var i = 0; i < items.length; i++) {
508 if (items[i] == treeItem) {
509
e331188f 510 for (var j = i-1; j > 0; j--) {
452e75cc 511 var id = String(items[j].id);
d026d372
AD
512 var box = this._itemNodesMap[id];
513
514 if (box) {
c2b2869c
AD
515 var row = box[0].rowNode;
516 var cat = box[0].rowNode.parentNode.parentNode;
d026d372 517
c2b2869c 518 if (Element.visible(cat) && Element.visible(row)) {
d026d372
AD
519 item = items[j];
520 break;
521 }
522 }
c2b2869c 523
d026d372
AD
524 }
525 break;
526 }
527 }
528
529 if (item) {
452e75cc 530 return [this.model.store.getValue(item, 'bare_id'),
d026d372
AD
531 !this.model.store.getValue(item, 'id').match('FEED:')];
532 } else {
533 return false;
534 }
535
536 },
6c8e8fbd
AD
537 getFeedCategory: function(feed) {
538 try {
539 return this.getNodesByItem(this.model.store.
540 _itemsByIdentity["FEED:" + feed])[0].
541 getParent().item.bare_id[0];
d026d372 542
6c8e8fbd
AD
543 } catch (e) {
544 return false;
545 }
546 },
05f224a3 547});