]>
git.wh0rd.org - tt-rss.git/blob - js/tt-rss.js
5f8604f94aa354f57b9b8d125a6ce7ac0a564b83
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() {
25 return _active_feed_id
;
27 exception_error("getActiveFeedId", e
);
31 function setActiveFeedId(id
, is_cat
) {
34 hash_set('c', is_cat
? 1 : 0);
37 _active_feed_is_cat
= is_cat
;
39 $("headlines-frame").setAttribute("feed-id", id
);
40 $("headlines-frame").setAttribute("is-cat", is_cat
? 1 : 0);
42 selectFeed(id
, is_cat
);
44 exception_error("setActiveFeedId", e
);
49 function updateFeedList() {
52 // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
53 // __("Loading, please wait...") + "</div>";
55 Element
.show("feedlistLoading");
57 if (dijit
.byId("feedTree")) {
58 dijit
.byId("feedTree").destroyRecursive();
61 var store
= new dojo
.data
.ItemFileWriteStore({
62 url
: "backend.php?op=pref_feeds&method=getfeedtree&mode=2"});
64 var treeModel
= new fox
.FeedStoreModel({
67 "type": getInitParam('enable_feed_cats') == 1 ? "category" : "feed"
71 childrenAttrs
: ["items"]
74 var tree
= new fox
.FeedTree({
76 onClick: function (item
, node
) {
77 var id
= String(item
.id
);
78 var is_cat
= id
.match("^CAT:");
79 var feed
= id
.substr(id
.indexOf(":")+1);
80 viewfeed(feed
, '', is_cat
);
88 /* var menu = new dijit.Menu({id: 'feedMenu'});
90 menu.addChild(new dijit.MenuItem({
91 label: "Simple menu item"
94 // menu.bindDomNode(tree.domNode); */
96 var tmph
= dojo
.connect(dijit
.byId('feedMenu'), '_openMyself', function (event
) {
97 console
.log(dijit
.getEnclosingWidget(event
.target
));
98 dojo
.disconnect(tmph
);
101 $("feeds-holder").appendChild(tree
.domNode
);
103 var tmph
= dojo
.connect(tree
, 'onLoad', function() {
104 dojo
.disconnect(tmph
);
105 Element
.hide("feedlistLoading");
109 // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
110 // menu.bindDomNode(node);
112 loading_set_progress(25);
118 exception_error("updateFeedList", e
);
122 function catchupAllFeeds() {
124 var str
= __("Mark all articles as read?");
126 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
128 var query_str
= "backend.php?op=feeds&method=catchupAll";
130 notify_progress("Marking all feeds as read...");
132 //console.log("catchupAllFeeds Q=" + query_str);
134 new Ajax
.Request("backend.php", {
135 parameters
: query_str
,
136 onComplete: function(transport
) {
137 feedlist_callback2(transport
);
145 function viewCurrentFeed(method
) {
146 console
.log("viewCurrentFeed");
148 if (getActiveFeedId() != undefined) {
149 viewfeed(getActiveFeedId(), method
, activeFeedIsCat());
151 return false; // block unneeded form submits
155 if (getInitParam("bw_limit") != "1") {
157 setTimeout("timeout()", 60*1000);
162 var query
= "backend.php?op=dlg&method=search¶m=" +
163 param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
165 if (dijit
.byId("searchDlg"))
166 dijit
.byId("searchDlg").destroyRecursive();
168 dialog
= new dijit
.Dialog({
171 style
: "width: 600px",
172 execute: function() {
173 if (this.validate()) {
174 _search_query
= dojo
.objectToQuery(this.attr('value'));
184 function updateTitle() {
185 var tmp
= "Tiny Tiny RSS";
187 if (global_unread
> 0) {
188 tmp
= "(" + global_unread
+ ") " + tmp
;
192 if (global_unread
> 0) {
193 window
.fluid
.dockBadge
= global_unread
;
195 window
.fluid
.dockBadge
= "";
199 document
.title
= tmp
;
202 function genericSanityCheck() {
203 setCookie("ttrss_test", "TEST");
205 if (getCookie("ttrss_test") != "TEST") {
206 return fatalError(2);
215 //dojo.registerModulePath("fox", "../../js/");
217 dojo
.require("fox.FeedTree");
219 dojo
.require("dijit.ColorPalette");
220 dojo
.require("dijit.Dialog");
221 dojo
.require("dijit.form.Button");
222 dojo
.require("dijit.form.CheckBox");
223 dojo
.require("dijit.form.DropDownButton");
224 dojo
.require("dijit.form.FilteringSelect");
225 dojo
.require("dijit.form.Form");
226 dojo
.require("dijit.form.RadioButton");
227 dojo
.require("dijit.form.Select");
228 dojo
.require("dijit.form.SimpleTextarea");
229 dojo
.require("dijit.form.TextBox");
230 dojo
.require("dijit.form.ValidationTextBox");
231 dojo
.require("dijit.InlineEditBox");
232 dojo
.require("dijit.layout.AccordionContainer");
233 dojo
.require("dijit.layout.BorderContainer");
234 dojo
.require("dijit.layout.ContentPane");
235 dojo
.require("dijit.layout.TabContainer");
236 dojo
.require("dijit.Menu");
237 dojo
.require("dijit.ProgressBar");
238 dojo
.require("dijit.ProgressBar");
239 dojo
.require("dijit.Toolbar");
240 dojo
.require("dijit.Tree");
241 dojo
.require("dijit.tree.dndSource");
242 dojo
.require("dojo.data.ItemFileWriteStore");
246 if (!genericSanityCheck())
249 loading_set_progress(20);
251 var hasAudio
= !!((myAudioTag
= document
.createElement('audio')).canPlayType
);
252 var hasSandbox
= "sandbox" in document
.createElement("iframe");
254 new Ajax
.Request("backend.php", {
255 parameters
: {op
: "rpc", method
: "sanityCheck", hasAudio
: hasAudio
,
256 hasSandbox
: hasSandbox
},
257 onComplete: function(transport
) {
258 backend_sanity_check_callback(transport
);
261 hotkey_actions
["next_feed"] = function() {
262 var rv
= dijit
.byId("feedTree").getNextFeed(
263 getActiveFeedId(), activeFeedIsCat());
265 if (rv
) viewfeed(rv
[0], '', rv
[1]);
267 hotkey_actions
["prev_feed"] = function() {
268 var rv
= dijit
.byId("feedTree").getPreviousFeed(
269 getActiveFeedId(), activeFeedIsCat());
271 if (rv
) viewfeed(rv
[0], '', rv
[1]);
273 hotkey_actions
["next_article"] = function() {
276 hotkey_actions
["prev_article"] = function() {
279 hotkey_actions
["next_article_noscroll"] = function() {
280 moveToPost('next', true);
282 hotkey_actions
["prev_article_noscroll"] = function() {
283 moveToPost('prev', true);
285 hotkey_actions
["collapse_article"] = function() {
286 var id
= getActiveArticleId();
287 var elem
= $("CICD-"+id
);
289 cdmCollapseArticle(null, id
);
292 cdmExpandArticle(id
);
295 hotkey_actions
["search_dialog"] = function() {
298 hotkey_actions
["toggle_mark"] = function() {
299 selectionToggleMarked(undefined, false, true);
301 hotkey_actions
["toggle_publ"] = function() {
302 selectionTogglePublished(undefined, false, true);
304 hotkey_actions
["toggle_unread"] = function() {
305 selectionToggleUnread(undefined, false, true);
307 hotkey_actions
["edit_tags"] = function() {
308 var id
= getActiveArticleId();
310 editArticleTags(id
, getActiveFeedId(), isCdmMode());
313 hotkey_actions
["dismiss_selected"] = function() {
314 dismissSelectedArticles();
316 hotkey_actions
["open_in_new_window"] = function() {
317 if (getActiveArticleId()) {
318 openArticleInNewWindow(getActiveArticleId());
322 hotkey_actions
["catchup_below"] = function() {
323 catchupRelativeToArticle(1);
325 hotkey_actions
["catchup_above"] = function() {
326 catchupRelativeToArticle(0);
328 hotkey_actions
["article_scroll_down"] = function() {
329 var ctr
= $("content_insert") ? $("content_insert") : $("headlines-frame");
333 hotkey_actions
["article_scroll_up"] = function() {
334 var ctr
= $("content_insert") ? $("content_insert") : $("headlines-frame");
338 hotkey_actions
["close_article"] = function() {
340 if (!getInitParam("cdm_expanded")) {
341 cdmCollapseArticle(false, getActiveArticleId());
343 dismissArticle(getActiveArticleId());
349 hotkey_actions
["email_article"] = function() {
350 if (typeof emailArticle
!= "undefined") {
352 } else if (typeof mailtoArticle
!= "undefined") {
355 alert(__("Please enable mail plugin first."));
358 hotkey_actions
["select_all"] = function() {
359 selectArticles('all');
361 hotkey_actions
["select_unread"] = function() {
362 selectArticles('unread');
364 hotkey_actions
["select_marked"] = function() {
365 selectArticles('marked');
367 hotkey_actions
["select_published"] = function() {
368 selectArticles('published');
370 hotkey_actions
["select_invert"] = function() {
371 selectArticles('invert');
373 hotkey_actions
["select_none"] = function() {
374 selectArticles('none');
376 hotkey_actions
["feed_refresh"] = function() {
377 if (getActiveFeedId() != undefined) {
378 viewfeed(getActiveFeedId(), '', activeFeedIsCat());
382 hotkey_actions
["feed_unhide_read"] = function() {
385 hotkey_actions
["feed_subscribe"] = function() {
388 hotkey_actions
["feed_debug_update"] = function() {
389 window
.open("backend.php?op=feeds&method=view&feed=" + getActiveFeedId() +
390 "&view_mode=adaptive&order_by=default&update=&m=ForceUpdate&cat=" +
391 activeFeedIsCat() + "&DevForceUpdate=1&debug=1&xdebug=1&csrf_token=" +
392 getInitParam("csrf_token"));
394 hotkey_actions
["feed_edit"] = function() {
395 if (activeFeedIsCat())
396 alert(__("You can't edit this kind of feed."));
398 editFeed(getActiveFeedId());
400 hotkey_actions
["feed_catchup"] = function() {
401 if (getActiveFeedId() != undefined) {
402 catchupCurrentFeed();
406 hotkey_actions
["feed_reverse"] = function() {
407 reverseHeadlineOrder();
409 hotkey_actions
["catchup_all"] = function() {
412 hotkey_actions
["cat_toggle_collapse"] = function() {
413 if (activeFeedIsCat()) {
414 dijit
.byId("feedTree").collapseCat(getActiveFeedId());
418 hotkey_actions
["goto_all"] = function() {
421 hotkey_actions
["goto_fresh"] = function() {
424 hotkey_actions
["goto_marked"] = function() {
427 hotkey_actions
["goto_published"] = function() {
430 hotkey_actions
["goto_tagcloud"] = function() {
431 displayDlg("printTagCloud");
433 hotkey_actions
["goto_prefs"] = function() {
436 hotkey_actions
["select_article_cursor"] = function() {
437 var id
= getArticleUnderPointer();
439 var cb
= dijit
.byId("RCHK-" + id
);
441 cb
.attr("checked", !cb
.attr("checked"));
442 toggleSelectRowById(cb
, "RROW-" + id
);
447 hotkey_actions
["create_label"] = function() {
450 hotkey_actions
["create_filter"] = function() {
453 hotkey_actions
["collapse_sidebar"] = function() {
456 hotkey_actions
["toggle_embed_original"] = function() {
457 if (typeof embedOriginalArticle
!= "undefined") {
458 if (getActiveArticleId())
459 embedOriginalArticle(getActiveArticleId());
461 alert(__("Please enable embed_original plugin first."));
464 hotkey_actions
["toggle_widescreen"] = function() {
466 _widescreen_mode
= !_widescreen_mode
;
468 switchPanelMode(_widescreen_mode
);
471 hotkey_actions
["help_dialog"] = function() {
474 hotkey_actions
["toggle_combined_mode"] = function() {
475 notify_progress("Loading, please wait...");
477 var value
= isCdmMode() ? "false" : "true";
478 var query
= "?op=rpc&method=setpref&key=COMBINED_DISPLAY_MODE&value=" + value
;
480 new Ajax
.Request("backend.php", {
482 onComplete: function(transport
) {
483 setInitParam("combined_display_mode",
484 !getInitParam("combined_display_mode"));
491 hotkey_actions
["toggle_cdm_expanded"] = function() {
492 notify_progress("Loading, please wait...");
494 var value
= getInitParam("cdm_expanded") ? "false" : "true";
495 var query
= "?op=rpc&method=setpref&key=CDM_EXPANDED&value=" + value
;
497 new Ajax
.Request("backend.php", {
499 onComplete: function(transport
) {
500 setInitParam("cdm_expanded", !getInitParam("cdm_expanded"));
507 exception_error("init", e
);
511 function init_second_stage() {
514 dojo
.addOnLoad(function() {
518 _widescreen_mode
= getInitParam("widescreen");
520 if (_widescreen_mode
) {
521 switchPanelMode(_widescreen_mode
);
526 delCookie("ttrss_test");
528 var toolbar
= document
.forms
["main_toolbar_form"];
530 dijit
.getEnclosingWidget(toolbar
.view_mode
).attr('value',
531 getInitParam("default_view_mode"));
533 dijit
.getEnclosingWidget(toolbar
.order_by
).attr('value',
534 getInitParam("default_view_order_by"));
536 feeds_sort_by_unread
= getInitParam("feeds_sort_by_unread") == 1;
538 var hash_feed_id
= hash_get('f');
539 var hash_feed_is_cat
= hash_get('c') == "1";
541 if (hash_feed_id
!= undefined) {
542 setActiveFeedId(hash_feed_id
, hash_feed_is_cat
);
545 loading_set_progress(30);
547 // can't use cache_clear() here because viewfeed might not have initialized yet
548 if ('sessionStorage' in window
&& window
['sessionStorage'] !== null)
549 sessionStorage
.clear();
551 var hotkeys
= getInitParam("hotkeys");
554 for (sequence
in hotkeys
[1]) {
555 filtered
= sequence
.replace(/\|.*$/, "");
556 tmp
[filtered
] = hotkeys
[1][sequence
];
560 setInitParam("hotkeys", hotkeys
);
562 console
.log("second stage ok");
564 if (getInitParam("simple_update")) {
565 console
.log("scheduling simple feed updater...");
566 window
.setTimeout("update_random_feed()", 30*1000);
570 exception_error("init_second_stage", e
);
574 function quickMenuGo(opid
) {
584 displayDlg("printTagCloud");
587 displayDlg("printTagSelect");
596 window
.location
.href
= "backend.php?op=digest";
599 if (activeFeedIsCat())
600 alert(__("You can't edit this kind of feed."));
602 editFeed(getActiveFeedId());
604 case "qmcRemoveFeed":
605 var actid
= getActiveFeedId();
607 if (activeFeedIsCat()) {
608 alert(__("You can't unsubscribe from the category."));
613 alert(__("Please select some feed first."));
617 var fn
= getFeedName(actid
);
619 var pr
= __("Unsubscribe from %s?").replace("%s", fn
);
622 unsubscribeFeed(actid
);
625 case "qmcCatchupAll":
628 case "qmcShowOnlyUnread":
637 case "qmcRescoreFeed":
638 rescoreCurrentFeed();
640 case "qmcToggleWidescreen":
642 _widescreen_mode
= !_widescreen_mode
;
644 switchPanelMode(_widescreen_mode
);
651 console
.log("quickMenuGo: unknown action: " + opid
);
655 exception_error("quickMenuGo", e
);
659 function toggleDispRead() {
662 var hide
= !(getInitParam("hide_read_feeds") == "1");
664 hideOrShowFeeds(hide
);
666 var query
= "?op=rpc&method=setpref&key=HIDE_READ_FEEDS&value=" +
669 setInitParam("hide_read_feeds", hide
);
671 new Ajax
.Request("backend.php", {
673 onComplete: function(transport
) {
677 exception_error("toggleDispRead", e
);
681 function parse_runtime_info(data
) {
683 //console.log("parsing runtime info...");
688 // console.log("RI: " + k + " => " + v);
690 if (k
== "new_version_available") {
692 Element
.show(dijit
.byId("newVersionIcon").domNode
);
694 Element
.hide(dijit
.byId("newVersionIcon").domNode
);
699 if (k
== "dep_ts" && parseInt(getInitParam("dep_ts")) > 0) {
700 if (parseInt(getInitParam("dep_ts")) < parseInt(v
) && getInitParam("reload_on_ts_change")) {
701 window
.location
.reload();
705 if (k
== "daemon_is_running" && v
!= 1) {
706 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
710 if (k
== "daemon_stamp_ok" && v
!= 1) {
711 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
715 if (k
== "max_feed_id" || k
== "num_feeds") {
716 if (init_params
[k
] != v
) {
717 console
.log("feed count changed, need to reload feedlist.");
727 function collapse_feedlist() {
730 if (!Element
.visible('feeds-holder')) {
731 Element
.show('feeds-holder');
732 Element
.show('feeds-holder_splitter');
733 $("collapse_feeds_btn").innerHTML
= "<<";
735 Element
.hide('feeds-holder');
736 Element
.hide('feeds-holder_splitter');
737 $("collapse_feeds_btn").innerHTML
= ">>";
740 dijit
.byId("main").resize();
742 query
= "?op=rpc&method=setpref&key=_COLLAPSED_FEEDLIST&value=true";
743 new Ajax
.Request("backend.php", { parameters
: query
});
746 exception_error("collapse_feedlist", e
);
750 function viewModeChanged() {
752 return viewCurrentFeed('');
755 function rescoreCurrentFeed() {
757 var actid
= getActiveFeedId();
759 if (activeFeedIsCat() || actid
< 0) {
760 alert(__("You can't rescore this kind of feed."));
765 alert(__("Please select some feed first."));
769 var fn
= getFeedName(actid
);
770 var pr
= __("Rescore articles in %s?").replace("%s", fn
);
773 notify_progress("Rescoring articles...");
775 var query
= "?op=pref-feeds&method=rescore&quiet=1&ids=" + actid
;
777 new Ajax
.Request("backend.php", {
779 onComplete: function(transport
) {
785 function hotkey_handler(e
) {
788 if (e
.target
.nodeName
== "INPUT" || e
.target
.nodeName
== "TEXTAREA") return;
791 var shift_key
= false;
793 var cmdline
= $('cmdline');
795 shift_key
= e
.shiftKey
;
796 ctrl_key
= e
.ctrlKey
;
799 keycode
= window
.event
.keyCode
;
804 var keychar
= String
.fromCharCode(keycode
);
806 if (keycode
== 27) { // escape
807 hotkey_prefix
= false;
810 if (keycode
== 16) return; // ignore lone shift
811 if (keycode
== 17) return; // ignore lone ctrl
813 keychar
= keychar
.toLowerCase();
815 var hotkeys
= getInitParam("hotkeys");
817 if (!hotkey_prefix
&& hotkeys
[0].indexOf(keychar
) != -1) {
819 var date
= new Date();
820 var ts
= Math
.round(date
.getTime() / 1000);
822 hotkey_prefix
= keychar
;
823 hotkey_prefix_pressed
= ts
;
825 cmdline
.innerHTML
= keychar
;
826 Element
.show(cmdline
);
831 Element
.hide(cmdline
);
833 var hotkey
= keychar
.search(/[a-zA-Z0-9]/) != -1 ? keychar
: "(" + keycode
+ ")";
835 // ensure ^*char notation
836 if (shift_key
) hotkey
= "*" + hotkey
;
837 if (ctrl_key
) hotkey
= "^" + hotkey
;
839 hotkey
= hotkey_prefix
? hotkey_prefix
+ " " + hotkey
: hotkey
;
840 hotkey_prefix
= false;
842 var hotkey_action
= false;
843 var hotkeys
= getInitParam("hotkeys");
845 for (sequence
in hotkeys
[1]) {
846 if (sequence
== hotkey
) {
847 hotkey_action
= hotkeys
[1][sequence
];
852 var action
= hotkey_actions
[hotkey_action
];
854 if (action
!= null) {
860 exception_error("hotkey_handler", e
);
864 function inPreferences() {
868 function reverseHeadlineOrder() {
871 /* var query_str = "?op=rpc&method=togglepref&key=REVERSE_HEADLINES";
873 new Ajax.Request("backend.php", {
874 parameters: query_str,
875 onComplete: function(transport) {
879 var toolbar
= document
.forms
["main_toolbar_form"];
880 var order_by
= dijit
.getEnclosingWidget(toolbar
.order_by
);
882 var value
= order_by
.attr('value');
884 if (value
== "date_reverse")
887 value
= "date_reverse";
889 order_by
.attr('value', value
);
894 exception_error("reverseHeadlineOrder", e
);
898 function newVersionDlg() {
900 var query
= "backend.php?op=dlg&method=newVersion";
902 if (dijit
.byId("newVersionDlg"))
903 dijit
.byId("newVersionDlg").destroyRecursive();
905 dialog
= new dijit
.Dialog({
907 title
: __("New version available!"),
908 style
: "width: 600px",
915 exception_error("newVersionDlg", e
);
919 function handle_rpc_json(transport
, scheduled_call
) {
921 var reply
= JSON
.parse(transport
.responseText
);
925 var error
= reply
['error'];
928 var code
= error
['code'];
929 var msg
= error
['msg'];
931 console
.warn("[handle_rpc_json] received fatal error " + code
+ "/" + msg
);
934 fatalError(code
, msg
);
939 var seq
= reply
['seq'];
942 if (get_seq() != seq
) {
943 console
.log("[handle_rpc_json] sequence mismatch: " + seq
+
944 " (want: " + get_seq() + ")");
949 var message
= reply
['message'];
952 if (message
== "UPDATE_COUNTERS") {
953 console
.log("need to refresh counters...");
954 setInitParam("last_article_id", -1);
955 request_counters(true);
959 var counters
= reply
['counters'];
962 parse_counters(counters
, scheduled_call
);
964 var runtime_info
= reply
['runtime-info'];;
967 parse_runtime_info(runtime_info
);
969 Element
.hide(dijit
.byId("net-alert").domNode
);
972 //notify_error("Error communicating with server.");
973 Element
.show(dijit
.byId("net-alert").domNode
);
977 Element
.show(dijit
.byId("net-alert").domNode
);
978 //notify_error("Error communicating with server.");
980 //exception_error("handle_rpc_json", e, transport);
986 function switchPanelMode(wide
) {
988 if (isCdmMode()) return;
990 article_id
= getActiveArticleId();
993 dijit
.byId("headlines-wrap-inner").attr("design", 'sidebar');
994 dijit
.byId("content-insert").attr("region", "trailing");
996 dijit
.byId("content-insert").domNode
.setStyle({width
: '50%',
998 borderLeftWidth
: '1px',
999 borderLeftColor
: '#c0c0c0',
1000 borderTopWidth
: '0px' });
1002 $("headlines-toolbar").setStyle({ borderBottomWidth
: '0px' });
1006 dijit
.byId("content-insert").attr("region", "bottom");
1008 dijit
.byId("content-insert").domNode
.setStyle({width
: 'auto',
1010 borderLeftWidth
: '0px',
1011 borderTopWidth
: '1px'});
1013 $("headlines-toolbar").setStyle({ borderBottomWidth
: '1px' });
1016 closeArticlePanel();
1018 if (article_id
) view(article_id
);
1020 new Ajax
.Request("backend.php", {
1021 parameters
: "op=rpc&method=setpanelmode&wide=" + (wide
? 1 : 0),
1022 onComplete: function(transport
) {
1023 console
.log(transport
.responseText
);
1028 exception_error("switchPanelMode", e
);
1032 function update_random_feed() {
1034 console
.log("in update_random_feed");
1036 new Ajax
.Request("backend.php", {
1037 parameters
: "op=rpc&method=updateRandomFeed",
1038 onComplete: function(transport
) {
1039 handle_rpc_json(transport
, true);
1040 window
.setTimeout("update_random_feed()", 30*1000);
1044 exception_error("update_random_feed", e
);
1048 function hash_get(key
) {
1050 kv
= window
.location
.hash
.substring(1).toQueryParams();
1053 exception_error("hash_set", e
);
1056 function hash_set(key
, value
) {
1058 kv
= window
.location
.hash
.substring(1).toQueryParams();
1060 window
.location
.hash
= $H(kv
).toQueryString();
1062 exception_error("hash_set", e
);