Merge pull request #470 from thesabbir/patch-01
[project/luci.git] / themes / luci-theme-material / htdocs / luci-static / material / js / script.js
1 /**
2 * Material is a clean HTML5 theme for LuCI. It is based on luci-theme-bootstrap and MUI
3 *
4 * luci-theme-material
5 * Copyright 2015 Lutty Yang <lutty@wcan.in>
6 *
7 * Have a bug? Please create an issue here on GitHub!
8 * https://github.com/LuttyYang/luci-theme-material/issues
9 *
10 * luci-theme-bootstrap:
11 * Copyright 2008 Steven Barth <steven@midlink.org>
12 * Copyright 2008 Jo-Philipp Wich <jow@openwrt.org>
13 * Copyright 2012 David Menting <david@nut-bolt.nl>
14 *
15 * MUI:
16 * https://github.com/muicss/mui
17 *
18 * Licensed to the public under the Apache License 2.0
19 */
20 (function ($) {
21 var b64 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",
22 a256 = '',
23 r64 = [256],
24 r256 = [256],
25 i = 0;
26 var UTF8 = {
27 /**
28 * Encode multi-byte Unicode string into utf-8 multiple single-byte characters
29 * (BMP / basic multilingual plane only)
30 *
31 * Chars in range U+0080 - U+07FF are encoded in 2 chars, U+0800 - U+FFFF in 3 chars
32 *
33 * @param {String} strUni Unicode string to be encoded as UTF-8
34 * @returns {String} encoded string
35 */
36 encode: function (strUni) {
37 // use regular expressions & String.replace callback function for better efficiency
38 // than procedural approaches
39 var strUtf = strUni.replace(/[\u0080-\u07ff]/g, // U+0080 - U+07FF => 2 bytes 110yyyyy, 10zzzzzz
40 function (c) {
41 var cc = c.charCodeAt(0);
42 return String.fromCharCode(0xc0 | cc >> 6, 0x80 | cc & 0x3f);
43 })
44 .replace(/[\u0800-\uffff]/g, // U+0800 - U+FFFF => 3 bytes 1110xxxx, 10yyyyyy, 10zzzzzz
45 function (c) {
46 var cc = c.charCodeAt(0);
47 return String.fromCharCode(0xe0 | cc >> 12, 0x80 | cc >> 6 & 0x3F, 0x80 | cc & 0x3f);
48 });
49 return strUtf;
50 },
51 /**
52 * Decode utf-8 encoded string back into multi-byte Unicode characters
53 *
54 * @param {String} strUtf UTF-8 string to be decoded back to Unicode
55 * @returns {String} decoded string
56 */
57 decode: function (strUtf) {
58 // note: decode 3-byte chars first as decoded 2-byte strings could appear to be 3-byte char!
59 var strUni = strUtf.replace(/[\u00e0-\u00ef][\u0080-\u00bf][\u0080-\u00bf]/g, // 3-byte chars
60 function (c) { // (note parentheses for precence)
61 var cc = ((c.charCodeAt(0) & 0x0f) << 12) | ((c.charCodeAt(1) & 0x3f) << 6) | (c.charCodeAt(2) & 0x3f);
62 return String.fromCharCode(cc);
63 })
64 .replace(/[\u00c0-\u00df][\u0080-\u00bf]/g, // 2-byte chars
65 function (c) { // (note parentheses for precence)
66 var cc = (c.charCodeAt(0) & 0x1f) << 6 | c.charCodeAt(1) & 0x3f;
67 return String.fromCharCode(cc);
68 });
69 return strUni;
70 }
71 };
72 while (i < 256) {
73 var c = String.fromCharCode(i);
74 a256 += c;
75 r256[i] = i;
76 r64[i] = b64.indexOf(c);
77 ++i;
78 }
79 function code(s, discard, alpha, beta, w1, w2) {
80 s = String(s);
81 var buffer = 0,
82 i = 0,
83 length = s.length,
84 result = '',
85 bitsInBuffer = 0;
86 while (i < length) {
87 var c = s.charCodeAt(i);
88 c = c < 256 ? alpha[c] : -1;
89 buffer = (buffer << w1) + c;
90 bitsInBuffer += w1;
91 while (bitsInBuffer >= w2) {
92 bitsInBuffer -= w2;
93 var tmp = buffer >> bitsInBuffer;
94 result += beta.charAt(tmp);
95 buffer ^= tmp << bitsInBuffer;
96 }
97 ++i;
98 }
99 if (!discard && bitsInBuffer > 0) result += beta.charAt(buffer << (w2 - bitsInBuffer));
100 return result;
101 }
102
103 var Plugin = $.base64 = function (dir, input, encode) {
104 return input ? Plugin[dir](input, encode) : dir ? null : this;
105 };
106 Plugin.btoa = Plugin.encode = function (plain, utf8encode) {
107 plain = Plugin.raw === false || Plugin.utf8encode || utf8encode ? UTF8.encode(plain) : plain;
108 plain = code(plain, false, r256, b64, 8, 6);
109 return plain + '===='.slice((plain.length % 4) || 4);
110 };
111 Plugin.atob = Plugin.decode = function (coded, utf8decode) {
112 coded = String(coded).split('=');
113 var i = coded.length;
114 do {
115 --i;
116 coded[i] = code(coded[i], true, r64, a256, 6, 8);
117 } while (i > 0);
118 coded = coded.join('');
119 return Plugin.raw === false || Plugin.utf8decode || utf8decode ? UTF8.decode(coded) : coded;
120 };
121 }(jQuery));
122
123 (function ($) {
124 /**
125 * trim text, Remove spaces, wrap
126 * @param text
127 * @returns {string}
128 */
129 function trimText(text) {
130 return text.replace(/[ \t\n\r]+/g, " ");
131 }
132
133
134 var tree = undefined;
135 var lastNode = undefined;
136
137 /**
138 * get the current node by Hash (reserve)
139 * @returns {boolean} success?
140 */
141 function getCurrentNodeByHash() {
142 var ret = false;
143 var hash = window.location.hash;
144 if (hash.substr(0, 6) == "#tree-") {
145 hash = $.base64.decode(hash.substr(6));
146 tree = hash.split("|");
147 $(".main > .main-left > .nav > .slide > .menu").each(function () {
148 var that = $(this);
149 if (trimText(that.data("title")) == tree[0]) {
150 that.click();
151 that.next().find("a").each(function () {
152 var that = $(this);
153 if (trimText(that.data("title")) == tree[1]) {
154 lastNode = that.parent();
155 lastNode.addClass("active");
156 ret = true;
157 return true;
158 }
159 });
160 }
161 });
162 }
163 return ret;
164 }
165
166 /**
167 * get the current node by Burl (primary)
168 * @returns {boolean} success?
169 */
170 function getCurrentNodeByUrl() {
171 var ret = false;
172 var getUrlNode = function (href){
173 var linkPos = href.indexOf(";");
174 if (linkPos == -1){
175 return "login";
176 }else{
177 linkPos = href.indexOf("/", linkPos);
178 if (linkPos == -1){
179 return "overview";
180 }else{
181 var link = href.substr(linkPos);
182 if (link == "/")
183 return "overview";
184 else
185 return link;
186 }
187 }
188 };
189
190 var currentNode = getUrlNode(window.location.pathname);
191
192 if (currentNode == "login"){
193 tree = ["Main", "Login"];
194 return false;
195 }else if(currentNode == "overview"){
196 tree = ["Status", "Overview"];
197 lastNode = $($($(".main > .main-left > .nav > .slide > .menu")[0]).next().find("a")[0]).parent();
198 return false;
199 }
200
201 $(".main > .main-left > .nav > .slide > .menu").each(function () {
202 var ulNode = $(this);
203 ulNode.next().find("a").each(function () {
204 var that = $(this);
205 var href = that.attr("href");
206
207 if (currentNode.indexOf(getUrlNode(href)) != -1){
208 ulNode.click();
209 lastNode = that.parent();
210 tree = [trimText(ulNode.data("title")), trimText(that.data("title"))];
211 lastNode.addClass("active");
212 ret = true;
213 return true;
214 }
215 });
216 });
217 return ret;
218 }
219
220 /**
221 * menu click
222 */
223 $(".main > .main-left > .nav > .slide > .menu").click(function () {
224 var ul = $(this).next(".slide-menu");
225 var menu = $(this);
226 if (!ul.is(":visible")) {
227 menu.addClass("active");
228 ul.addClass("active");
229 ul.stop(true).slideDown();
230 } else {
231 ul.slideUp(function () {
232 menu.removeClass("active");
233 ul.removeClass("active");
234 });
235 }
236 });
237
238 /**
239 * hook menu click and add the hash
240 */
241 $(".main > .main-left > .nav > .slide > .slide-menu > li > a").click(function () {
242 var href = $(this).attr("href");
243 var tree = trimText($(this).parent().parent().prev().data("title")) + "|" + trimText($(this).data("title"));
244 tree = $.base64.encode(tree);
245 window.location = href + "#tree-" + tree;
246 if (lastNode != undefined) lastNode.removeClass("active");
247 $(this).parent().addClass("active");
248 return false;
249 });
250
251 /**
252 * fix menu click
253 */
254 $(".main > .main-left > .nav > .slide > .slide-menu > li").click(function () {
255 $(this).find("a").click();
256 });
257
258 /**
259 * get current node and open it
260 */
261 if (!getCurrentNodeByUrl()){
262 getCurrentNodeByHash();
263 if (tree[0] == "Status" && tree[1] == "Overview"){
264 //overview
265 lastNode.addClass("active");
266 $($(".main > .main-left > .nav > .slide > .menu")[0]).click();
267 }
268 }
269 var mainNodeName = "node-"+ tree[0] + "-" + tree[1];
270 $("body").addClass(mainNodeName.replace(/[ \t\n\r\/]+/g,"_").toLowerCase());
271 $(".cbi-button-up").val("");
272 $(".cbi-button-down").val("");
273
274
275 /**
276 * hook other "A Label" and add hash to it.
277 */
278 $("#maincontent > .container").find("a").each(function () {
279 var that = $(this);
280 var onclick = that.attr("onclick");
281 if (onclick == undefined || onclick == ""){
282 that.click(function () {
283 var href = that.attr("href");
284 if (tree != undefined && href.indexOf("Text") == -1) {
285 window.location = href + "#tree-" + $.base64.encode(tree[0] + "|" + tree[1]);
286 return false;
287 }else{
288 return true;
289 }
290 });
291 }
292 });
293
294 /**
295 * Sidebar expand
296 */
297 var showSide = false;
298 $(".showSide").click(function () {
299 if (showSide){
300 $(".darkMask").stop(true).fadeOut();
301 $(".main-left").stop(true).animate({
302 width: "0"
303 });
304 showSide = false;
305 }else{
306 $(".darkMask").stop(true).fadeIn();
307 $(".main-left").stop(true).animate({
308 width: "15rem"
309 });
310 showSide = true;
311 }
312 });
313
314
315 $(".darkMask").click(function () {
316 if (showSide){
317 showSide = false;
318 $(".darkMask").stop(true).fadeOut();
319 $(".main-left").stop(true).animate({
320 width: "0"
321 });
322 }
323 });
324
325 $(window).resize(function() {
326 if ($(window).width() > 921) {
327 $(".main-left").css("width", "");
328 $(".darkMask").stop(true);
329 $(".darkMask").css("display", "none");
330 showSide = false;
331 }
332 });
333
334 /**
335 * fix legend position
336 */
337 $("legend").each(function () {
338 var that = $(this);
339 that.after("<span class='panel-title'>" + that.text() + "</span>");
340 });
341 })(jQuery);