split main objects to dojo modules
This commit is contained in:
parent
fda3ad39c8
commit
807ff07454
|
@ -114,7 +114,7 @@
|
||||||
require({cache:{}});
|
require({cache:{}});
|
||||||
<?php
|
<?php
|
||||||
print get_minified_js(["tt-rss.js",
|
print get_minified_js(["tt-rss.js",
|
||||||
"functions.js", "feedlist.js", "viewfeed.js", "PluginHost.js"]);
|
"functions.js", "PluginHost.js"]);
|
||||||
?>
|
?>
|
||||||
</script>
|
</script>
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
|
|
|
@ -0,0 +1,335 @@
|
||||||
|
define(["dojo/_base/declare"], function (declare) {
|
||||||
|
return declare("fox.Article", null, {
|
||||||
|
_active_article_id: 0,
|
||||||
|
selectionSetScore: function() {
|
||||||
|
const ids = Headlines.getSelected();
|
||||||
|
|
||||||
|
if (ids.length > 0) {
|
||||||
|
console.log(ids);
|
||||||
|
|
||||||
|
const score = prompt(__("Please enter new score for selected articles:"));
|
||||||
|
|
||||||
|
if (score != undefined) {
|
||||||
|
const query = {
|
||||||
|
op: "article", method: "setScore", id: ids.toString(),
|
||||||
|
score: score
|
||||||
|
};
|
||||||
|
|
||||||
|
xhrJson("backend.php", query, (reply) => {
|
||||||
|
if (reply) {
|
||||||
|
reply.id.each((id) => {
|
||||||
|
const row = $("RROW-" + id);
|
||||||
|
|
||||||
|
if (row) {
|
||||||
|
const pic = row.getElementsByClassName("score-pic")[0];
|
||||||
|
|
||||||
|
if (pic) {
|
||||||
|
pic.src = pic.src.replace(/score_.*?\.png/,
|
||||||
|
reply["score_pic"]);
|
||||||
|
pic.setAttribute("score", reply["score"]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
alert(__("No articles selected."));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
setScore: function(id, pic) {
|
||||||
|
const score = pic.getAttribute("score");
|
||||||
|
|
||||||
|
const new_score = prompt(__("Please enter new score for this article:"), score);
|
||||||
|
|
||||||
|
if (new_score != undefined) {
|
||||||
|
const query = {op: "article", method: "setScore", id: id, score: new_score};
|
||||||
|
|
||||||
|
xhrJson("backend.php", query, (reply) => {
|
||||||
|
if (reply) {
|
||||||
|
pic.src = pic.src.replace(/score_.*?\.png/, reply["score_pic"]);
|
||||||
|
pic.setAttribute("score", new_score);
|
||||||
|
pic.setAttribute("title", new_score);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
cdmUnsetActive: function(event) {
|
||||||
|
const row = $("RROW-" + Article.getActive());
|
||||||
|
|
||||||
|
if (row) {
|
||||||
|
row.removeClassName("active");
|
||||||
|
const cb = dijit.getEnclosingWidget(row.select(".rchk")[0]);
|
||||||
|
|
||||||
|
if (cb && !row.hasClassName("Selected"))
|
||||||
|
cb.attr("checked", false);
|
||||||
|
|
||||||
|
Article.setActive(0);
|
||||||
|
|
||||||
|
if (event)
|
||||||
|
event.stopPropagation();
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
close: function () {
|
||||||
|
if (dijit.byId("content-insert"))
|
||||||
|
dijit.byId("headlines-wrap-inner").removeChild(
|
||||||
|
dijit.byId("content-insert"));
|
||||||
|
},
|
||||||
|
displayUrl: function (id) {
|
||||||
|
const query = {op: "rpc", method: "getlinktitlebyid", id: id};
|
||||||
|
|
||||||
|
xhrJson("backend.php", query, (reply) => {
|
||||||
|
if (reply && reply.link) {
|
||||||
|
prompt(__("Article URL:"), reply.link);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
openInNewWindow: function (id) {
|
||||||
|
const w = window.open("");
|
||||||
|
w.opener = null;
|
||||||
|
w.location = "backend.php?op=article&method=redirect&id=" + id;
|
||||||
|
|
||||||
|
Article.setActive(id);
|
||||||
|
},
|
||||||
|
render: function (article) {
|
||||||
|
Utils.cleanupMemory("content-insert");
|
||||||
|
|
||||||
|
dijit.byId("headlines-wrap-inner").addChild(
|
||||||
|
dijit.byId("content-insert"));
|
||||||
|
|
||||||
|
const c = dijit.byId("content-insert");
|
||||||
|
|
||||||
|
try {
|
||||||
|
c.domNode.scrollTop = 0;
|
||||||
|
} catch (e) {
|
||||||
|
}
|
||||||
|
|
||||||
|
c.attr('content', article);
|
||||||
|
PluginHost.run(PluginHost.HOOK_ARTICLE_RENDERED, c.domNode);
|
||||||
|
|
||||||
|
Headlines.correctHeadlinesOffset(Article.getActive());
|
||||||
|
|
||||||
|
try {
|
||||||
|
c.focus();
|
||||||
|
} catch (e) {
|
||||||
|
}
|
||||||
|
},
|
||||||
|
view: function(id, noexpand) {
|
||||||
|
this.setActive(id);
|
||||||
|
|
||||||
|
if (!noexpand) {
|
||||||
|
console.log("loading article", id);
|
||||||
|
|
||||||
|
const cids = [];
|
||||||
|
|
||||||
|
/* only request uncached articles */
|
||||||
|
|
||||||
|
this.getRelativeIds(id).each((n) => {
|
||||||
|
if (!ArticleCache.get(n))
|
||||||
|
cids.push(n);
|
||||||
|
});
|
||||||
|
|
||||||
|
const cached_article = ArticleCache.get(id);
|
||||||
|
|
||||||
|
if (cached_article) {
|
||||||
|
console.log('rendering cached', id);
|
||||||
|
this.render(cached_article);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
xhrPost("backend.php", {op: "article", method: "view", id: id, cids: cids.toString()}, (transport) => {
|
||||||
|
try {
|
||||||
|
const reply = Utils.handleRpcJson(transport);
|
||||||
|
|
||||||
|
if (reply) {
|
||||||
|
|
||||||
|
reply.each(function (article) {
|
||||||
|
if (Article.getActive() == article['id']) {
|
||||||
|
Article.render(article['content']);
|
||||||
|
}
|
||||||
|
ArticleCache.set(article['id'], article['content']);
|
||||||
|
});
|
||||||
|
|
||||||
|
} else {
|
||||||
|
console.error("Invalid object received: " + transport.responseText);
|
||||||
|
|
||||||
|
Article.render("<div class='whiteBox'>" +
|
||||||
|
__('Could not display article (invalid object received - see error console for details)') + "</div>");
|
||||||
|
}
|
||||||
|
|
||||||
|
//const unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length;
|
||||||
|
//request_counters(unread_in_buffer == 0);
|
||||||
|
|
||||||
|
notify("");
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
exception_error(e);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
editTags: function(id) {
|
||||||
|
const query = "backend.php?op=article&method=editArticleTags¶m=" + param_escape(id);
|
||||||
|
|
||||||
|
if (dijit.byId("editTagsDlg"))
|
||||||
|
dijit.byId("editTagsDlg").destroyRecursive();
|
||||||
|
|
||||||
|
const dialog = new dijit.Dialog({
|
||||||
|
id: "editTagsDlg",
|
||||||
|
title: __("Edit article Tags"),
|
||||||
|
style: "width: 600px",
|
||||||
|
execute: function () {
|
||||||
|
if (this.validate()) {
|
||||||
|
notify_progress("Saving article tags...", true);
|
||||||
|
|
||||||
|
xhrPost("backend.php", this.attr('value'), (transport) => {
|
||||||
|
try {
|
||||||
|
notify('');
|
||||||
|
dialog.hide();
|
||||||
|
|
||||||
|
const data = JSON.parse(transport.responseText);
|
||||||
|
|
||||||
|
if (data) {
|
||||||
|
const id = data.id;
|
||||||
|
|
||||||
|
const tags = $("ATSTR-" + id);
|
||||||
|
const tooltip = dijit.byId("ATSTRTIP-" + id);
|
||||||
|
|
||||||
|
if (tags) tags.innerHTML = data.content;
|
||||||
|
if (tooltip) tooltip.attr('label', data.content_full);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
exception_error(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
href: query
|
||||||
|
});
|
||||||
|
|
||||||
|
const tmph = dojo.connect(dialog, 'onLoad', function () {
|
||||||
|
dojo.disconnect(tmph);
|
||||||
|
|
||||||
|
new Ajax.Autocompleter('tags_str', 'tags_choices',
|
||||||
|
"backend.php?op=article&method=completeTags",
|
||||||
|
{tokens: ',', paramName: "search"});
|
||||||
|
});
|
||||||
|
|
||||||
|
dialog.show();
|
||||||
|
},
|
||||||
|
cdmScrollToId: function(id, force) {
|
||||||
|
const ctr = $("headlines-frame");
|
||||||
|
const e = $("RROW-" + id);
|
||||||
|
|
||||||
|
if (!e || !ctr) return;
|
||||||
|
|
||||||
|
if (force || e.offsetTop + e.offsetHeight > (ctr.scrollTop + ctr.offsetHeight) ||
|
||||||
|
e.offsetTop < ctr.scrollTop) {
|
||||||
|
|
||||||
|
// expanded cdm has a 4px margin now
|
||||||
|
ctr.scrollTop = parseInt(e.offsetTop) - 4;
|
||||||
|
|
||||||
|
Element.hide("floatingTitle");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
setActive: function(id) {
|
||||||
|
console.log("setActive", id);
|
||||||
|
|
||||||
|
$$("div[id*=RROW][class*=active]").each((e) => {
|
||||||
|
e.removeClassName("active");
|
||||||
|
|
||||||
|
if (!e.hasClassName("Selected")) {
|
||||||
|
const cb = dijit.getEnclosingWidget(e.select(".rchk")[0]);
|
||||||
|
if (cb) cb.attr("checked", false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
this._active_article_id = id;
|
||||||
|
|
||||||
|
const row = $("RROW-" + id);
|
||||||
|
|
||||||
|
if (row) {
|
||||||
|
if (row.hasAttribute("data-content")) {
|
||||||
|
console.log("unpacking: " + row.id);
|
||||||
|
|
||||||
|
row.select(".content-inner")[0].innerHTML = row.getAttribute("data-content");
|
||||||
|
row.removeAttribute("data-content");
|
||||||
|
|
||||||
|
PluginHost.run(PluginHost.HOOK_ARTICLE_RENDERED_CDM, row);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (row.hasClassName("Unread")) {
|
||||||
|
|
||||||
|
Headlines.catchupBatched(() => {
|
||||||
|
Feeds.decrementFeedCounter(Feeds.getActive(), Feeds.activeIsCat());
|
||||||
|
Headlines.toggleUnread(id, 0);
|
||||||
|
Headlines.updateFloatingTitle(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
row.addClassName("active");
|
||||||
|
|
||||||
|
if (!row.hasClassName("Selected")) {
|
||||||
|
const cb = dijit.getEnclosingWidget(row.select(".rchk")[0]);
|
||||||
|
if (cb) cb.attr("checked", true);
|
||||||
|
}
|
||||||
|
|
||||||
|
PluginHost.run(PluginHost.HOOK_ARTICLE_SET_ACTIVE, this._active_article_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
Headlines.updateSelectedPrompt();
|
||||||
|
},
|
||||||
|
getActive: function() {
|
||||||
|
return this._active_article_id;
|
||||||
|
},
|
||||||
|
scroll: function(offset) {
|
||||||
|
if (!App.isCombinedMode()) {
|
||||||
|
const ci = $("content-insert");
|
||||||
|
if (ci) {
|
||||||
|
ci.scrollTop += offset;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const hi = $("headlines-frame");
|
||||||
|
if (hi) {
|
||||||
|
hi.scrollTop += offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getRelativeIds: function(id, limit) {
|
||||||
|
|
||||||
|
const tmp = [];
|
||||||
|
|
||||||
|
if (!limit) limit = 6; //3
|
||||||
|
|
||||||
|
const ids = Headlines.getLoaded();
|
||||||
|
|
||||||
|
for (let i = 0; i < ids.length; i++) {
|
||||||
|
if (ids[i] == id) {
|
||||||
|
for (let k = 1; k <= limit; k++) {
|
||||||
|
//if (i > k-1) tmp.push(ids[i-k]);
|
||||||
|
if (i < ids.length - k) tmp.push(ids[i + k]);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return tmp;
|
||||||
|
},
|
||||||
|
mouseIn: function(id) {
|
||||||
|
this.post_under_pointer = id;
|
||||||
|
},
|
||||||
|
mouseOut: function(id) {
|
||||||
|
this.post_under_pointer = false;
|
||||||
|
},
|
||||||
|
getUnderPointer: function() {
|
||||||
|
return this.post_under_pointer;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,25 @@
|
||||||
|
define(["dojo/_base/declare"], function (declare) {
|
||||||
|
return declare("fox.ArticleCache", null, {
|
||||||
|
has_storage: 'sessionStorage' in window && window['sessionStorage'] !== null,
|
||||||
|
set: function(id, obj) {
|
||||||
|
if (this.has_storage)
|
||||||
|
try {
|
||||||
|
sessionStorage["article:" + id] = obj;
|
||||||
|
} catch (e) {
|
||||||
|
sessionStorage.clear();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
get: function(id) {
|
||||||
|
if (this.has_storage)
|
||||||
|
return sessionStorage["article:" + id];
|
||||||
|
},
|
||||||
|
clear: function() {
|
||||||
|
if (this.has_storage)
|
||||||
|
sessionStorage.clear();
|
||||||
|
},
|
||||||
|
del: function(id) {
|
||||||
|
if (this.has_storage)
|
||||||
|
sessionStorage.removeItem("article:" + id);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,638 @@
|
||||||
|
define(["dojo/_base/declare"], function (declare) {
|
||||||
|
return declare("fox.Feeds", null, {
|
||||||
|
counters_last_request: 0,
|
||||||
|
_active_feed_id: 0,
|
||||||
|
_active_feed_is_cat: false,
|
||||||
|
infscroll_in_progress: 0,
|
||||||
|
infscroll_disabled: 0,
|
||||||
|
_infscroll_timeout: false,
|
||||||
|
_search_query: false,
|
||||||
|
last_search_query: [],
|
||||||
|
_viewfeed_wait_timeout: false,
|
||||||
|
_counters_prev: [],
|
||||||
|
// NOTE: this implementation is incomplete
|
||||||
|
// for general objects but good enough for counters
|
||||||
|
// http://adripofjavascript.com/blog/drips/object-equality-in-javascript.html
|
||||||
|
counterEquals: function(a, b) {
|
||||||
|
// Create arrays of property names
|
||||||
|
const aProps = Object.getOwnPropertyNames(a);
|
||||||
|
const bProps = Object.getOwnPropertyNames(b);
|
||||||
|
|
||||||
|
// If number of properties is different,
|
||||||
|
// objects are not equivalent
|
||||||
|
if (aProps.length != bProps.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < aProps.length; i++) {
|
||||||
|
const propName = aProps[i];
|
||||||
|
|
||||||
|
// If values of same property are not equal,
|
||||||
|
// objects are not equivalent
|
||||||
|
if (a[propName] !== b[propName]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we made it this far, objects
|
||||||
|
// are considered equivalent
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
resetCounters: function () {
|
||||||
|
this._counters_prev = [];
|
||||||
|
},
|
||||||
|
parseCounters: function (elems) {
|
||||||
|
for (let l = 0; l < elems.length; l++) {
|
||||||
|
|
||||||
|
if (Feeds._counters_prev[l] && this.counterEquals(elems[l], this._counters_prev[l])) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const id = elems[l].id;
|
||||||
|
const kind = elems[l].kind;
|
||||||
|
const ctr = parseInt(elems[l].counter);
|
||||||
|
const error = elems[l].error;
|
||||||
|
const has_img = elems[l].has_img;
|
||||||
|
const updated = elems[l].updated;
|
||||||
|
const auxctr = parseInt(elems[l].auxcounter);
|
||||||
|
|
||||||
|
if (id == "global-unread") {
|
||||||
|
App.global_unread = ctr;
|
||||||
|
App.updateTitle();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (id == "subscribed-feeds") {
|
||||||
|
/* feeds_found = ctr; */
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*if (this.getUnread(id, (kind == "cat")) != ctr ||
|
||||||
|
(kind == "cat")) {
|
||||||
|
}*/
|
||||||
|
|
||||||
|
this.setUnread(id, (kind == "cat"), ctr);
|
||||||
|
this.setValue(id, (kind == "cat"), 'auxcounter', auxctr);
|
||||||
|
|
||||||
|
if (kind != "cat") {
|
||||||
|
this.setValue(id, false, 'error', error);
|
||||||
|
this.setValue(id, false, 'updated', updated);
|
||||||
|
|
||||||
|
if (id > 0) {
|
||||||
|
if (has_img) {
|
||||||
|
this.setIcon(id, false,
|
||||||
|
getInitParam("icons_url") + "/" + id + ".ico?" + has_img);
|
||||||
|
} else {
|
||||||
|
this.setIcon(id, false, 'images/blank_icon.gif');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
|
||||||
|
this._counters_prev = elems;
|
||||||
|
},
|
||||||
|
reloadCurrent: function(method) {
|
||||||
|
console.log("reloadCurrent: " + method);
|
||||||
|
|
||||||
|
if (this.getActive() != undefined) {
|
||||||
|
this.open({feed: this.getActive(), is_cat: this.activeIsCat(), method: method});
|
||||||
|
}
|
||||||
|
return false; // block unneeded form submits
|
||||||
|
},
|
||||||
|
openNextUnread: function() {
|
||||||
|
const is_cat = this.activeIsCat();
|
||||||
|
const nuf = this.getNextUnread(this.getActive(), is_cat);
|
||||||
|
if (nuf) this.open({feed: nuf, is_cat: is_cat});
|
||||||
|
},
|
||||||
|
toggle: function() {
|
||||||
|
Element.toggle("feeds-holder");
|
||||||
|
|
||||||
|
const splitter = $("feeds-holder_splitter");
|
||||||
|
|
||||||
|
Element.visible("feeds-holder") ? splitter.show() : splitter.hide();
|
||||||
|
|
||||||
|
dijit.byId("main").resize();
|
||||||
|
},
|
||||||
|
cancelSearch: function() {
|
||||||
|
this._search_query = "";
|
||||||
|
this.reloadCurrent();
|
||||||
|
},
|
||||||
|
requestCounters: function(force) {
|
||||||
|
const date = new Date();
|
||||||
|
const timestamp = Math.round(date.getTime() / 1000);
|
||||||
|
|
||||||
|
if (force || timestamp - this.counters_last_request > 5) {
|
||||||
|
console.log("scheduling request of counters...");
|
||||||
|
|
||||||
|
this.counters_last_request = timestamp;
|
||||||
|
|
||||||
|
let query = {op: "rpc", method: "getAllCounters", seq: Utils.next_seq()};
|
||||||
|
|
||||||
|
if (!force)
|
||||||
|
query.last_article_id = getInitParam("last_article_id");
|
||||||
|
|
||||||
|
xhrPost("backend.php", query, (transport) => {
|
||||||
|
Utils.handleRpcJson(transport);
|
||||||
|
});
|
||||||
|
|
||||||
|
} else {
|
||||||
|
console.log("request_counters: rate limit reached: " + (timestamp - this.counters_last_request));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
reload: function() {
|
||||||
|
try {
|
||||||
|
Element.show("feedlistLoading");
|
||||||
|
|
||||||
|
this.resetCounters();
|
||||||
|
|
||||||
|
if (dijit.byId("feedTree")) {
|
||||||
|
dijit.byId("feedTree").destroyRecursive();
|
||||||
|
}
|
||||||
|
|
||||||
|
const store = new dojo.data.ItemFileWriteStore({
|
||||||
|
url: "backend.php?op=pref_feeds&method=getfeedtree&mode=2"
|
||||||
|
});
|
||||||
|
|
||||||
|
// noinspection JSUnresolvedFunction
|
||||||
|
const treeModel = new fox.FeedStoreModel({
|
||||||
|
store: store,
|
||||||
|
query: {
|
||||||
|
"type": getInitParam('enable_feed_cats') == 1 ? "category" : "feed"
|
||||||
|
},
|
||||||
|
rootId: "root",
|
||||||
|
rootLabel: "Feeds",
|
||||||
|
childrenAttrs: ["items"]
|
||||||
|
});
|
||||||
|
|
||||||
|
// noinspection JSUnresolvedFunction
|
||||||
|
const tree = new fox.FeedTree({
|
||||||
|
model: treeModel,
|
||||||
|
onClick: function (item/*, node*/) {
|
||||||
|
const id = String(item.id);
|
||||||
|
const is_cat = id.match("^CAT:");
|
||||||
|
const feed = id.substr(id.indexOf(":") + 1);
|
||||||
|
Feeds.open({feed: feed, is_cat: is_cat});
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
openOnClick: false,
|
||||||
|
showRoot: false,
|
||||||
|
persist: true,
|
||||||
|
id: "feedTree",
|
||||||
|
}, "feedTree");
|
||||||
|
|
||||||
|
const tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
|
||||||
|
console.log(dijit.getEnclosingWidget(event.target));
|
||||||
|
dojo.disconnect(tmph);
|
||||||
|
});
|
||||||
|
|
||||||
|
$("feeds-holder").appendChild(tree.domNode);
|
||||||
|
|
||||||
|
const tmph2 = dojo.connect(tree, 'onLoad', function () {
|
||||||
|
dojo.disconnect(tmph2);
|
||||||
|
Element.hide("feedlistLoading");
|
||||||
|
|
||||||
|
try {
|
||||||
|
Feeds.init();
|
||||||
|
Utils.setLoadingProgress(25);
|
||||||
|
} catch (e) {
|
||||||
|
exception_error(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
tree.startup();
|
||||||
|
} catch (e) {
|
||||||
|
exception_error(e);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
init: function() {
|
||||||
|
console.log("in feedlist init");
|
||||||
|
|
||||||
|
Utils.setLoadingProgress(50);
|
||||||
|
|
||||||
|
document.onkeydown = () => { App.hotkeyHandler(event) };
|
||||||
|
window.setInterval(() => { Headlines.catchupBatched() }, 10 * 1000);
|
||||||
|
|
||||||
|
if (!this.getActive()) {
|
||||||
|
this.open({feed: -3});
|
||||||
|
} else {
|
||||||
|
this.open({feed: this.getActive(), is_cat: this.activeIsCat()});
|
||||||
|
}
|
||||||
|
|
||||||
|
this.hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
|
||||||
|
|
||||||
|
if (getInitParam("is_default_pw")) {
|
||||||
|
console.warn("user password is at default value");
|
||||||
|
|
||||||
|
const dialog = new dijit.Dialog({
|
||||||
|
title: __("Your password is at default value"),
|
||||||
|
href: "backend.php?op=dlg&method=defaultpasswordwarning",
|
||||||
|
id: 'infoBox',
|
||||||
|
style: "width: 600px",
|
||||||
|
onCancel: function () {
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
onExecute: function () {
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
onClose: function () {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
dialog.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
// bw_limit disables timeout() so we request initial counters separately
|
||||||
|
if (getInitParam("bw_limit") == "1") {
|
||||||
|
this.requestCounters(true);
|
||||||
|
} else {
|
||||||
|
setTimeout(() => {
|
||||||
|
this.requestCounters(true);
|
||||||
|
setInterval(() => { this.requestCounters(); }, 60 * 1000)
|
||||||
|
}, 250);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
activeIsCat: function() {
|
||||||
|
return !!this._active_feed_is_cat;
|
||||||
|
},
|
||||||
|
getActive: function() {
|
||||||
|
return this._active_feed_id;
|
||||||
|
},
|
||||||
|
setActive: function(id, is_cat) {
|
||||||
|
hash_set('f', id);
|
||||||
|
hash_set('c', is_cat ? 1 : 0);
|
||||||
|
|
||||||
|
this._active_feed_id = id;
|
||||||
|
this._active_feed_is_cat = is_cat;
|
||||||
|
|
||||||
|
$("headlines-frame").setAttribute("feed-id", id);
|
||||||
|
$("headlines-frame").setAttribute("is-cat", is_cat ? 1 : 0);
|
||||||
|
|
||||||
|
this.select(id, is_cat);
|
||||||
|
|
||||||
|
PluginHost.run(PluginHost.HOOK_FEED_SET_ACTIVE, [this._active_feed_id, this._active_feed_is_cat]);
|
||||||
|
},
|
||||||
|
select: function(feed, is_cat) {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree) return tree.selectFeed(feed, is_cat);
|
||||||
|
},
|
||||||
|
toggleUnread: function() {
|
||||||
|
const hide = !(getInitParam("hide_read_feeds") == "1");
|
||||||
|
|
||||||
|
xhrPost("backend.php", {op: "rpc", method: "setpref", key: "HIDE_READ_FEEDS", value: hide}, () => {
|
||||||
|
this.hideOrShowFeeds(hide);
|
||||||
|
setInitParam("hide_read_feeds", hide);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
hideOrShowFeeds: function(hide) {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree)
|
||||||
|
return tree.hideRead(hide, getInitParam("hide_read_shows_special"));
|
||||||
|
},
|
||||||
|
open: function(params) {
|
||||||
|
const feed = params.feed;
|
||||||
|
const is_cat = !!params.is_cat || false;
|
||||||
|
const offset = params.offset || 0;
|
||||||
|
const viewfeed_debug = params.viewfeed_debug;
|
||||||
|
const method = params.method;
|
||||||
|
// this is used to quickly switch between feeds, sets active but xhr is on a timeout
|
||||||
|
const delayed = params.delayed || false;
|
||||||
|
|
||||||
|
if (feed != this.getActive() || this.activeIsCat() != is_cat) {
|
||||||
|
this._search_query = false;
|
||||||
|
Article.setActive(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (offset != 0) {
|
||||||
|
if (this.infscroll_in_progress)
|
||||||
|
return;
|
||||||
|
|
||||||
|
this.infscroll_in_progress = 1;
|
||||||
|
|
||||||
|
window.clearTimeout(this._infscroll_timeout);
|
||||||
|
this._infscroll_timeout = window.setTimeout(() => {
|
||||||
|
console.log('infscroll request timed out, aborting');
|
||||||
|
this.infscroll_in_progress = 0;
|
||||||
|
|
||||||
|
// call scroll handler to maybe repeat infscroll request
|
||||||
|
Headlines.scrollHandler();
|
||||||
|
}, 10 * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
Form.enable("main_toolbar_form");
|
||||||
|
|
||||||
|
let query = Object.assign({op: "feeds", method: "view", feed: feed},
|
||||||
|
dojo.formToObject("main_toolbar_form"));
|
||||||
|
|
||||||
|
if (method) query.m = method;
|
||||||
|
|
||||||
|
if (offset > 0) {
|
||||||
|
if (Headlines.current_first_id) {
|
||||||
|
query.fid = Headlines.current_first_id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this._search_query) {
|
||||||
|
query = Object.assign(query, this._search_query);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (offset != 0) {
|
||||||
|
query.skip = offset;
|
||||||
|
|
||||||
|
// to prevent duplicate feed titles when showing grouped vfeeds
|
||||||
|
if (Headlines.vgroup_last_feed != undefined) {
|
||||||
|
query.vgrlf = Headlines.vgroup_last_feed;
|
||||||
|
}
|
||||||
|
} else if (!is_cat && feed == this.getActive() && !params.method) {
|
||||||
|
query.m = "ForceUpdate";
|
||||||
|
}
|
||||||
|
|
||||||
|
Form.enable("main_toolbar_form");
|
||||||
|
|
||||||
|
if (!delayed)
|
||||||
|
if (!this.setExpando(feed, is_cat,
|
||||||
|
(is_cat) ? 'images/indicator_tiny.gif' : 'images/indicator_white.gif'))
|
||||||
|
notify_progress("Loading, please wait...", true);
|
||||||
|
|
||||||
|
query.cat = is_cat;
|
||||||
|
|
||||||
|
this.setActive(feed, is_cat);
|
||||||
|
|
||||||
|
if (viewfeed_debug) {
|
||||||
|
window.open("backend.php?" +
|
||||||
|
dojo.objectToQuery(
|
||||||
|
Object.assign({debug: 1, csrf_token: getInitParam("csrf_token")}, query)
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
window.clearTimeout(this._viewfeed_wait_timeout);
|
||||||
|
this._viewfeed_wait_timeout = window.setTimeout(() => {
|
||||||
|
Headlines.catchupBatched(() => {
|
||||||
|
xhrPost("backend.php", query, (transport) => {
|
||||||
|
try {
|
||||||
|
window.clearTimeout(this._infscroll_timeout);
|
||||||
|
this.setExpando(feed, is_cat, 'images/blank_icon.gif');
|
||||||
|
Headlines.onLoaded(transport, offset);
|
||||||
|
PluginHost.run(PluginHost.HOOK_FEED_LOADED, [feed, is_cat]);
|
||||||
|
} catch (e) {
|
||||||
|
exception_error(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}, delayed ? 250 : 0);
|
||||||
|
},
|
||||||
|
catchupAll: function() {
|
||||||
|
const str = __("Mark all articles as read?");
|
||||||
|
|
||||||
|
if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
|
||||||
|
|
||||||
|
notify_progress("Marking all feeds as read...");
|
||||||
|
|
||||||
|
xhrPost("backend.php", {op: "feeds", method: "catchupAll"}, () => {
|
||||||
|
this.requestCounters(true);
|
||||||
|
this.reloadCurrent();
|
||||||
|
});
|
||||||
|
|
||||||
|
App.global_unread = 0;
|
||||||
|
App.updateTitle();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
decrementFeedCounter: function(feed, is_cat) {
|
||||||
|
let ctr = this.getUnread(feed, is_cat);
|
||||||
|
|
||||||
|
if (ctr > 0) {
|
||||||
|
this.setUnread(feed, is_cat, ctr - 1);
|
||||||
|
App.global_unread -= 1;
|
||||||
|
App.updateTitle();
|
||||||
|
|
||||||
|
if (!is_cat) {
|
||||||
|
const cat = parseInt(this.getCategory(feed));
|
||||||
|
|
||||||
|
if (!isNaN(cat)) {
|
||||||
|
ctr = this.getUnread(cat, true);
|
||||||
|
|
||||||
|
if (ctr > 0) {
|
||||||
|
this.setUnread(cat, true, ctr - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
catchupFeed: function(feed, is_cat, mode) {
|
||||||
|
if (is_cat == undefined) is_cat = false;
|
||||||
|
|
||||||
|
let str = false;
|
||||||
|
|
||||||
|
switch (mode) {
|
||||||
|
case "1day":
|
||||||
|
str = __("Mark %w in %s older than 1 day as read?");
|
||||||
|
break;
|
||||||
|
case "1week":
|
||||||
|
str = __("Mark %w in %s older than 1 week as read?");
|
||||||
|
break;
|
||||||
|
case "2week":
|
||||||
|
str = __("Mark %w in %s older than 2 weeks as read?");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
str = __("Mark %w in %s as read?");
|
||||||
|
}
|
||||||
|
|
||||||
|
const mark_what = this.last_search_query && this.last_search_query[0] ? __("search results") : __("all articles");
|
||||||
|
const fn = this.getName(feed, is_cat);
|
||||||
|
|
||||||
|
str = str.replace("%s", fn)
|
||||||
|
.replace("%w", mark_what);
|
||||||
|
|
||||||
|
if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const catchup_query = {
|
||||||
|
op: 'rpc', method: 'catchupFeed', feed_id: feed,
|
||||||
|
is_cat: is_cat, mode: mode, search_query: this.last_search_query[0],
|
||||||
|
search_lang: this.last_search_query[1]
|
||||||
|
};
|
||||||
|
|
||||||
|
notify_progress("Loading, please wait...", true);
|
||||||
|
|
||||||
|
xhrPost("backend.php", catchup_query, (transport) => {
|
||||||
|
Utils.handleRpcJson(transport);
|
||||||
|
|
||||||
|
const show_next_feed = getInitParam("on_catchup_show_next_feed") == "1";
|
||||||
|
|
||||||
|
if (show_next_feed) {
|
||||||
|
const nuf = this.getNextUnread(feed, is_cat);
|
||||||
|
|
||||||
|
if (nuf) {
|
||||||
|
this.open({feed: nuf, is_cat: is_cat});
|
||||||
|
}
|
||||||
|
} else if (feed == this.getActive() && is_cat == this.activeIsCat()) {
|
||||||
|
this.reloadCurrent();
|
||||||
|
}
|
||||||
|
|
||||||
|
notify("");
|
||||||
|
});
|
||||||
|
},
|
||||||
|
catchupCurrent: function(mode) {
|
||||||
|
this.catchupFeed(this.getActive(), this.activeIsCat(), mode);
|
||||||
|
},
|
||||||
|
catchupFeedInGroup: function(id) {
|
||||||
|
const title = this.getName(id);
|
||||||
|
|
||||||
|
const str = __("Mark all articles in %s as read?").replace("%s", title);
|
||||||
|
|
||||||
|
if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
|
||||||
|
|
||||||
|
const rows = $$("#headlines-frame > div[id*=RROW][data-orig-feed-id='" + id + "']");
|
||||||
|
|
||||||
|
if (rows.length > 0) {
|
||||||
|
|
||||||
|
rows.each(function (row) {
|
||||||
|
row.removeClassName("Unread");
|
||||||
|
|
||||||
|
if (row.getAttribute("data-article-id") != Article.getActive()) {
|
||||||
|
new Effect.Fade(row, {duration: 0.5});
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
const feedTitles = $$("#headlines-frame > div[class='feed-title']");
|
||||||
|
|
||||||
|
for (let i = 0; i < feedTitles.length; i++) {
|
||||||
|
if (feedTitles[i].getAttribute("data-feed-id") == id) {
|
||||||
|
|
||||||
|
if (i < feedTitles.length - 1) {
|
||||||
|
new Effect.Fade(feedTitles[i], {duration: 0.5});
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Headlines.updateFloatingTitle(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
notify_progress("Loading, please wait...", true);
|
||||||
|
|
||||||
|
xhrPost("backend.php", {op: "rpc", method: "catchupFeed", feed_id: id, is_cat: false}, (transport) => {
|
||||||
|
Utils.handleRpcJson(transport);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getUnread: function(feed, is_cat) {
|
||||||
|
try {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree && tree.model)
|
||||||
|
return tree.model.getFeedUnread(feed, is_cat);
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
},
|
||||||
|
getCategory: function(feed) {
|
||||||
|
try {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree && tree.model)
|
||||||
|
return tree.getFeedCategory(feed);
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
getName: function(feed, is_cat) {
|
||||||
|
if (isNaN(feed)) return feed; // it's a tag
|
||||||
|
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree && tree.model)
|
||||||
|
return tree.model.getFeedValue(feed, is_cat, 'name');
|
||||||
|
},
|
||||||
|
setUnread: function(feed, is_cat, unread) {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree && tree.model)
|
||||||
|
return tree.model.setFeedUnread(feed, is_cat, unread);
|
||||||
|
},
|
||||||
|
setValue: function(feed, is_cat, key, value) {
|
||||||
|
try {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree && tree.model)
|
||||||
|
return tree.model.setFeedValue(feed, is_cat, key, value);
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getValue: function(feed, is_cat, key) {
|
||||||
|
try {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree && tree.model)
|
||||||
|
return tree.model.getFeedValue(feed, is_cat, key);
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
},
|
||||||
|
setIcon: function(feed, is_cat, src) {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree) return tree.setFeedIcon(feed, is_cat, src);
|
||||||
|
},
|
||||||
|
setExpando: function(feed, is_cat, src) {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
|
||||||
|
if (tree) return tree.setFeedExpandoIcon(feed, is_cat, src);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
getNextUnread: function(feed, is_cat) {
|
||||||
|
const tree = dijit.byId("feedTree");
|
||||||
|
const nuf = tree.model.getNextUnreadFeed(feed, is_cat);
|
||||||
|
|
||||||
|
if (nuf)
|
||||||
|
return tree.model.store.getValue(nuf, 'bare_id');
|
||||||
|
},
|
||||||
|
search: function() {
|
||||||
|
const query = "backend.php?op=feeds&method=search¶m=" +
|
||||||
|
param_escape(Feeds.getActive() + ":" + Feeds.activeIsCat());
|
||||||
|
|
||||||
|
if (dijit.byId("searchDlg"))
|
||||||
|
dijit.byId("searchDlg").destroyRecursive();
|
||||||
|
|
||||||
|
const dialog = new dijit.Dialog({
|
||||||
|
id: "searchDlg",
|
||||||
|
title: __("Search"),
|
||||||
|
style: "width: 600px",
|
||||||
|
execute: function () {
|
||||||
|
if (this.validate()) {
|
||||||
|
Feeds._search_query = this.attr('value');
|
||||||
|
this.hide();
|
||||||
|
Feeds.reloadCurrent();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
href: query
|
||||||
|
});
|
||||||
|
|
||||||
|
dialog.show();
|
||||||
|
},
|
||||||
|
updateRandom: function() {
|
||||||
|
console.log("in update_random_feed");
|
||||||
|
|
||||||
|
xhrPost("backend.php", {op: "rpc", method: "updateRandom"}, (transport) => {
|
||||||
|
Utils.handleRpcJson(transport, true);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
File diff suppressed because it is too large
Load Diff
638
js/feedlist.js
638
js/feedlist.js
|
@ -1,638 +0,0 @@
|
||||||
/* global notify,__,dijit,fox */
|
|
||||||
|
|
||||||
const Feeds = {
|
|
||||||
counters_last_request: 0,
|
|
||||||
_active_feed_id: 0,
|
|
||||||
_active_feed_is_cat: false,
|
|
||||||
infscroll_in_progress: 0,
|
|
||||||
infscroll_disabled: 0,
|
|
||||||
_infscroll_timeout: false,
|
|
||||||
_search_query: false,
|
|
||||||
last_search_query: [],
|
|
||||||
_viewfeed_wait_timeout: false,
|
|
||||||
_counters_prev: [],
|
|
||||||
// NOTE: this implementation is incomplete
|
|
||||||
// for general objects but good enough for counters
|
|
||||||
// http://adripofjavascript.com/blog/drips/object-equality-in-javascript.html
|
|
||||||
counterEquals: function(a, b) {
|
|
||||||
// Create arrays of property names
|
|
||||||
const aProps = Object.getOwnPropertyNames(a);
|
|
||||||
const bProps = Object.getOwnPropertyNames(b);
|
|
||||||
|
|
||||||
// If number of properties is different,
|
|
||||||
// objects are not equivalent
|
|
||||||
if (aProps.length != bProps.length) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (let i = 0; i < aProps.length; i++) {
|
|
||||||
const propName = aProps[i];
|
|
||||||
|
|
||||||
// If values of same property are not equal,
|
|
||||||
// objects are not equivalent
|
|
||||||
if (a[propName] !== b[propName]) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we made it this far, objects
|
|
||||||
// are considered equivalent
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
resetCounters: function () {
|
|
||||||
this._counters_prev = [];
|
|
||||||
},
|
|
||||||
parseCounters: function (elems) {
|
|
||||||
for (let l = 0; l < elems.length; l++) {
|
|
||||||
|
|
||||||
if (Feeds._counters_prev[l] && this.counterEquals(elems[l], this._counters_prev[l])) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const id = elems[l].id;
|
|
||||||
const kind = elems[l].kind;
|
|
||||||
const ctr = parseInt(elems[l].counter);
|
|
||||||
const error = elems[l].error;
|
|
||||||
const has_img = elems[l].has_img;
|
|
||||||
const updated = elems[l].updated;
|
|
||||||
const auxctr = parseInt(elems[l].auxcounter);
|
|
||||||
|
|
||||||
if (id == "global-unread") {
|
|
||||||
App.global_unread = ctr;
|
|
||||||
App.updateTitle();
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (id == "subscribed-feeds") {
|
|
||||||
/* feeds_found = ctr; */
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*if (this.getUnread(id, (kind == "cat")) != ctr ||
|
|
||||||
(kind == "cat")) {
|
|
||||||
}*/
|
|
||||||
|
|
||||||
this.setUnread(id, (kind == "cat"), ctr);
|
|
||||||
this.setValue(id, (kind == "cat"), 'auxcounter', auxctr);
|
|
||||||
|
|
||||||
if (kind != "cat") {
|
|
||||||
this.setValue(id, false, 'error', error);
|
|
||||||
this.setValue(id, false, 'updated', updated);
|
|
||||||
|
|
||||||
if (id > 0) {
|
|
||||||
if (has_img) {
|
|
||||||
this.setIcon(id, false,
|
|
||||||
getInitParam("icons_url") + "/" + id + ".ico?" + has_img);
|
|
||||||
} else {
|
|
||||||
this.setIcon(id, false, 'images/blank_icon.gif');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
|
|
||||||
this._counters_prev = elems;
|
|
||||||
},
|
|
||||||
reloadCurrent: function(method) {
|
|
||||||
console.log("reloadCurrent: " + method);
|
|
||||||
|
|
||||||
if (this.getActive() != undefined) {
|
|
||||||
this.open({feed: this.getActive(), is_cat: this.activeIsCat(), method: method});
|
|
||||||
}
|
|
||||||
return false; // block unneeded form submits
|
|
||||||
},
|
|
||||||
openNextUnread: function() {
|
|
||||||
const is_cat = this.activeIsCat();
|
|
||||||
const nuf = this.getNextUnread(this.getActive(), is_cat);
|
|
||||||
if (nuf) this.open({feed: nuf, is_cat: is_cat});
|
|
||||||
},
|
|
||||||
toggle: function() {
|
|
||||||
Element.toggle("feeds-holder");
|
|
||||||
|
|
||||||
const splitter = $("feeds-holder_splitter");
|
|
||||||
|
|
||||||
Element.visible("feeds-holder") ? splitter.show() : splitter.hide();
|
|
||||||
|
|
||||||
dijit.byId("main").resize();
|
|
||||||
},
|
|
||||||
cancelSearch: function() {
|
|
||||||
this._search_query = "";
|
|
||||||
this.reloadCurrent();
|
|
||||||
},
|
|
||||||
requestCounters: function(force) {
|
|
||||||
const date = new Date();
|
|
||||||
const timestamp = Math.round(date.getTime() / 1000);
|
|
||||||
|
|
||||||
if (force || timestamp - this.counters_last_request > 5) {
|
|
||||||
console.log("scheduling request of counters...");
|
|
||||||
|
|
||||||
this.counters_last_request = timestamp;
|
|
||||||
|
|
||||||
let query = {op: "rpc", method: "getAllCounters", seq: Utils.next_seq()};
|
|
||||||
|
|
||||||
if (!force)
|
|
||||||
query.last_article_id = getInitParam("last_article_id");
|
|
||||||
|
|
||||||
xhrPost("backend.php", query, (transport) => {
|
|
||||||
Utils.handleRpcJson(transport);
|
|
||||||
});
|
|
||||||
|
|
||||||
} else {
|
|
||||||
console.log("request_counters: rate limit reached: " + (timestamp - this.counters_last_request));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
reload: function() {
|
|
||||||
try {
|
|
||||||
Element.show("feedlistLoading");
|
|
||||||
|
|
||||||
this.resetCounters();
|
|
||||||
|
|
||||||
if (dijit.byId("feedTree")) {
|
|
||||||
dijit.byId("feedTree").destroyRecursive();
|
|
||||||
}
|
|
||||||
|
|
||||||
const store = new dojo.data.ItemFileWriteStore({
|
|
||||||
url: "backend.php?op=pref_feeds&method=getfeedtree&mode=2"
|
|
||||||
});
|
|
||||||
|
|
||||||
// noinspection JSUnresolvedFunction
|
|
||||||
const treeModel = new fox.FeedStoreModel({
|
|
||||||
store: store,
|
|
||||||
query: {
|
|
||||||
"type": getInitParam('enable_feed_cats') == 1 ? "category" : "feed"
|
|
||||||
},
|
|
||||||
rootId: "root",
|
|
||||||
rootLabel: "Feeds",
|
|
||||||
childrenAttrs: ["items"]
|
|
||||||
});
|
|
||||||
|
|
||||||
// noinspection JSUnresolvedFunction
|
|
||||||
const tree = new fox.FeedTree({
|
|
||||||
model: treeModel,
|
|
||||||
onClick: function (item/*, node*/) {
|
|
||||||
const id = String(item.id);
|
|
||||||
const is_cat = id.match("^CAT:");
|
|
||||||
const feed = id.substr(id.indexOf(":") + 1);
|
|
||||||
Feeds.open({feed: feed, is_cat: is_cat});
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
openOnClick: false,
|
|
||||||
showRoot: false,
|
|
||||||
persist: true,
|
|
||||||
id: "feedTree",
|
|
||||||
}, "feedTree");
|
|
||||||
|
|
||||||
const tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
|
|
||||||
console.log(dijit.getEnclosingWidget(event.target));
|
|
||||||
dojo.disconnect(tmph);
|
|
||||||
});
|
|
||||||
|
|
||||||
$("feeds-holder").appendChild(tree.domNode);
|
|
||||||
|
|
||||||
const tmph2 = dojo.connect(tree, 'onLoad', function () {
|
|
||||||
dojo.disconnect(tmph2);
|
|
||||||
Element.hide("feedlistLoading");
|
|
||||||
|
|
||||||
try {
|
|
||||||
Feeds.init();
|
|
||||||
Utils.setLoadingProgress(25);
|
|
||||||
} catch (e) {
|
|
||||||
exception_error(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
tree.startup();
|
|
||||||
} catch (e) {
|
|
||||||
exception_error(e);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
init: function() {
|
|
||||||
console.log("in feedlist init");
|
|
||||||
|
|
||||||
Utils.setLoadingProgress(50);
|
|
||||||
|
|
||||||
document.onkeydown = () => { App.hotkeyHandler(event) };
|
|
||||||
window.setInterval(() => { Headlines.catchupBatched() }, 10 * 1000);
|
|
||||||
|
|
||||||
if (!this.getActive()) {
|
|
||||||
this.open({feed: -3});
|
|
||||||
} else {
|
|
||||||
this.open({feed: this.getActive(), is_cat: this.activeIsCat()});
|
|
||||||
}
|
|
||||||
|
|
||||||
this.hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
|
|
||||||
|
|
||||||
if (getInitParam("is_default_pw")) {
|
|
||||||
console.warn("user password is at default value");
|
|
||||||
|
|
||||||
const dialog = new dijit.Dialog({
|
|
||||||
title: __("Your password is at default value"),
|
|
||||||
href: "backend.php?op=dlg&method=defaultpasswordwarning",
|
|
||||||
id: 'infoBox',
|
|
||||||
style: "width: 600px",
|
|
||||||
onCancel: function () {
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
onExecute: function () {
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
onClose: function () {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
dialog.show();
|
|
||||||
}
|
|
||||||
|
|
||||||
// bw_limit disables timeout() so we request initial counters separately
|
|
||||||
if (getInitParam("bw_limit") == "1") {
|
|
||||||
this.requestCounters(true);
|
|
||||||
} else {
|
|
||||||
setTimeout(() => {
|
|
||||||
this.requestCounters(true);
|
|
||||||
setInterval(() => { this.requestCounters(); }, 60 * 1000)
|
|
||||||
}, 250);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
activeIsCat: function() {
|
|
||||||
return !!this._active_feed_is_cat;
|
|
||||||
},
|
|
||||||
getActive: function() {
|
|
||||||
return this._active_feed_id;
|
|
||||||
},
|
|
||||||
setActive: function(id, is_cat) {
|
|
||||||
hash_set('f', id);
|
|
||||||
hash_set('c', is_cat ? 1 : 0);
|
|
||||||
|
|
||||||
this._active_feed_id = id;
|
|
||||||
this._active_feed_is_cat = is_cat;
|
|
||||||
|
|
||||||
$("headlines-frame").setAttribute("feed-id", id);
|
|
||||||
$("headlines-frame").setAttribute("is-cat", is_cat ? 1 : 0);
|
|
||||||
|
|
||||||
this.select(id, is_cat);
|
|
||||||
|
|
||||||
PluginHost.run(PluginHost.HOOK_FEED_SET_ACTIVE, [this._active_feed_id, this._active_feed_is_cat]);
|
|
||||||
},
|
|
||||||
select: function(feed, is_cat) {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree) return tree.selectFeed(feed, is_cat);
|
|
||||||
},
|
|
||||||
toggleUnread: function() {
|
|
||||||
const hide = !(getInitParam("hide_read_feeds") == "1");
|
|
||||||
|
|
||||||
xhrPost("backend.php", {op: "rpc", method: "setpref", key: "HIDE_READ_FEEDS", value: hide}, () => {
|
|
||||||
this.hideOrShowFeeds(hide);
|
|
||||||
setInitParam("hide_read_feeds", hide);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
hideOrShowFeeds: function(hide) {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree)
|
|
||||||
return tree.hideRead(hide, getInitParam("hide_read_shows_special"));
|
|
||||||
},
|
|
||||||
open: function(params) {
|
|
||||||
const feed = params.feed;
|
|
||||||
const is_cat = !!params.is_cat || false;
|
|
||||||
const offset = params.offset || 0;
|
|
||||||
const viewfeed_debug = params.viewfeed_debug;
|
|
||||||
const method = params.method;
|
|
||||||
// this is used to quickly switch between feeds, sets active but xhr is on a timeout
|
|
||||||
const delayed = params.delayed || false;
|
|
||||||
|
|
||||||
if (feed != this.getActive() || this.activeIsCat() != is_cat) {
|
|
||||||
this._search_query = false;
|
|
||||||
Article.setActive(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset != 0) {
|
|
||||||
if (this.infscroll_in_progress)
|
|
||||||
return;
|
|
||||||
|
|
||||||
this.infscroll_in_progress = 1;
|
|
||||||
|
|
||||||
window.clearTimeout(this._infscroll_timeout);
|
|
||||||
this._infscroll_timeout = window.setTimeout(() => {
|
|
||||||
console.log('infscroll request timed out, aborting');
|
|
||||||
this.infscroll_in_progress = 0;
|
|
||||||
|
|
||||||
// call scroll handler to maybe repeat infscroll request
|
|
||||||
Headlines.scrollHandler();
|
|
||||||
}, 10 * 1000);
|
|
||||||
}
|
|
||||||
|
|
||||||
Form.enable("main_toolbar_form");
|
|
||||||
|
|
||||||
let query = Object.assign({op: "feeds", method: "view", feed: feed},
|
|
||||||
dojo.formToObject("main_toolbar_form"));
|
|
||||||
|
|
||||||
if (method) query.m = method;
|
|
||||||
|
|
||||||
if (offset > 0) {
|
|
||||||
if (Headlines.current_first_id) {
|
|
||||||
query.fid = Headlines.current_first_id;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this._search_query) {
|
|
||||||
query = Object.assign(query, this._search_query);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset != 0) {
|
|
||||||
query.skip = offset;
|
|
||||||
|
|
||||||
// to prevent duplicate feed titles when showing grouped vfeeds
|
|
||||||
if (Headlines.vgroup_last_feed != undefined) {
|
|
||||||
query.vgrlf = Headlines.vgroup_last_feed;
|
|
||||||
}
|
|
||||||
} else if (!is_cat && feed == this.getActive() && !params.method) {
|
|
||||||
query.m = "ForceUpdate";
|
|
||||||
}
|
|
||||||
|
|
||||||
Form.enable("main_toolbar_form");
|
|
||||||
|
|
||||||
if (!delayed)
|
|
||||||
if (!this.setExpando(feed, is_cat,
|
|
||||||
(is_cat) ? 'images/indicator_tiny.gif' : 'images/indicator_white.gif'))
|
|
||||||
notify_progress("Loading, please wait...", true);
|
|
||||||
|
|
||||||
query.cat = is_cat;
|
|
||||||
|
|
||||||
this.setActive(feed, is_cat);
|
|
||||||
|
|
||||||
if (viewfeed_debug) {
|
|
||||||
window.open("backend.php?" +
|
|
||||||
dojo.objectToQuery(
|
|
||||||
Object.assign({debug: 1, csrf_token: getInitParam("csrf_token")}, query)
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
window.clearTimeout(this._viewfeed_wait_timeout);
|
|
||||||
this._viewfeed_wait_timeout = window.setTimeout(() => {
|
|
||||||
Headlines.catchupBatched(() => {
|
|
||||||
xhrPost("backend.php", query, (transport) => {
|
|
||||||
try {
|
|
||||||
window.clearTimeout(this._infscroll_timeout);
|
|
||||||
this.setExpando(feed, is_cat, 'images/blank_icon.gif');
|
|
||||||
Headlines.onLoaded(transport, offset);
|
|
||||||
PluginHost.run(PluginHost.HOOK_FEED_LOADED, [feed, is_cat]);
|
|
||||||
} catch (e) {
|
|
||||||
exception_error(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}, delayed ? 250 : 0);
|
|
||||||
},
|
|
||||||
catchupAll: function() {
|
|
||||||
const str = __("Mark all articles as read?");
|
|
||||||
|
|
||||||
if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
|
|
||||||
|
|
||||||
notify_progress("Marking all feeds as read...");
|
|
||||||
|
|
||||||
xhrPost("backend.php", {op: "feeds", method: "catchupAll"}, () => {
|
|
||||||
this.requestCounters(true);
|
|
||||||
this.reloadCurrent();
|
|
||||||
});
|
|
||||||
|
|
||||||
App.global_unread = 0;
|
|
||||||
App.updateTitle();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
decrementFeedCounter: function(feed, is_cat) {
|
|
||||||
let ctr = this.getUnread(feed, is_cat);
|
|
||||||
|
|
||||||
if (ctr > 0) {
|
|
||||||
this.setUnread(feed, is_cat, ctr - 1);
|
|
||||||
App.global_unread -= 1;
|
|
||||||
App.updateTitle();
|
|
||||||
|
|
||||||
if (!is_cat) {
|
|
||||||
const cat = parseInt(this.getCategory(feed));
|
|
||||||
|
|
||||||
if (!isNaN(cat)) {
|
|
||||||
ctr = this.getUnread(cat, true);
|
|
||||||
|
|
||||||
if (ctr > 0) {
|
|
||||||
this.setUnread(cat, true, ctr - 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
catchupFeed: function(feed, is_cat, mode) {
|
|
||||||
if (is_cat == undefined) is_cat = false;
|
|
||||||
|
|
||||||
let str = false;
|
|
||||||
|
|
||||||
switch (mode) {
|
|
||||||
case "1day":
|
|
||||||
str = __("Mark %w in %s older than 1 day as read?");
|
|
||||||
break;
|
|
||||||
case "1week":
|
|
||||||
str = __("Mark %w in %s older than 1 week as read?");
|
|
||||||
break;
|
|
||||||
case "2week":
|
|
||||||
str = __("Mark %w in %s older than 2 weeks as read?");
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
str = __("Mark %w in %s as read?");
|
|
||||||
}
|
|
||||||
|
|
||||||
const mark_what = this.last_search_query && this.last_search_query[0] ? __("search results") : __("all articles");
|
|
||||||
const fn = this.getName(feed, is_cat);
|
|
||||||
|
|
||||||
str = str.replace("%s", fn)
|
|
||||||
.replace("%w", mark_what);
|
|
||||||
|
|
||||||
if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const catchup_query = {
|
|
||||||
op: 'rpc', method: 'catchupFeed', feed_id: feed,
|
|
||||||
is_cat: is_cat, mode: mode, search_query: this.last_search_query[0],
|
|
||||||
search_lang: this.last_search_query[1]
|
|
||||||
};
|
|
||||||
|
|
||||||
notify_progress("Loading, please wait...", true);
|
|
||||||
|
|
||||||
xhrPost("backend.php", catchup_query, (transport) => {
|
|
||||||
Utils.handleRpcJson(transport);
|
|
||||||
|
|
||||||
const show_next_feed = getInitParam("on_catchup_show_next_feed") == "1";
|
|
||||||
|
|
||||||
if (show_next_feed) {
|
|
||||||
const nuf = this.getNextUnread(feed, is_cat);
|
|
||||||
|
|
||||||
if (nuf) {
|
|
||||||
this.open({feed: nuf, is_cat: is_cat});
|
|
||||||
}
|
|
||||||
} else if (feed == this.getActive() && is_cat == this.activeIsCat()) {
|
|
||||||
this.reloadCurrent();
|
|
||||||
}
|
|
||||||
|
|
||||||
notify("");
|
|
||||||
});
|
|
||||||
},
|
|
||||||
catchupCurrent: function(mode) {
|
|
||||||
this.catchupFeed(this.getActive(), this.activeIsCat(), mode);
|
|
||||||
},
|
|
||||||
catchupFeedInGroup: function(id) {
|
|
||||||
const title = this.getName(id);
|
|
||||||
|
|
||||||
const str = __("Mark all articles in %s as read?").replace("%s", title);
|
|
||||||
|
|
||||||
if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
|
|
||||||
|
|
||||||
const rows = $$("#headlines-frame > div[id*=RROW][data-orig-feed-id='" + id + "']");
|
|
||||||
|
|
||||||
if (rows.length > 0) {
|
|
||||||
|
|
||||||
rows.each(function (row) {
|
|
||||||
row.removeClassName("Unread");
|
|
||||||
|
|
||||||
if (row.getAttribute("data-article-id") != Article.getActive()) {
|
|
||||||
new Effect.Fade(row, {duration: 0.5});
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
const feedTitles = $$("#headlines-frame > div[class='feed-title']");
|
|
||||||
|
|
||||||
for (let i = 0; i < feedTitles.length; i++) {
|
|
||||||
if (feedTitles[i].getAttribute("data-feed-id") == id) {
|
|
||||||
|
|
||||||
if (i < feedTitles.length - 1) {
|
|
||||||
new Effect.Fade(feedTitles[i], {duration: 0.5});
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Headlines.updateFloatingTitle(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
notify_progress("Loading, please wait...", true);
|
|
||||||
|
|
||||||
xhrPost("backend.php", {op: "rpc", method: "catchupFeed", feed_id: id, is_cat: false}, (transport) => {
|
|
||||||
Utils.handleRpcJson(transport);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
getUnread: function(feed, is_cat) {
|
|
||||||
try {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree && tree.model)
|
|
||||||
return tree.model.getFeedUnread(feed, is_cat);
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
//
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1;
|
|
||||||
},
|
|
||||||
getCategory: function(feed) {
|
|
||||||
try {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree && tree.model)
|
|
||||||
return tree.getFeedCategory(feed);
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
//
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
getName: function(feed, is_cat) {
|
|
||||||
if (isNaN(feed)) return feed; // it's a tag
|
|
||||||
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree && tree.model)
|
|
||||||
return tree.model.getFeedValue(feed, is_cat, 'name');
|
|
||||||
},
|
|
||||||
setUnread: function(feed, is_cat, unread) {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree && tree.model)
|
|
||||||
return tree.model.setFeedUnread(feed, is_cat, unread);
|
|
||||||
},
|
|
||||||
setValue: function(feed, is_cat, key, value) {
|
|
||||||
try {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree && tree.model)
|
|
||||||
return tree.model.setFeedValue(feed, is_cat, key, value);
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
//
|
|
||||||
}
|
|
||||||
},
|
|
||||||
getValue: function(feed, is_cat, key) {
|
|
||||||
try {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree && tree.model)
|
|
||||||
return tree.model.getFeedValue(feed, is_cat, key);
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
//
|
|
||||||
}
|
|
||||||
return '';
|
|
||||||
},
|
|
||||||
setIcon: function(feed, is_cat, src) {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree) return tree.setFeedIcon(feed, is_cat, src);
|
|
||||||
},
|
|
||||||
setExpando: function(feed, is_cat, src) {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
|
|
||||||
if (tree) return tree.setFeedExpandoIcon(feed, is_cat, src);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
getNextUnread: function(feed, is_cat) {
|
|
||||||
const tree = dijit.byId("feedTree");
|
|
||||||
const nuf = tree.model.getNextUnreadFeed(feed, is_cat);
|
|
||||||
|
|
||||||
if (nuf)
|
|
||||||
return tree.model.store.getValue(nuf, 'bare_id');
|
|
||||||
},
|
|
||||||
search: function() {
|
|
||||||
const query = "backend.php?op=feeds&method=search¶m=" +
|
|
||||||
param_escape(Feeds.getActive() + ":" + Feeds.activeIsCat());
|
|
||||||
|
|
||||||
if (dijit.byId("searchDlg"))
|
|
||||||
dijit.byId("searchDlg").destroyRecursive();
|
|
||||||
|
|
||||||
const dialog = new dijit.Dialog({
|
|
||||||
id: "searchDlg",
|
|
||||||
title: __("Search"),
|
|
||||||
style: "width: 600px",
|
|
||||||
execute: function () {
|
|
||||||
if (this.validate()) {
|
|
||||||
Feeds._search_query = this.attr('value');
|
|
||||||
this.hide();
|
|
||||||
Feeds.reloadCurrent();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
href: query
|
|
||||||
});
|
|
||||||
|
|
||||||
dialog.show();
|
|
||||||
},
|
|
||||||
updateRandom: function() {
|
|
||||||
console.log("in update_random_feed");
|
|
||||||
|
|
||||||
xhrPost("backend.php", {op: "rpc", method: "updateRandom"}, (transport) => {
|
|
||||||
Utils.handleRpcJson(transport, true);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
};
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* global dijit, __,fox */
|
/* global dijit, __ */
|
||||||
|
|
||||||
let Utils;
|
let Utils;
|
||||||
let CommonDialogs;
|
let CommonDialogs;
|
||||||
|
|
14
js/tt-rss.js
14
js/tt-rss.js
|
@ -1,8 +1,12 @@
|
||||||
/* global dijit,__,fox */
|
/* global dijit,__ */
|
||||||
|
|
||||||
let Utils;
|
let Utils;
|
||||||
let CommonDialogs;
|
let CommonDialogs;
|
||||||
let Filters;
|
let Filters;
|
||||||
|
let Feeds;
|
||||||
|
let Headlines;
|
||||||
|
let Article;
|
||||||
|
let ArticleCache;
|
||||||
|
|
||||||
const App = {
|
const App = {
|
||||||
global_unread: -1,
|
global_unread: -1,
|
||||||
|
@ -51,6 +55,10 @@ const App = {
|
||||||
"fox/Utils",
|
"fox/Utils",
|
||||||
"fox/CommonDialogs",
|
"fox/CommonDialogs",
|
||||||
"fox/CommonFilters",
|
"fox/CommonFilters",
|
||||||
|
"fox/Feeds",
|
||||||
|
"fox/Headlines",
|
||||||
|
"fox/Article",
|
||||||
|
"fox/ArticleCache",
|
||||||
"fox/FeedStoreModel",
|
"fox/FeedStoreModel",
|
||||||
"fox/FeedTree"], function (dojo, ready, parser) {
|
"fox/FeedTree"], function (dojo, ready, parser) {
|
||||||
|
|
||||||
|
@ -60,6 +68,10 @@ const App = {
|
||||||
Utils = fox.Utils();
|
Utils = fox.Utils();
|
||||||
CommonDialogs = fox.CommonDialogs();
|
CommonDialogs = fox.CommonDialogs();
|
||||||
Filters = fox.CommonFilters();
|
Filters = fox.CommonFilters();
|
||||||
|
Feeds = fox.Feeds();
|
||||||
|
Headlines = fox.Headlines();
|
||||||
|
Article = fox.Article();
|
||||||
|
ArticleCache = fox.ArticleCache();
|
||||||
|
|
||||||
parser.parse();
|
parser.parse();
|
||||||
|
|
||||||
|
|
1569
js/viewfeed.js
1569
js/viewfeed.js
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue