tt-rss.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149
  1. var total_unread = 0;
  2. var global_unread = -1;
  3. var firsttime_update = true;
  4. var _active_feed_id = 0;
  5. var _active_feed_is_cat = false;
  6. var hotkey_prefix = false;
  7. var hotkey_prefix_pressed = false;
  8. var init_params = {};
  9. var _force_scheduled_update = false;
  10. var last_scheduled_update = false;
  11. var _rpc_seq = 0;
  12. function next_seq() {
  13. _rpc_seq += 1;
  14. return _rpc_seq;
  15. }
  16. function get_seq() {
  17. return _rpc_seq;
  18. }
  19. function activeFeedIsCat() {
  20. return _active_feed_is_cat;
  21. }
  22. function getActiveFeedId() {
  23. try {
  24. //console.log("gAFID: " + _active_feed_id);
  25. return _active_feed_id;
  26. } catch (e) {
  27. exception_error("getActiveFeedId", e);
  28. }
  29. }
  30. function setActiveFeedId(id, is_cat) {
  31. try {
  32. _active_feed_id = id;
  33. if (is_cat != undefined) {
  34. _active_feed_is_cat = is_cat;
  35. }
  36. selectFeed(id, is_cat);
  37. } catch (e) {
  38. exception_error("setActiveFeedId", e);
  39. }
  40. }
  41. function updateFeedList() {
  42. try {
  43. // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
  44. // __("Loading, please wait...") + "</div>";
  45. Element.show("feedlistLoading");
  46. if (dijit.byId("feedTree")) {
  47. dijit.byId("feedTree").destroyRecursive();
  48. }
  49. var store = new dojo.data.ItemFileWriteStore({
  50. url: "backend.php?op=feeds"});
  51. var treeModel = new fox.FeedStoreModel({
  52. store: store,
  53. query: {
  54. "type": "feed"
  55. },
  56. rootId: "root",
  57. rootLabel: "Feeds",
  58. childrenAttrs: ["items"]
  59. });
  60. var tree = new fox.FeedTree({
  61. persist: false,
  62. model: treeModel,
  63. onOpen: function (item, node) {
  64. var id = String(item.id);
  65. var cat_id = id.substr(id.indexOf(":")+1);
  66. new Ajax.Request("backend.php",
  67. { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
  68. param_escape(cat_id) + "&mode=0" } );
  69. },
  70. onClose: function (item, node) {
  71. var id = String(item.id);
  72. var cat_id = id.substr(id.indexOf(":")+1);
  73. new Ajax.Request("backend.php",
  74. { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
  75. param_escape(cat_id) + "&mode=1" } );
  76. },
  77. onClick: function (item, node) {
  78. var id = String(item.id);
  79. var is_cat = id.match("^CAT:");
  80. var feed = id.substr(id.indexOf(":")+1);
  81. viewfeed(feed, '', is_cat);
  82. return false;
  83. },
  84. openOnClick: false,
  85. showRoot: false,
  86. id: "feedTree",
  87. }, "feedTree");
  88. /* var menu = new dijit.Menu({id: 'feedMenu'});
  89. menu.addChild(new dijit.MenuItem({
  90. label: "Simple menu item"
  91. }));
  92. // menu.bindDomNode(tree.domNode); */
  93. var tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
  94. console.log(dijit.getEnclosingWidget(event.target));
  95. dojo.disconnect(tmph);
  96. });
  97. $("feeds-holder").appendChild(tree.domNode);
  98. var tmph = dojo.connect(tree, 'onLoad', function() {
  99. dojo.disconnect(tmph);
  100. Element.hide("feedlistLoading");
  101. tree.collapseHiddenCats();
  102. feedlist_init();
  103. // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
  104. // menu.bindDomNode(node);
  105. loading_set_progress(25);
  106. });
  107. tree.startup();
  108. } catch (e) {
  109. exception_error("updateFeedList", e);
  110. }
  111. }
  112. function catchupAllFeeds() {
  113. var str = __("Mark all articles as read?");
  114. if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
  115. var query_str = "backend.php?op=feeds&subop=catchupAll";
  116. notify_progress("Marking all feeds as read...");
  117. //console.log("catchupAllFeeds Q=" + query_str);
  118. new Ajax.Request("backend.php", {
  119. parameters: query_str,
  120. onComplete: function(transport) {
  121. feedlist_callback2(transport);
  122. } });
  123. global_unread = 0;
  124. updateTitle("");
  125. }
  126. }
  127. function viewCurrentFeed(subop) {
  128. if (getActiveFeedId() != undefined) {
  129. viewfeed(getActiveFeedId(), subop, activeFeedIsCat());
  130. }
  131. return false; // block unneeded form submits
  132. }
  133. function timeout() {
  134. if (getInitParam("bw_limit") == "1") return;
  135. try {
  136. var date = new Date();
  137. var ts = Math.round(date.getTime() / 1000);
  138. if (ts - last_scheduled_update > 10 || _force_scheduled_update) {
  139. //console.log("timeout()");
  140. window.clearTimeout(counter_timeout_id);
  141. var query_str = "?op=rpc&subop=getAllCounters&seq=" + next_seq();
  142. var omode;
  143. if (firsttime_update && !navigator.userAgent.match("Opera")) {
  144. firsttime_update = false;
  145. omode = "T";
  146. } else {
  147. omode = "flc";
  148. }
  149. query_str = query_str + "&omode=" + omode;
  150. if (!_force_scheduled_update)
  151. query_str = query_str + "&last_article_id=" + getInitParam("last_article_id");
  152. //console.log("[timeout]" + query_str);
  153. new Ajax.Request("backend.php", {
  154. parameters: query_str,
  155. onComplete: function(transport) {
  156. handle_rpc_json(transport, !_force_scheduled_update);
  157. _force_scheduled_update = false;
  158. } });
  159. last_scheduled_update = ts;
  160. }
  161. } catch (e) {
  162. exception_error("timeout", e);
  163. }
  164. setTimeout("timeout()", 3000);
  165. }
  166. function search() {
  167. var query = "backend.php?op=dlg&id=search&param=" +
  168. param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
  169. if (dijit.byId("searchDlg"))
  170. dijit.byId("searchDlg").destroyRecursive();
  171. dialog = new dijit.Dialog({
  172. id: "searchDlg",
  173. title: __("Search"),
  174. style: "width: 600px",
  175. execute: function() {
  176. if (this.validate()) {
  177. _search_query = dojo.objectToQuery(this.attr('value'));
  178. this.hide();
  179. viewCurrentFeed();
  180. }
  181. },
  182. href: query});
  183. dialog.show();
  184. }
  185. function updateTitle() {
  186. var tmp = "Tiny Tiny RSS";
  187. if (global_unread > 0) {
  188. tmp = tmp + " (" + global_unread + ")";
  189. }
  190. if (window.fluid) {
  191. if (global_unread > 0) {
  192. window.fluid.dockBadge = global_unread;
  193. } else {
  194. window.fluid.dockBadge = "";
  195. }
  196. }
  197. document.title = tmp;
  198. }
  199. function genericSanityCheck() {
  200. setCookie("ttrss_test", "TEST");
  201. if (getCookie("ttrss_test") != "TEST") {
  202. return fatalError(2);
  203. }
  204. return true;
  205. }
  206. function init() {
  207. try {
  208. //Form.disable("main_toolbar_form");
  209. // Our layer takes care of Dojo dependencies.
  210. /* dojo.require("dijit.layout.BorderContainer");
  211. dojo.require("dijit.layout.TabContainer");
  212. dojo.require("dijit.layout.ContentPane");
  213. dojo.require("dijit.Dialog");
  214. dojo.require("dijit.form.Button");
  215. dojo.require("dijit.Menu");
  216. dojo.require("dojo.data.ItemFileWriteStore");
  217. dojo.require("dijit.Tree");
  218. dojo.require("dijit.form.Select");
  219. dojo.require("dijit.form.TextBox");
  220. dojo.require("dijit.form.ValidationTextBox");
  221. dojo.require("dijit.form.FilteringSelect");
  222. dojo.require("dijit.form.CheckBox");
  223. dojo.require("dijit.form.SimpleTextarea");
  224. dojo.require("dijit.Toolbar");
  225. dojo.require("dijit.ProgressBar");
  226. dojo.require("dijit.Menu");
  227. dojo.require("dojo.parser"); */
  228. dojo.registerModulePath("fox", "../..");
  229. dojo.require("fox.FeedTree");
  230. dojo.parser.parse();
  231. if (typeof themeBeforeLayout == 'function') {
  232. themeBeforeLayout();
  233. }
  234. dojo.addOnLoad(function() {
  235. updateFeedList();
  236. closeArticlePanel();
  237. if (typeof themeAfterLayout == 'function') {
  238. themeAfterLayout();
  239. }
  240. });
  241. if (!genericSanityCheck())
  242. return false;
  243. loading_set_progress(20);
  244. var hasAudio = !!((myAudioTag = document.createElement('audio')).canPlayType);
  245. new Ajax.Request("backend.php", {
  246. parameters: {op: "rpc", subop: "sanityCheck", hasAudio: hasAudio},
  247. onComplete: function(transport) {
  248. backend_sanity_check_callback(transport);
  249. } });
  250. } catch (e) {
  251. exception_error("init", e);
  252. }
  253. }
  254. function init_second_stage() {
  255. try {
  256. delCookie("ttrss_test");
  257. var toolbar = document.forms["main_toolbar_form"];
  258. dijit.getEnclosingWidget(toolbar.view_mode).attr('value',
  259. getInitParam("default_view_mode"));
  260. dijit.getEnclosingWidget(toolbar.order_by).attr('value',
  261. getInitParam("default_view_order_by"));
  262. feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
  263. loading_set_progress(30);
  264. if (has_local_storage())
  265. sessionStorage.clear();
  266. console.log("second stage ok");
  267. } catch (e) {
  268. exception_error("init_second_stage", e);
  269. }
  270. }
  271. function quickMenuGo(opid) {
  272. try {
  273. if (opid == "qmcPrefs") {
  274. gotoPreferences();
  275. }
  276. if (opid == "qmcTagCloud") {
  277. displayDlg("printTagCloud");
  278. }
  279. if (opid == "qmcSearch") {
  280. search();
  281. return;
  282. }
  283. if (opid == "qmcAddFeed") {
  284. quickAddFeed();
  285. return;
  286. }
  287. if (opid == "qmcDigest") {
  288. window.location.href = "digest.php";
  289. return;
  290. }
  291. if (opid == "qmcEditFeed") {
  292. if (activeFeedIsCat())
  293. alert(__("You can't edit this kind of feed."));
  294. else
  295. editFeed(getActiveFeedId());
  296. return;
  297. }
  298. if (opid == "qmcRemoveFeed") {
  299. var actid = getActiveFeedId();
  300. if (activeFeedIsCat()) {
  301. alert(__("You can't unsubscribe from the category."));
  302. return;
  303. }
  304. if (!actid) {
  305. alert(__("Please select some feed first."));
  306. return;
  307. }
  308. var fn = getFeedName(actid);
  309. var pr = __("Unsubscribe from %s?").replace("%s", fn);
  310. if (confirm(pr)) {
  311. unsubscribeFeed(actid);
  312. }
  313. return;
  314. }
  315. if (opid == "qmcCatchupAll") {
  316. catchupAllFeeds();
  317. return;
  318. }
  319. if (opid == "qmcShowOnlyUnread") {
  320. toggleDispRead();
  321. return;
  322. }
  323. if (opid == "qmcAddFilter") {
  324. quickAddFilter();
  325. return;
  326. }
  327. if (opid == "qmcAddLabel") {
  328. addLabel();
  329. return;
  330. }
  331. if (opid == "qmcRescoreFeed") {
  332. rescoreCurrentFeed();
  333. return;
  334. }
  335. if (opid == "qmcHKhelp") {
  336. //Element.show("hotkey_help_overlay");
  337. Effect.Appear("hotkey_help_overlay", {duration : 0.3});
  338. }
  339. } catch (e) {
  340. exception_error("quickMenuGo", e);
  341. }
  342. }
  343. function toggleDispRead() {
  344. try {
  345. var hide = !(getInitParam("hide_read_feeds") == "1");
  346. hideOrShowFeeds(hide);
  347. var query = "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
  348. param_escape(hide);
  349. setInitParam("hide_read_feeds", hide);
  350. new Ajax.Request("backend.php", {
  351. parameters: query,
  352. onComplete: function(transport) {
  353. } });
  354. } catch (e) {
  355. exception_error("toggleDispRead", e);
  356. }
  357. }
  358. function parse_runtime_info(data) {
  359. //console.log("parsing runtime info...");
  360. for (k in data) {
  361. var v = data[k];
  362. // console.log("RI: " + k + " => " + v);
  363. if (k == "new_version_available") {
  364. var icon = $("newVersionIcon");
  365. if (icon) {
  366. if (v == "1") {
  367. icon.style.display = "inline";
  368. } else {
  369. icon.style.display = "none";
  370. }
  371. }
  372. return;
  373. }
  374. var error_flag;
  375. if (k == "daemon_is_running" && v != 1) {
  376. notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
  377. return;
  378. }
  379. if (k == "daemon_stamp_ok" && v != 1) {
  380. notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
  381. return;
  382. }
  383. if (k == "max_feed_id" || k == "num_feeds") {
  384. if (init_params[k] != v) {
  385. console.log("feed count changed, need to reload feedlist.");
  386. updateFeedList();
  387. }
  388. }
  389. init_params[k] = v;
  390. notify('');
  391. }
  392. }
  393. function catchupCurrentFeed() {
  394. var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
  395. var str = __("Mark all articles in %s as read?").replace("%s", fn);
  396. if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
  397. return viewCurrentFeed('MarkAllRead')
  398. }
  399. }
  400. function catchupFeedInGroup(id) {
  401. try {
  402. var title = getFeedName(id);
  403. var str = __("Mark all articles in %s as read?").replace("%s", title);
  404. if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
  405. return viewCurrentFeed('MarkAllReadGR:' + id)
  406. }
  407. } catch (e) {
  408. exception_error("catchupFeedInGroup", e);
  409. }
  410. }
  411. function collapse_feedlist() {
  412. try {
  413. if (!Element.visible('feeds-holder')) {
  414. Element.show('feeds-holder');
  415. Element.show('feeds-holder_splitter');
  416. $("collapse_feeds_btn").innerHTML = "&lt;&lt;";
  417. } else {
  418. Element.hide('feeds-holder');
  419. Element.hide('feeds-holder_splitter');
  420. $("collapse_feeds_btn").innerHTML = "&gt;&gt;";
  421. }
  422. dijit.byId("main").resize();
  423. query = "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
  424. new Ajax.Request("backend.php", { parameters: query });
  425. } catch (e) {
  426. exception_error("collapse_feedlist", e);
  427. }
  428. }
  429. function viewModeChanged() {
  430. cache_flush();
  431. return viewCurrentFeed('')
  432. }
  433. function viewLimitChanged() {
  434. cache_flush();
  435. return viewCurrentFeed('')
  436. }
  437. /* function adjustArticleScore(id, score) {
  438. try {
  439. var pr = prompt(__("Assign score to article:"), score);
  440. if (pr != undefined) {
  441. var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
  442. new Ajax.Request("backend.php", {
  443. parameters: query,
  444. onComplete: function(transport) {
  445. viewCurrentFeed();
  446. } });
  447. }
  448. } catch (e) {
  449. exception_error("adjustArticleScore", e);
  450. }
  451. } */
  452. function rescoreCurrentFeed() {
  453. var actid = getActiveFeedId();
  454. if (activeFeedIsCat() || actid < 0) {
  455. alert(__("You can't rescore this kind of feed."));
  456. return;
  457. }
  458. if (!actid) {
  459. alert(__("Please select some feed first."));
  460. return;
  461. }
  462. var fn = getFeedName(actid);
  463. var pr = __("Rescore articles in %s?").replace("%s", fn);
  464. if (confirm(pr)) {
  465. notify_progress("Rescoring articles...");
  466. var query = "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid;
  467. new Ajax.Request("backend.php", {
  468. parameters: query,
  469. onComplete: function(transport) {
  470. viewCurrentFeed();
  471. } });
  472. }
  473. }
  474. function hotkey_handler(e) {
  475. try {
  476. if (e.target.nodeName == "INPUT" || e.target.nodeName == "TEXTAREA") return;
  477. var keycode;
  478. var shift_key = false;
  479. var cmdline = $('cmdline');
  480. try {
  481. shift_key = e.shiftKey;
  482. } catch (e) {
  483. }
  484. if (window.event) {
  485. keycode = window.event.keyCode;
  486. } else if (e) {
  487. keycode = e.which;
  488. }
  489. var keychar = String.fromCharCode(keycode);
  490. if (keycode == 27) { // escape
  491. if (Element.visible("hotkey_help_overlay")) {
  492. Element.hide("hotkey_help_overlay");
  493. }
  494. hotkey_prefix = false;
  495. }
  496. if (keycode == 16) return; // ignore lone shift
  497. if (keycode == 17) return; // ignore lone ctrl
  498. if ((keycode == 70 || keycode == 67 || keycode == 71)
  499. && !hotkey_prefix) {
  500. var date = new Date();
  501. var ts = Math.round(date.getTime() / 1000);
  502. hotkey_prefix = keycode;
  503. hotkey_prefix_pressed = ts;
  504. cmdline.innerHTML = keychar;
  505. Element.show(cmdline);
  506. console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
  507. return true;
  508. }
  509. if (Element.visible("hotkey_help_overlay")) {
  510. Element.hide("hotkey_help_overlay");
  511. }
  512. /* Global hotkeys */
  513. Element.hide(cmdline);
  514. if (!hotkey_prefix) {
  515. if ((keycode == 191 || keychar == '?') && shift_key) { // ?
  516. if (!Element.visible("hotkey_help_overlay")) {
  517. Effect.Appear("hotkey_help_overlay", {duration : 0.3});
  518. } else {
  519. Element.hide("hotkey_help_overlay");
  520. }
  521. return false;
  522. }
  523. if (keycode == 191 || keychar == '/') { // /
  524. search();
  525. return false;
  526. }
  527. if (keycode == 74) { // j
  528. var rv = dijit.byId("feedTree").getPreviousFeed(
  529. getActiveFeedId(), activeFeedIsCat());
  530. if (rv) viewfeed(rv[0], '', rv[1]);
  531. return;
  532. }
  533. if (keycode == 75) { // k
  534. var rv = dijit.byId("feedTree").getNextFeed(
  535. getActiveFeedId(), activeFeedIsCat());
  536. if (rv) viewfeed(rv[0], '', rv[1]);
  537. return;
  538. }
  539. if (shift_key && keycode == 40) { // shift-down
  540. catchupRelativeToArticle(1);
  541. return;
  542. }
  543. if (shift_key && keycode == 38) { // shift-up
  544. catchupRelativeToArticle(0);
  545. return;
  546. }
  547. if (shift_key && keycode == 78) { // N
  548. scrollArticle(50);
  549. return;
  550. }
  551. if (shift_key && keycode == 80) { // P
  552. scrollArticle(-50);
  553. return;
  554. }
  555. if (keycode == 68 && shift_key) { // shift-D
  556. dismissSelectedArticles();
  557. return;
  558. }
  559. if (keycode == 88 && shift_key) { // shift-X
  560. dismissReadArticles();
  561. return;
  562. }
  563. if (keycode == 78 || keycode == 40) { // n, down
  564. if (typeof moveToPost != 'undefined') {
  565. moveToPost('next');
  566. return false;
  567. }
  568. }
  569. if (keycode == 80 || keycode == 38) { // p, up
  570. if (typeof moveToPost != 'undefined') {
  571. moveToPost('prev');
  572. return false;
  573. }
  574. }
  575. if (keycode == 83 && shift_key) { // S
  576. selectionTogglePublished(undefined, false, true);
  577. return;
  578. }
  579. if (keycode == 83) { // s
  580. selectionToggleMarked(undefined, false, true);
  581. return;
  582. }
  583. if (keycode == 85) { // u
  584. selectionToggleUnread(undefined, false, true)
  585. return;
  586. }
  587. if (keycode == 84 && shift_key) { // T
  588. var id = getActiveArticleId();
  589. if (id) {
  590. editArticleTags(id, getActiveFeedId(), isCdmMode());
  591. return;
  592. }
  593. }
  594. if (keycode == 9) { // tab
  595. var id = getArticleUnderPointer();
  596. if (id) {
  597. var cb = $("RCHK-" + id);
  598. if (cb) {
  599. cb.checked = !cb.checked;
  600. toggleSelectRowById(cb, "RROW-" + id);
  601. return false;
  602. }
  603. }
  604. }
  605. if (keycode == 79) { // o
  606. if (getActiveArticleId()) {
  607. openArticleInNewWindow(getActiveArticleId());
  608. return;
  609. }
  610. }
  611. if (keycode == 81 && shift_key) { // Q
  612. if (typeof catchupAllFeeds != 'undefined') {
  613. catchupAllFeeds();
  614. return;
  615. }
  616. }
  617. if (keycode == 88 && !shift_key) { // x
  618. if (activeFeedIsCat()) {
  619. dijit.byId("feedTree").collapseCat(getActiveFeedId());
  620. return;
  621. }
  622. }
  623. }
  624. /* Prefix f */
  625. if (hotkey_prefix == 70) { // f
  626. hotkey_prefix = false;
  627. if (keycode == 81) { // q
  628. if (getActiveFeedId()) {
  629. catchupCurrentFeed();
  630. return;
  631. }
  632. }
  633. if (keycode == 82) { // r
  634. if (getActiveFeedId()) {
  635. viewfeed(getActiveFeedId(), '', activeFeedIsCat());
  636. return;
  637. }
  638. }
  639. if (keycode == 65) { // a
  640. toggleDispRead();
  641. return false;
  642. }
  643. if (keycode == 85) { // u
  644. if (getActiveFeedId()) {
  645. viewfeed(getActiveFeedId(), '');
  646. return false;
  647. }
  648. }
  649. if (keycode == 69) { // e
  650. if (activeFeedIsCat())
  651. alert(__("You can't edit this kind of feed."));
  652. else
  653. editFeed(getActiveFeedId());
  654. return;
  655. return false;
  656. }
  657. if (keycode == 83) { // s
  658. quickAddFeed();
  659. return false;
  660. }
  661. if (keycode == 67 && shift_key) { // C
  662. if (typeof catchupAllFeeds != 'undefined') {
  663. catchupAllFeeds();
  664. return false;
  665. }
  666. }
  667. if (keycode == 67) { // c
  668. if (getActiveFeedId()) {
  669. catchupCurrentFeed();
  670. return false;
  671. }
  672. }
  673. if (keycode == 87) { // w
  674. feeds_sort_by_unread = !feeds_sort_by_unread;
  675. return resort_feedlist();
  676. }
  677. if (keycode == 88) { // x
  678. reverseHeadlineOrder();
  679. return;
  680. }
  681. }
  682. /* Prefix c */
  683. if (hotkey_prefix == 67) { // c
  684. hotkey_prefix = false;
  685. if (keycode == 70) { // f
  686. quickAddFilter();
  687. return false;
  688. }
  689. if (keycode == 76) { // l
  690. addLabel();
  691. return false;
  692. }
  693. if (keycode == 83) { // s
  694. if (typeof collapse_feedlist != 'undefined') {
  695. collapse_feedlist();
  696. return false;
  697. }
  698. }
  699. if (keycode == 77) { // m
  700. // TODO: sortable feedlist
  701. return;
  702. }
  703. if (keycode == 78) { // n
  704. catchupRelativeToArticle(1);
  705. return;
  706. }
  707. if (keycode == 80) { // p
  708. catchupRelativeToArticle(0);
  709. return;
  710. }
  711. }
  712. /* Prefix g */
  713. if (hotkey_prefix == 71) { // g
  714. hotkey_prefix = false;
  715. if (keycode == 65) { // a
  716. viewfeed(-4);
  717. return false;
  718. }
  719. if (keycode == 83) { // s
  720. viewfeed(-1);
  721. return false;
  722. }
  723. if (keycode == 80 && shift_key) { // P
  724. gotoPreferences();
  725. return false;
  726. }
  727. if (keycode == 80) { // p
  728. viewfeed(-2);
  729. return false;
  730. }
  731. if (keycode == 70) { // f
  732. viewfeed(-3);
  733. return false;
  734. }
  735. if (keycode == 84 && shift_key) { // T
  736. toggleTags();
  737. return false;
  738. }
  739. }
  740. /* Cmd */
  741. if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
  742. hotkey_prefix = false;
  743. return;
  744. }
  745. if (hotkey_prefix) {
  746. console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
  747. } else {
  748. console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
  749. }
  750. } catch (e) {
  751. exception_error("hotkey_handler", e);
  752. }
  753. }
  754. function inPreferences() {
  755. return false;
  756. }
  757. function reverseHeadlineOrder() {
  758. try {
  759. var query_str = "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
  760. new Ajax.Request("backend.php", {
  761. parameters: query_str,
  762. onComplete: function(transport) {
  763. viewCurrentFeed();
  764. } });
  765. } catch (e) {
  766. exception_error("reverseHeadlineOrder", e);
  767. }
  768. }
  769. function showFeedsWithErrors() {
  770. displayDlg('feedUpdateErrors');
  771. }
  772. function scheduleFeedUpdate(id, is_cat) {
  773. try {
  774. if (!id) {
  775. id = getActiveFeedId();
  776. is_cat = activeFeedIsCat();
  777. }
  778. if (!id) {
  779. alert(__("Please select some feed first."));
  780. return;
  781. }
  782. var query = "?op=rpc&subop=scheduleFeedUpdate&id=" +
  783. param_escape(id) +
  784. "&is_cat=" + param_escape(is_cat);
  785. console.log(query);
  786. new Ajax.Request("backend.php", {
  787. parameters: query,
  788. onComplete: function(transport) {
  789. handle_rpc_json(transport);
  790. var reply = JSON.parse(transport.responseText);
  791. var message = reply['message'];
  792. if (message) {
  793. notify_info(message);
  794. return;
  795. }
  796. } });
  797. } catch (e) {
  798. exception_error("scheduleFeedUpdate", e);
  799. }
  800. }
  801. function newVersionDlg() {
  802. try {
  803. var query = "backend.php?op=dlg&id=newVersion";
  804. if (dijit.byId("newVersionDlg"))
  805. dijit.byId("newVersionDlg").destroyRecursive();
  806. dialog = new dijit.Dialog({
  807. id: "newVersionDlg",
  808. title: __("New version available!"),
  809. style: "width: 600px",
  810. href: query,
  811. });
  812. dialog.show();
  813. } catch (e) {
  814. exception_error("newVersionDlg", e);
  815. }
  816. }
  817. function handle_rpc_json(transport, scheduled_call) {
  818. try {
  819. var reply = JSON.parse(transport.responseText);
  820. if (reply) {
  821. var error = reply['error'];
  822. if (error) {
  823. var code = error['code'];
  824. var msg = error['msg'];
  825. console.warn("[handle_rpc_json] received fatal error " + code + "/" + msg);
  826. if (code != 0) {
  827. fatalError(code, msg);
  828. return false;
  829. }
  830. }
  831. var seq = reply['seq'];
  832. if (seq) {
  833. if (get_seq() != seq) {
  834. console.log("[handle_rpc_json] sequence mismatch: " + seq +
  835. " (want: " + get_seq() + ")");
  836. return true;
  837. }
  838. }
  839. var message = reply['message'];
  840. if (message) {
  841. if (message == "UPDATE_COUNTERS") {
  842. console.log("need to refresh counters...");
  843. setInitParam("last_article_id", -1);
  844. _force_scheduled_update = true;
  845. }
  846. }
  847. var counters = reply['counters'];
  848. if (counters)
  849. parse_counters(counters, scheduled_call);
  850. var runtime_info = reply['runtime-info'];;
  851. if (runtime_info)
  852. parse_runtime_info(runtime_info);
  853. hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
  854. } else {
  855. notify_error("Error communicating with server.");
  856. }
  857. } catch (e) {
  858. notify_error("Error communicating with server.");
  859. console.log(e);
  860. //exception_error("handle_rpc_json", e, transport);
  861. }
  862. return true;
  863. }