]> git.wh0rd.org - tt-rss.git/blob - js/FeedTree.js
f3ac8ba19587362c93e6b5c596f18daaaafba6db
[tt-rss.git] / js / FeedTree.js
1 dojo.provide("fox.FeedTree");
2 dojo.provide("fox.FeedStoreModel");
3
4 dojo.require("dijit.Tree");
5 dojo.require("dijit.Menu");
6
7 dojo.declare("fox.FeedStoreModel", dijit.tree.ForestStoreModel, {
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 },
19 getItemById: function(id) {
20 return this.store._itemsByIdentity[id];
21 },
22 getFeedValue: function(feed, is_cat, key) {
23 if (!this.store._itemsByIdentity) return undefined;
24
25 if (is_cat)
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 },
33 getFeedName: function(feed, is_cat) {
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 = '';
45 if (!this.store._itemsByIdentity) return undefined;
46
47 if (is_cat)
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 },
55 getNextUnreadFeed: function (feed, is_cat) {
56 if (!this.store._itemsByIdentity)
57 return null;
58
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
70 for (var j = i+1; j < items.length; j++) {
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
77 for (var j = 0; j < i; j++) {
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 }
85
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 },
94 });
95
96 dojo.declare("fox.FeedTree", dijit.Tree, {
97 _onKeyPress: function(/* Event */ e) {
98 return; // Stop dijit.Tree from interpreting keystrokes
99 },
100 _createTreeNode: function(args) {
101 var tnode = new dijit._TreeNode(args);
102
103 if (args.item.icon && args.item.icon[0])
104 tnode.iconNode.src = args.item.icon[0];
105
106 var id = args.item.id[0];
107 var bare_id = parseInt(id.substr(id.indexOf(':')+1));
108
109 if (bare_id < _label_base_index) {
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
123 if (id.match("FEED:")) {
124 var menu = new dijit.Menu();
125 menu.row_id = bare_id;
126
127 menu.addChild(new dijit.MenuItem({
128 label: __("Mark as read"),
129 onClick: function() {
130 catchupFeed(this.getParent().row_id);
131 }}));
132
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
140 /* menu.addChild(new dijit.MenuItem({
141 label: __("Update feed"),
142 onClick: function() {
143 heduleFeedUpdate(this.getParent().row_id, false);
144 }})); */
145 }
146
147 menu.bindDomNode(tnode.domNode);
148 tnode._menu = menu;
149 }
150
151 if (id.match("CAT:") && bare_id >= 0) {
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
165 if (id.match("CAT:")) {
166 loading = dojo.doc.createElement('img');
167 loading.className = 'loadingNode';
168 loading.src = 'images/blank_icon.gif';
169 dojo.place(loading, tnode.labelNode, 'after');
170 tnode.loadingNode = loading;
171 }
172
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
187 ctr = dojo.doc.createElement('span');
188 ctr.className = 'counterNode';
189 ctr.innerHTML = args.item.unread;
190
191 //args.item.unread > 0 ? ctr.addClassName("unread") : ctr.removeClassName("unread");
192
193 args.item.unread > 0 ? Element.show(ctr) : Element.hide(ctr);
194
195 dojo.place(ctr, tnode.rowNode, 'first');
196 tnode.counterNode = ctr;
197
198 //tnode.labelNode.innerHTML = args.label;
199 return tnode;
200 },
201 postCreate: function() {
202 this.connect(this.model, "onChange", "updateCounter");
203 this.connect(this, "_expandNode", function() {
204 this.hideRead(getInitParam("hide_read_feeds"), getInitParam("hide_read_shows_special"));
205 });
206
207 this.inherited(arguments);
208 },
209 updateCounter: function (item) {
210 var tree = this;
211
212 //console.log("updateCounter: " + item.id[0] + " " + item.unread + " " + tree);
213
214 var node = tree._itemNodesMap[item.id];
215
216 if (node) {
217 node = node[0];
218
219 if (node.counterNode) {
220 ctr = node.counterNode;
221 ctr.innerHTML = item.unread;
222 item.unread > 0 ? Effect.Appear(ctr, {duration : 0.3,
223 queue: { position: 'end', scope: 'CAPPEAR-' + item.id, limit: 1 }}) :
224 Element.hide(ctr);
225 }
226 }
227
228 },
229 getTooltip: function (item) {
230 if (item.updated)
231 return item.updated;
232 else
233 return "";
234 },
235 getIconClass: function (item, opened) {
236 return (!item || this.model.mayHaveChildren(item)) ? (opened ? "dijitFolderOpened" : "dijitFolderClosed") : "feedIcon";
237 },
238 getLabelClass: function (item, opened) {
239 return (item.unread == 0) ? "dijitTreeLabel" : "dijitTreeLabel Unread";
240 },
241 getRowClass: function (item, opened) {
242 var rc = (!item.error || item.error == '') ? "dijitTreeRow" :
243 "dijitTreeRow Error";
244
245 if (item.unread > 0) rc += " Unread";
246
247 return rc;
248 },
249 getLabel: function(item) {
250 var name = String(item.name);
251
252 /* Horrible */
253 name = name.replace(/&quot;/g, "\"");
254 name = name.replace(/&amp;/g, "&");
255 name = name.replace(/&mdash;/g, "-");
256 name = name.replace(/&lt;/g, "<");
257 name = name.replace(/&gt;/g, ">");
258
259 /* var label;
260
261 if (item.unread > 0) {
262 label = name + " (" + item.unread + ")";
263 } else {
264 label = name;
265 } */
266
267 return name;
268 },
269 expandParentNodes: function(feed, is_cat, list) {
270 try {
271 for (var i = 0; i < list.length; i++) {
272 var id = String(list[i].id);
273 var item = this._itemNodesMap[id];
274
275 if (item) {
276 item = item[0];
277 this._expandNode(item);
278 }
279 }
280 } catch (e) {
281 exception_error("expandParentNodes", e);
282 }
283 },
284 findNodeParentsAndExpandThem: function(feed, is_cat, root, parents) {
285 // expands all parents of specified feed to properly mark it as active
286 // my fav thing about frameworks is doing everything myself
287 try {
288 var test_id = is_cat ? 'CAT:' + feed : 'FEED:' + feed;
289
290 if (!root) {
291 if (!this.model || !this.model.store) return false;
292
293 var items = this.model.store._arrayOfTopLevelItems;
294
295 for (var i = 0; i < items.length; i++) {
296 if (String(items[i].id) == test_id) {
297 this.expandParentNodes(feed, is_cat, parents);
298 } else {
299 this.findNodeParentsAndExpandThem(feed, is_cat, items[i], []);
300 }
301 }
302 } else {
303 if (root.items) {
304 parents.push(root);
305
306 for (var i = 0; i < root.items.length; i++) {
307 if (String(root.items[i].id) == test_id) {
308 this.expandParentNodes(feed, is_cat, parents);
309 } else {
310 this.findNodeParentsAndExpandThem(feed, is_cat, root.items[i], parents.slice(0));
311 }
312 }
313 } else {
314 if (String(root.id) == test_id) {
315 this.expandParentNodes(feed, is_cat, parents.slice(0));
316 }
317 }
318 }
319 } catch (e) {
320 exception_error("findNodeParentsAndExpandThem", e);
321 }
322 },
323 selectFeed: function(feed, is_cat) {
324 this.findNodeParentsAndExpandThem(feed, is_cat, false, false);
325
326 if (is_cat)
327 treeNode = this._itemNodesMap['CAT:' + feed];
328 else
329 treeNode = this._itemNodesMap['FEED:' + feed];
330
331 if (treeNode) {
332 treeNode = treeNode[0];
333 if (!is_cat) this._expandNode(treeNode);
334 this.set("selectedNodes", [treeNode]);
335 }
336 },
337 setFeedIcon: function(feed, is_cat, src) {
338 if (is_cat)
339 treeNode = this._itemNodesMap['CAT:' + feed];
340 else
341 treeNode = this._itemNodesMap['FEED:' + feed];
342
343 if (treeNode) {
344 treeNode = treeNode[0];
345 treeNode.iconNode.src = src;
346 return true;
347 }
348 return false;
349 },
350 setFeedExpandoIcon: function(feed, is_cat, src) {
351 if (is_cat)
352 treeNode = this._itemNodesMap['CAT:' + feed];
353 else
354 treeNode = this._itemNodesMap['FEED:' + feed];
355
356 if (treeNode) {
357 treeNode = treeNode[0];
358 if (treeNode.loadingNode) {
359 treeNode.loadingNode.src = src;
360 return true;
361 } else {
362 treeNode.expandoNode.src = src;
363 return true;
364 }
365 }
366
367 return false;
368 },
369 hasCats: function() {
370 return this.model.hasCats();
371 },
372 hideReadCat: function (cat, hide, show_special) {
373 if (this.hasCats()) {
374 var tree = this;
375
376 if (cat && cat.items) {
377 var cat_unread = tree.hideReadFeeds(cat.items, hide, show_special);
378
379 var id = String(cat.id);
380 var node = tree._itemNodesMap[id];
381 var bare_id = parseInt(id.substr(id.indexOf(":")+1));
382
383 if (node) {
384 var check_unread = tree.model.getFeedUnread(bare_id, true);
385
386 if (hide && cat_unread == 0 && check_unread == 0) {
387 Effect.Fade(node[0].rowNode, {duration : 0.3,
388 queue: { position: 'end', scope: 'FFADE-' + id, limit: 1 }});
389 } else {
390 Element.show(node[0].rowNode);
391 ++cat_unread;
392 }
393 }
394 }
395 }
396 },
397 hideRead: function (hide, show_special) {
398 if (this.hasCats()) {
399
400 var tree = this;
401 var cats = this.model.store._arrayOfTopLevelItems;
402
403 cats.each(function(cat) {
404 tree.hideReadCat(cat, hide, show_special);
405 });
406
407 } else {
408 this.hideReadFeeds(this.model.store._arrayOfTopLevelItems, hide,
409 show_special);
410 }
411 },
412 hideReadFeeds: function (items, hide, show_special) {
413 var tree = this;
414 var cat_unread = 0;
415
416 items.each(function(feed) {
417 var id = String(feed.id);
418
419 // it's a subcategory
420 if (feed.items) {
421 tree.hideReadCat(feed, hide, show_special);
422 } else { // it's a feed
423 var bare_id = parseInt(feed.bare_id);;
424
425 var unread = feed.unread[0];
426 var node = tree._itemNodesMap[id];
427
428 if (node) {
429 if (hide && unread == 0 && (bare_id > 0 || bare_id < _label_base_index || !show_special)) {
430 Effect.Fade(node[0].rowNode, {duration : 0.3,
431 queue: { position: 'end', scope: 'FFADE-' + id, limit: 1 }});
432 } else {
433 Element.show(node[0].rowNode);
434 ++cat_unread;
435 }
436 }
437 }
438 });
439
440 return cat_unread;
441 },
442 collapseCat: function(id) {
443 if (!this.model.hasCats()) return;
444
445 var tree = this;
446
447 var node = tree._itemNodesMap['CAT:' + id][0];
448 var item = tree.model.store._itemsByIdentity['CAT:' + id];
449
450 if (node && item) {
451 if (!node.isExpanded)
452 tree._expandNode(node);
453 else
454 tree._collapseNode(node);
455
456 }
457 },
458 getVisibleUnreadFeeds: function() {
459 var items = this.model.store._arrayOfAllItems;
460 var rv = [];
461
462 for (var i = 0; i < items.length; i++) {
463 var id = String(items[i].id);
464 var box = this._itemNodesMap[id];
465
466 if (box) {
467 var row = box[0].rowNode;
468 var cat = false;
469
470 try {
471 cat = box[0].rowNode.parentNode.parentNode;
472 } catch (e) { }
473
474 if (row) {
475 if (Element.visible(row) && (!cat || Element.visible(cat))) {
476 var feed_id = String(items[i].bare_id);
477 var is_cat = !id.match('FEED:');
478 var unread = this.model.getFeedUnread(feed_id, is_cat);
479
480 if (unread > 0)
481 rv.push([feed_id, is_cat]);
482
483 }
484 }
485 }
486 }
487
488 return rv;
489 },
490 getNextFeed: function (feed, is_cat) {
491 if (is_cat) {
492 treeItem = this.model.store._itemsByIdentity['CAT:' + feed];
493 } else {
494 treeItem = this.model.store._itemsByIdentity['FEED:' + feed];
495 }
496
497 items = this.model.store._arrayOfAllItems;
498 var item = items[0];
499
500 for (var i = 0; i < items.length; i++) {
501 if (items[i] == treeItem) {
502
503 for (var j = i+1; j < items.length; j++) {
504 var id = String(items[j].id);
505 var box = this._itemNodesMap[id];
506
507 if (box) {
508 var row = box[0].rowNode;
509 var cat = box[0].rowNode.parentNode.parentNode;
510
511 if (Element.visible(cat) && Element.visible(row)) {
512 item = items[j];
513 break;
514 }
515 }
516 }
517 break;
518 }
519 }
520
521 if (item) {
522 return [this.model.store.getValue(item, 'bare_id'),
523 !this.model.store.getValue(item, 'id').match('FEED:')];
524 } else {
525 return false;
526 }
527 },
528 getPreviousFeed: function (feed, is_cat) {
529 if (is_cat) {
530 treeItem = this.model.store._itemsByIdentity['CAT:' + feed];
531 } else {
532 treeItem = this.model.store._itemsByIdentity['FEED:' + feed];
533 }
534
535 items = this.model.store._arrayOfAllItems;
536 var item = items[0];
537
538 for (var i = 0; i < items.length; i++) {
539 if (items[i] == treeItem) {
540
541 for (var j = i-1; j > 0; j--) {
542 var id = String(items[j].id);
543 var box = this._itemNodesMap[id];
544
545 if (box) {
546 var row = box[0].rowNode;
547 var cat = box[0].rowNode.parentNode.parentNode;
548
549 if (Element.visible(cat) && Element.visible(row)) {
550 item = items[j];
551 break;
552 }
553 }
554
555 }
556 break;
557 }
558 }
559
560 if (item) {
561 return [this.model.store.getValue(item, 'bare_id'),
562 !this.model.store.getValue(item, 'id').match('FEED:')];
563 } else {
564 return false;
565 }
566
567 },
568 getFeedCategory: function(feed) {
569 try {
570 return this.getNodesByItem(this.model.store.
571 _itemsByIdentity["FEED:" + feed])[0].
572 getParent().item.bare_id[0];
573
574 } catch (e) {
575 return false;
576 }
577 },
578 });