1 var global_unread = -1;
2 var hotkey_prefix = false;
3 var hotkey_prefix_pressed = false;
4 var hotkey_actions = {};
5 var _widescreen_mode = false;
7 var _active_feed_id = 0;
8 var _active_feed_is_cat = false;
19 function activeFeedIsCat() {
20 return _active_feed_is_cat;
23 function getActiveFeedId() {
24 return _active_feed_id;
27 function setActiveFeedId(id, is_cat) {
29 hash_set('c', is_cat ? 1 : 0);
32 _active_feed_is_cat = is_cat;
34 $("headlines-frame").setAttribute("feed-id", id);
35 $("headlines-frame").setAttribute("is-cat", is_cat ? 1 : 0);
37 selectFeed(id, is_cat);
39 PluginHost.run(PluginHost.HOOK_FEED_SET_ACTIVE, _active_article_id);
43 function updateFeedList() {
45 Element.show("feedlistLoading");
49 if (dijit.byId("feedTree")) {
50 dijit.byId("feedTree").destroyRecursive();
53 var store = new dojo.data.ItemFileWriteStore({
54 url: "backend.php?op=pref_feeds&method=getfeedtree&mode=2"
57 var treeModel = new fox.FeedStoreModel({
60 "type": getInitParam('enable_feed_cats') == 1 ? "category" : "feed"
64 childrenAttrs: ["items"]
67 var tree = new fox.FeedTree({
69 onClick: function (item, node) {
70 var id = String(item.id);
71 var is_cat = id.match("^CAT:");
72 var feed = id.substr(id.indexOf(":") + 1);
73 viewfeed({feed: feed, is_cat: is_cat});
82 /* var menu = new dijit.Menu({id: 'feedMenu'});
84 menu.addChild(new dijit.MenuItem({
85 label: "Simple menu item"
88 // menu.bindDomNode(tree.domNode); */
90 var tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
91 console.log(dijit.getEnclosingWidget(event.target));
92 dojo.disconnect(tmph);
95 $("feeds-holder").appendChild(tree.domNode);
97 var tmph = dojo.connect(tree, 'onLoad', function () {
98 dojo.disconnect(tmph);
99 Element.hide("feedlistLoading");
104 loading_set_progress(25);
116 function catchupAllFeeds() {
118 var str = __("Mark all articles as read?");
120 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
122 var query_str = "backend.php?op=feeds&method=catchupAll";
124 notify_progress("Marking all feeds as read...");
126 //console.log("catchupAllFeeds Q=" + query_str);
128 new Ajax.Request("backend.php", {
129 parameters: query_str,
130 onComplete: function(transport) {
131 request_counters(true);
140 function viewCurrentFeed(method) {
141 console.log("viewCurrentFeed: " + method);
143 if (getActiveFeedId() != undefined) {
144 viewfeed({feed: getActiveFeedId(), is_cat: activeFeedIsCat(), method: method});
146 return false; // block unneeded form submits
150 if (getInitParam("bw_limit") != "1") {
152 setTimeout(timeout, 60*1000);
157 var query = "backend.php?op=feeds&method=search¶m=" +
158 param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
160 if (dijit.byId("searchDlg"))
161 dijit.byId("searchDlg").destroyRecursive();
163 dialog = new dijit.Dialog({
166 style: "width: 600px",
167 execute: function() {
168 if (this.validate()) {
169 _search_query = dojo.objectToQuery(this.attr('value'));
179 function updateTitle() {
180 var tmp = "Tiny Tiny RSS";
182 if (global_unread > 0) {
183 tmp = "(" + global_unread + ") " + tmp;
187 if (global_unread > 0) {
188 window.fluid.dockBadge = global_unread;
190 window.fluid.dockBadge = "";
194 document.title = tmp;
197 function genericSanityCheck() {
198 setCookie("ttrss_test", "TEST");
200 if (getCookie("ttrss_test") != "TEST") {
201 return fatalError(2);
210 window.onerror = function(message, filename, lineno, colno, error) {
211 report_error(message, filename, lineno, colno, error);
214 require(["dojo/_base/kernel",
221 "dijit/ColorPalette",
224 "dijit/form/ComboButton",
225 "dijit/form/CheckBox",
226 "dijit/form/DropDownButton",
227 "dijit/form/FilteringSelect",
229 "dijit/form/RadioButton",
231 "dijit/form/SimpleTextarea",
232 "dijit/form/TextBox",
233 "dijit/form/ComboBox",
234 "dijit/form/ValidationTextBox",
235 "dijit/InlineEditBox",
236 "dijit/layout/AccordionContainer",
237 "dijit/layout/BorderContainer",
238 "dijit/layout/ContentPane",
239 "dijit/layout/TabContainer",
240 "dijit/PopupMenuItem",
244 "dijit/tree/dndSource",
245 "dijit/tree/ForestStoreModel",
246 "dojo/data/ItemFileWriteStore",
247 "fox/FeedTree" ], function (dojo, ready, parser) {
254 if (!genericSanityCheck())
257 loading_set_progress(30);
259 var a = document.createElement('audio');
261 var hasAudio = !!a.canPlayType;
262 var hasSandbox = "sandbox" in document.createElement("iframe");
263 var hasMp3 = !!(a.canPlayType && a.canPlayType('audio/mpeg;').replace(/no/, ''));
264 var clientTzOffset = new Date().getTimezoneOffset() * 60;
266 init_hotkey_actions();
268 new Ajax.Request("backend.php", {
270 op: "rpc", method: "sanityCheck", hasAudio: hasAudio,
272 clientTzOffset: clientTzOffset,
273 hasSandbox: hasSandbox
275 onComplete: function (transport) {
276 backend_sanity_check_callback(transport);
289 function init_hotkey_actions() {
290 hotkey_actions["next_feed"] = function() {
291 var rv = dijit.byId("feedTree").getNextFeed(
292 getActiveFeedId(), activeFeedIsCat());
294 if (rv) viewfeed({feed: rv[0], is_cat: rv[1], can_wait: true})
296 hotkey_actions["prev_feed"] = function() {
297 var rv = dijit.byId("feedTree").getPreviousFeed(
298 getActiveFeedId(), activeFeedIsCat());
300 if (rv) viewfeed({feed: rv[0], is_cat: rv[1], can_wait: true})
302 hotkey_actions["next_article"] = function() {
305 hotkey_actions["prev_article"] = function() {
308 hotkey_actions["next_article_noscroll"] = function() {
309 moveToPost('next', true);
311 hotkey_actions["prev_article_noscroll"] = function() {
312 moveToPost('prev', true);
314 hotkey_actions["next_article_noexpand"] = function() {
315 moveToPost('next', true, true);
317 hotkey_actions["prev_article_noexpand"] = function() {
318 moveToPost('prev', true, true);
320 hotkey_actions["collapse_article"] = function() {
321 var id = getActiveArticleId();
322 var elem = $("CICD-"+id);
325 if (elem.visible()) {
326 cdmCollapseArticle(null, id);
329 cdmExpandArticle(id);
333 hotkey_actions["toggle_expand"] = function() {
334 var id = getActiveArticleId();
335 var elem = $("CICD-"+id);
338 if (elem.visible()) {
339 cdmCollapseArticle(null, id, false);
342 cdmExpandArticle(id);
346 hotkey_actions["search_dialog"] = function() {
349 hotkey_actions["toggle_mark"] = function() {
350 selectionToggleMarked(undefined, false, true);
352 hotkey_actions["toggle_publ"] = function() {
353 selectionTogglePublished(undefined, false, true);
355 hotkey_actions["toggle_unread"] = function() {
356 selectionToggleUnread(undefined, false, true);
358 hotkey_actions["edit_tags"] = function() {
359 var id = getActiveArticleId();
364 hotkey_actions["open_in_new_window"] = function() {
365 if (getActiveArticleId()) {
366 openArticleInNewWindow(getActiveArticleId());
370 hotkey_actions["catchup_below"] = function() {
371 catchupRelativeToArticle(1);
373 hotkey_actions["catchup_above"] = function() {
374 catchupRelativeToArticle(0);
376 hotkey_actions["article_scroll_down"] = function() {
377 var ctr = $("content_insert") ? $("content_insert") : $("headlines-frame");
381 hotkey_actions["article_scroll_up"] = function() {
382 var ctr = $("content_insert") ? $("content_insert") : $("headlines-frame");
386 hotkey_actions["close_article"] = function() {
388 if (!getInitParam("cdm_expanded")) {
389 cdmCollapseArticle(false, getActiveArticleId());
395 hotkey_actions["email_article"] = function() {
396 if (typeof emailArticle != "undefined") {
398 } else if (typeof mailtoArticle != "undefined") {
401 alert(__("Please enable mail plugin first."));
404 hotkey_actions["select_all"] = function() {
405 selectArticles('all');
407 hotkey_actions["select_unread"] = function() {
408 selectArticles('unread');
410 hotkey_actions["select_marked"] = function() {
411 selectArticles('marked');
413 hotkey_actions["select_published"] = function() {
414 selectArticles('published');
416 hotkey_actions["select_invert"] = function() {
417 selectArticles('invert');
419 hotkey_actions["select_none"] = function() {
420 selectArticles('none');
422 hotkey_actions["feed_refresh"] = function() {
423 if (getActiveFeedId() != undefined) {
424 viewfeed({feed: getActiveFeedId(), is_cat: activeFeedIsCat()});
428 hotkey_actions["feed_unhide_read"] = function() {
431 hotkey_actions["feed_subscribe"] = function() {
434 hotkey_actions["feed_debug_update"] = function() {
435 if (!activeFeedIsCat() && parseInt(getActiveFeedId()) > 0) {
436 window.open("backend.php?op=feeds&method=update_debugger&feed_id=" + getActiveFeedId() +
437 "&csrf_token=" + getInitParam("csrf_token"));
439 alert("You can't debug this kind of feed.");
443 hotkey_actions["feed_debug_viewfeed"] = function() {
444 viewfeed({feed: getActiveFeedId(), is_cat: activeFeedIsCat(), viewfeed_debug: true});
447 hotkey_actions["feed_edit"] = function() {
448 if (activeFeedIsCat())
449 alert(__("You can't edit this kind of feed."));
451 editFeed(getActiveFeedId());
453 hotkey_actions["feed_catchup"] = function() {
454 if (getActiveFeedId() != undefined) {
455 catchupCurrentFeed();
459 hotkey_actions["feed_reverse"] = function() {
460 reverseHeadlineOrder();
462 hotkey_actions["feed_toggle_vgroup"] = function() {
463 var query_str = "?op=rpc&method=togglepref&key=VFEED_GROUP_BY_FEED";
465 new Ajax.Request("backend.php", {
466 parameters: query_str,
467 onComplete: function(transport) {
472 hotkey_actions["catchup_all"] = function() {
475 hotkey_actions["cat_toggle_collapse"] = function() {
476 if (activeFeedIsCat()) {
477 dijit.byId("feedTree").collapseCat(getActiveFeedId());
481 hotkey_actions["goto_all"] = function() {
482 viewfeed({feed: -4});
484 hotkey_actions["goto_fresh"] = function() {
485 viewfeed({feed: -3});
487 hotkey_actions["goto_marked"] = function() {
488 viewfeed({feed: -1});
490 hotkey_actions["goto_published"] = function() {
491 viewfeed({feed: -2});
493 hotkey_actions["goto_tagcloud"] = function() {
494 displayDlg(__("Tag cloud"), "printTagCloud");
496 hotkey_actions["goto_prefs"] = function() {
499 hotkey_actions["select_article_cursor"] = function() {
500 var id = getArticleUnderPointer();
502 var row = $("RROW-" + id);
505 var cb = dijit.getEnclosingWidget(
506 row.getElementsByClassName("rchk")[0]);
509 cb.attr("checked", !cb.attr("checked"));
510 toggleSelectRowById(cb, "RROW-" + id);
516 hotkey_actions["create_label"] = function() {
519 hotkey_actions["create_filter"] = function() {
522 hotkey_actions["collapse_sidebar"] = function() {
525 hotkey_actions["toggle_embed_original"] = function() {
526 if (typeof embedOriginalArticle != "undefined") {
527 if (getActiveArticleId())
528 embedOriginalArticle(getActiveArticleId());
530 alert(__("Please enable embed_original plugin first."));
533 hotkey_actions["toggle_widescreen"] = function() {
535 _widescreen_mode = !_widescreen_mode;
537 // reset stored sizes because geometry changed
538 setCookie("ttrss_ci_width", 0);
539 setCookie("ttrss_ci_height", 0);
541 switchPanelMode(_widescreen_mode);
543 alert(__("Widescreen is not available in combined mode."));
546 hotkey_actions["help_dialog"] = function() {
549 hotkey_actions["toggle_combined_mode"] = function() {
550 notify_progress("Loading, please wait...");
552 var value = isCdmMode() ? "false" : "true";
553 var query = "?op=rpc&method=setpref&key=COMBINED_DISPLAY_MODE&value=" + value;
555 new Ajax.Request("backend.php", {
557 onComplete: function(transport) {
558 setInitParam("combined_display_mode",
559 !getInitParam("combined_display_mode"));
566 hotkey_actions["toggle_cdm_expanded"] = function() {
567 notify_progress("Loading, please wait...");
569 var value = getInitParam("cdm_expanded") ? "false" : "true";
570 var query = "?op=rpc&method=setpref&key=CDM_EXPANDED&value=" + value;
572 new Ajax.Request("backend.php", {
574 onComplete: function(transport) {
575 setInitParam("cdm_expanded", !getInitParam("cdm_expanded"));
581 function init_second_stage() {
585 if (parseInt(getCookie("ttrss_fh_width")) > 0) {
586 dijit.byId("feeds-holder").domNode.setStyle(
587 {width: getCookie("ttrss_fh_width") + "px" });
590 dijit.byId("main").resize();
592 var tmph = dojo.connect(dijit.byId('feeds-holder'), 'resize',
594 if (args && args.w >= 0) {
595 setCookie("ttrss_fh_width", args.w, getInitParam("cookie_lifetime"));
599 var tmph = dojo.connect(dijit.byId('content-insert'), 'resize',
601 if (args && args.w >= 0 && args.h >= 0) {
602 setCookie("ttrss_ci_width", args.w, getInitParam("cookie_lifetime"));
603 setCookie("ttrss_ci_height", args.h, getInitParam("cookie_lifetime"));
607 delCookie("ttrss_test");
609 var toolbar = document.forms["main_toolbar_form"];
611 dijit.getEnclosingWidget(toolbar.view_mode).attr('value',
612 getInitParam("default_view_mode"));
614 dijit.getEnclosingWidget(toolbar.order_by).attr('value',
615 getInitParam("default_view_order_by"));
617 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
619 var hash_feed_id = hash_get('f');
620 var hash_feed_is_cat = hash_get('c') == "1";
622 if (hash_feed_id != undefined) {
623 setActiveFeedId(hash_feed_id, hash_feed_is_cat);
626 loading_set_progress(50);
628 // can't use cache_clear() here because viewfeed might not have initialized yet
629 if ('sessionStorage' in window && window['sessionStorage'] !== null)
630 sessionStorage.clear();
632 var hotkeys = getInitParam("hotkeys");
635 for (sequence in hotkeys[1]) {
636 filtered = sequence.replace(/\|.*$/, "");
637 tmp[filtered] = hotkeys[1][sequence];
641 setInitParam("hotkeys", hotkeys);
643 _widescreen_mode = getInitParam("widescreen");
644 switchPanelMode(_widescreen_mode);
646 console.log("second stage ok");
648 if (getInitParam("simple_update")) {
649 console.log("scheduling simple feed updater...");
650 window.setTimeout(update_random_feed, 30*1000);
654 function quickMenuGo(opid) {
663 displayDlg(__("Tag cloud"), "printTagCloud");
672 window.location.href = "backend.php?op=digest";
675 if (activeFeedIsCat())
676 alert(__("You can't edit this kind of feed."));
678 editFeed(getActiveFeedId());
680 case "qmcRemoveFeed":
681 var actid = getActiveFeedId();
683 if (activeFeedIsCat()) {
684 alert(__("You can't unsubscribe from the category."));
689 alert(__("Please select some feed first."));
693 var fn = getFeedName(actid);
695 var pr = __("Unsubscribe from %s?").replace("%s", fn);
698 unsubscribeFeed(actid);
701 case "qmcCatchupAll":
704 case "qmcShowOnlyUnread":
713 case "qmcRescoreFeed":
714 rescoreCurrentFeed();
716 case "qmcToggleWidescreen":
718 _widescreen_mode = !_widescreen_mode;
720 // reset stored sizes because geometry changed
721 setCookie("ttrss_ci_width", 0);
722 setCookie("ttrss_ci_height", 0);
724 switchPanelMode(_widescreen_mode);
726 alert(__("Widescreen is not available in combined mode."));
733 console.log("quickMenuGo: unknown action: " + opid);
737 function toggleDispRead() {
739 var hide = !(getInitParam("hide_read_feeds") == "1");
741 hideOrShowFeeds(hide);
743 var query = "?op=rpc&method=setpref&key=HIDE_READ_FEEDS&value=" +
746 setInitParam("hide_read_feeds", hide);
748 new Ajax.Request("backend.php", {
750 onComplete: function(transport) {
755 function parse_runtime_info(data) {
757 //console.log("parsing runtime info...");
762 // console.log("RI: " + k + " => " + v);
764 if (k == "dep_ts" && parseInt(getInitParam("dep_ts")) > 0) {
765 if (parseInt(getInitParam("dep_ts")) < parseInt(v) && getInitParam("reload_on_ts_change")) {
766 window.location.reload();
770 if (k == "daemon_is_running" && v != 1) {
771 notify_error("<span onclick=\"explainError(1)\">Update daemon is not running.</span>", true);
775 if (k == "update_result") {
776 var updatesIcon = dijit.byId("updatesIcon").domNode;
779 Element.show(updatesIcon);
781 Element.hide(updatesIcon);
785 if (k == "daemon_stamp_ok" && v != 1) {
786 notify_error("<span onclick=\"explainError(3)\">Update daemon is not updating feeds.</span>", true);
790 if (k == "max_feed_id" || k == "num_feeds") {
791 if (init_params[k] != v) {
792 console.log("feed count changed, need to reload feedlist.");
801 PluginHost.run(PluginHost.HOOK_RUNTIME_INFO_LOADED, data);
804 function collapse_feedlist() {
805 Element.toggle("feeds-holder");
807 var splitter = $("feeds-holder_splitter");
809 Element.visible("feeds-holder") ? splitter.show() : splitter.hide();
811 dijit.byId("main").resize();
814 function viewModeChanged() {
816 return viewCurrentFeed('');
819 function rescoreCurrentFeed() {
821 var actid = getActiveFeedId();
823 if (activeFeedIsCat() || actid < 0) {
824 alert(__("You can't rescore this kind of feed."));
829 alert(__("Please select some feed first."));
833 var fn = getFeedName(actid);
834 var pr = __("Rescore articles in %s?").replace("%s", fn);
837 notify_progress("Rescoring articles...");
839 var query = "?op=pref-feeds&method=rescore&quiet=1&ids=" + actid;
841 new Ajax.Request("backend.php", {
843 onComplete: function(transport) {
849 function hotkey_handler(e) {
851 if (e.target.nodeName == "INPUT" || e.target.nodeName == "TEXTAREA") return;
855 var cmdline = $('cmdline');
858 keycode = window.event.keyCode;
863 if (keycode == 27) { // escape
864 hotkey_prefix = false;
867 if (keycode == 16) return; // ignore lone shift
868 if (keycode == 17) return; // ignore lone ctrl
870 var hotkeys = getInitParam("hotkeys");
871 var keychar = String.fromCharCode(keycode).toLowerCase();
873 if (!hotkey_prefix && hotkeys[0].indexOf(keychar) != -1) {
875 var date = new Date();
876 var ts = Math.round(date.getTime() / 1000);
878 hotkey_prefix = keychar;
879 hotkey_prefix_pressed = ts;
881 cmdline.innerHTML = keychar;
882 Element.show(cmdline);
886 // returning false here literally disables ctrl-c in browser lol (because C is a valid prefix)
890 Element.hide(cmdline);
892 var hotkey = keychar.search(/[a-zA-Z0-9]/) != -1 ? keychar : "(" + keycode + ")";
894 // ensure ^*char notation
895 if (e.shiftKey) hotkey = "*" + hotkey;
896 if (e.ctrlKey) hotkey = "^" + hotkey;
897 if (e.altKey) hotkey = "+" + hotkey;
898 if (e.metaKey) hotkey = "%" + hotkey;
900 hotkey = hotkey_prefix ? hotkey_prefix + " " + hotkey : hotkey;
901 hotkey_prefix = false;
903 var hotkey_action = false;
904 var hotkeys = getInitParam("hotkeys");
906 for (sequence in hotkeys[1]) {
907 if (sequence == hotkey) {
908 hotkey_action = hotkeys[1][sequence];
913 var action = hotkey_actions[hotkey_action];
915 if (action != null) {
922 function inPreferences() {
926 function reverseHeadlineOrder() {
928 var toolbar = document.forms["main_toolbar_form"];
929 var order_by = dijit.getEnclosingWidget(toolbar.order_by);
931 var value = order_by.attr('value');
933 if (value == "date_reverse")
936 value = "date_reverse";
938 order_by.attr('value', value);
944 function handle_rpc_json(transport, scheduled_call) {
946 var netalert_dijit = dijit.byId("net-alert");
947 var netalert = false;
949 if (netalert_dijit) netalert = netalert_dijit.domNode;
952 var reply = JSON.parse(transport.responseText);
956 var error = reply['error'];
959 var code = error['code'];
960 var msg = error['msg'];
962 console.warn("[handle_rpc_json] received fatal error " + code + "/" + msg);
965 fatalError(code, msg);
970 var seq = reply['seq'];
973 if (get_seq() != seq) {
974 console.log("[handle_rpc_json] sequence mismatch: " + seq +
975 " (want: " + get_seq() + ")");
980 var message = reply['message'];
983 if (message == "UPDATE_COUNTERS") {
984 console.log("need to refresh counters...");
985 setInitParam("last_article_id", -1);
986 request_counters(true);
990 var counters = reply['counters'];
993 parse_counters(counters, scheduled_call);
995 var runtime_info = reply['runtime-info'];
998 parse_runtime_info(runtime_info);
1000 if (netalert) netalert.hide();
1006 notify_error("Communication problem with server.");
1013 notify_error("Communication problem with server.");
1021 function switchPanelMode(wide) {
1022 if (isCdmMode()) return;
1024 article_id = getActiveArticleId();
1027 dijit.byId("headlines-wrap-inner").attr("design", 'sidebar');
1028 dijit.byId("content-insert").attr("region", "trailing");
1030 dijit.byId("content-insert").domNode.setStyle({width: '50%',
1032 borderTopWidth: '0px' });
1034 if (parseInt(getCookie("ttrss_ci_width")) > 0) {
1035 dijit.byId("content-insert").domNode.setStyle(
1036 {width: getCookie("ttrss_ci_width") + "px" });
1039 $("headlines-frame").setStyle({ borderBottomWidth: '0px' });
1040 $("headlines-frame").addClassName("wide");
1044 dijit.byId("content-insert").attr("region", "bottom");
1046 dijit.byId("content-insert").domNode.setStyle({width: 'auto',
1048 borderTopWidth: '0px'});
1050 if (parseInt(getCookie("ttrss_ci_height")) > 0) {
1051 dijit.byId("content-insert").domNode.setStyle(
1052 {height: getCookie("ttrss_ci_height") + "px" });
1055 $("headlines-frame").setStyle({ borderBottomWidth: '1px' });
1056 $("headlines-frame").removeClassName("wide");
1060 closeArticlePanel();
1062 if (article_id) view(article_id);
1064 new Ajax.Request("backend.php", {
1065 parameters: "op=rpc&method=setpanelmode&wide=" + (wide ? 1 : 0),
1066 onComplete: function(transport) {
1067 console.log(transport.responseText);
1071 function update_random_feed() {
1072 console.log("in update_random_feed");
1074 new Ajax.Request("backend.php", {
1075 parameters: "op=rpc&method=updateRandomFeed",
1076 onComplete: function(transport) {
1077 handle_rpc_json(transport, true);
1078 window.setTimeout(update_random_feed, 30*1000);
1082 function hash_get(key) {
1083 kv = window.location.hash.substring(1).toQueryParams();
1086 function hash_set(key, value) {
1087 kv = window.location.hash.substring(1).toQueryParams();
1089 window.location.hash = $H(kv).toQueryString();