ssrurl.htm 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  1. <%+cbi/valueheader%>
  2. <script type="text/javascript">
  3. //<![CDATA[
  4. function padright(str, cnt, pad) {
  5. return str + Array(cnt + 1).join(pad);
  6. }
  7. function b64EncodeUnicode(str) {
  8. return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g, function (match, p1) {
  9. return String.fromCharCode('0x' + p1);
  10. }));
  11. }
  12. function b64encutf8safe(str) {
  13. return b64EncodeUnicode(str).replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/g, '');
  14. }
  15. function b64DecodeUnicode(str) {
  16. return decodeURIComponent(Array.prototype.map.call(atob(str), function (c) {
  17. return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2);
  18. }).join(''));
  19. }
  20. function b64decutf8safe(str) {
  21. var l;
  22. str = str.replace(/-/g, "+").replace(/_/g, "/");
  23. l = str.length;
  24. l = (4 - l % 4) % 4;
  25. if (l) str = padright(str, l, "=");
  26. return b64DecodeUnicode(str);
  27. }
  28. function b64encsafe(str) {
  29. return btoa(str).replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/g, '')
  30. }
  31. function b64decsafe(str) {
  32. var l;
  33. str = str.replace(/-/g, "+").replace(/_/g, "/");
  34. l = str.length;
  35. l = (4 - l % 4) % 4;
  36. if (l) str = padright(str, l, "=");
  37. return atob(str);
  38. }
  39. function dictvalue(d, key) {
  40. var v = d[key];
  41. if (typeof (v) == 'undefined' || v == '') return '';
  42. return b64decsafe(v);
  43. }
  44. function export_ssr_url(btn, urlname, sid) {
  45. var s = document.getElementById(urlname + '-status');
  46. if (!s) return false;
  47. var v_server = document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0];
  48. var v_port = document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0];
  49. var v_protocol = document.getElementsByName('cbid.shadowsocksr.' + sid + '.protocol')[0];
  50. var v_method = document.getElementsByName('cbid.shadowsocksr.' + sid + '.encrypt_method')[0];
  51. var v_obfs = document.getElementsByName('cbid.shadowsocksr.' + sid + '.obfs')[0];
  52. var v_password = document.getElementsByName('cbid.shadowsocksr.' + sid + '.password')[0];
  53. var v_obfs_param = document.getElementsByName('cbid.shadowsocksr.' + sid + '.obfs_param')[0];
  54. var v_protocol_param = document.getElementsByName('cbid.shadowsocksr.' + sid + '.protocol_param')[0];
  55. var v_alias = document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0];
  56. var ssr_str = v_server.value + ":" + v_port.value + ":" + v_protocol.value + ":" + v_method.value + ":" + v_obfs.value + ":" + b64encsafe(v_password.value) + "/?obfsparam=" + b64encsafe(v_obfs_param.value) + "&protoparam=" + b64encsafe(v_protocol_param.value) + "&remarks=" + b64encutf8safe(v_alias.value);
  57. var textarea = document.createElement("textarea");
  58. textarea.textContent = "ssr://" + b64encsafe(ssr_str);
  59. textarea.style.position = "fixed";
  60. document.body.appendChild(textarea);
  61. textarea.select();
  62. try {
  63. document.execCommand("copy"); // Security exception may be thrown by some browsers.
  64. s.innerHTML = "<font color='green'><%:Copy SSR to clipboard successfully.%></font>";
  65. } catch (ex) {
  66. s.innerHTML = "<font color='red'><%:Unable to copy SSR to clipboard.%></font>";
  67. } finally {
  68. document.body.removeChild(textarea);
  69. }
  70. return false;
  71. }
  72. function import_ssr_url(btn, urlname, sid) {
  73. var s = document.getElementById(urlname + '-status');
  74. if (!s) return false;
  75. var ssrurl = prompt("<%:Paste sharing link here%>", "");
  76. if (ssrurl == null || ssrurl == "") {
  77. s.innerHTML = "<font color='red'><%:User cancelled.%></font>";
  78. return false;
  79. }
  80. s.innerHTML = "";
  81. //var ssu = ssrurl.match(/ssr:\/\/([A-Za-z0-9_-]+)/i);
  82. var ssu = ssrurl.split('://');
  83. //console.log(ssu.length);
  84. var event = document.createEvent("HTMLEvents");
  85. event.initEvent("change", true, true);
  86. switch (ssu[0]) {
  87. case "hysteria":
  88. try {
  89. var url = new URL("http://" + ssu[1]);
  90. var params = url.searchParams;
  91. } catch(e) {
  92. alert(e);
  93. return false;
  94. }
  95. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].value = ssu[0];
  96. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].dispatchEvent(event);
  97. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0].value = url.hostname;
  98. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0].value = url.port || "80";
  99. document.getElementsByName('cbid.shadowsocksr.' + sid + '.hysteria_protocol')[0].value = params.get("protocol") || "udp";
  100. document.getElementsByName('cbid.shadowsocksr.' + sid + '.auth_type')[0].value = params.get("auth") ? "2" : "0";
  101. document.getElementsByName('cbid.shadowsocksr.' + sid + '.auth_type')[0].dispatchEvent(event);
  102. document.getElementsByName('cbid.shadowsocksr.' + sid + '.auth_payload')[0].value = params.get("auth") || "";
  103. document.getElementsByName('cbid.shadowsocksr.' + sid + '.uplink_capacity')[0].value = params.get("upmbps") || "";
  104. document.getElementsByName('cbid.shadowsocksr.' + sid + '.downlink_capacity')[0].value = params.get("downmbps") || "";
  105. document.getElementsByName('cbid.shadowsocksr.' + sid + '.seed')[0].value = params.get("obfsParam") || "";
  106. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls_host')[0].value = params.get("peer") || "";
  107. document.getElementsByName('cbid.shadowsocksr.' + sid + '.quic_tls_alpn')[0].value = params.get("alpn") || "";
  108. document.getElementsByName('cbid.shadowsocksr.' + sid + '.insecure')[0].checked = params.get("insecure") ? true : false;
  109. document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0].value = url.hash ? decodeURIComponent(url.hash.slice(1)) : "";
  110. s.innerHTML = "<font color='green'><%:Import configuration information successfully.%></font>";
  111. return false;
  112. case "ss":
  113. var url0, param = "";
  114. var sipIndex = ssu[1].indexOf("@");
  115. var ploc = ssu[1].indexOf("#");
  116. if (ploc > 0) {
  117. url0 = ssu[1].substr(0, ploc);
  118. param = ssu[1].substr(ploc + 1);
  119. } else {
  120. url0 = ssu[1];
  121. }
  122. if (sipIndex != -1) {
  123. // SIP002
  124. var userInfo = b64decsafe(url0.substr(0, sipIndex));
  125. var temp = url0.substr(sipIndex + 1).split("/?");
  126. var serverInfo = temp[0].split(":");
  127. var server = serverInfo[0];
  128. var port = serverInfo[1].replace("/","");
  129. var method, password, plugin, pluginOpts;
  130. if (temp[1]) {
  131. var pluginInfo = decodeURIComponent(temp[1]);
  132. var pluginIndex = pluginInfo.indexOf(";");
  133. var pluginNameInfo = pluginInfo.substr(0, pluginIndex);
  134. plugin = pluginNameInfo.substr(pluginNameInfo.indexOf("=") + 1);
  135. pluginOpts = pluginInfo.substr(pluginIndex + 1);
  136. }
  137. var userInfoSplitIndex = userInfo.indexOf(":");
  138. if (userInfoSplitIndex != -1) {
  139. method = userInfo.substr(0, userInfoSplitIndex);
  140. password = userInfo.substr(userInfoSplitIndex + 1);
  141. }
  142. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].value = ssu[0];
  143. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].dispatchEvent(event);
  144. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0].value = server;
  145. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0].value = port;
  146. document.getElementsByName('cbid.shadowsocksr.' + sid + '.password')[0].value = password || "";
  147. document.getElementsByName('cbid.shadowsocksr.' + sid + '.encrypt_method_ss')[0].value = method || "";
  148. document.getElementsByName('cbid.shadowsocksr.' + sid + '.plugin')[0].value = plugin || "none";
  149. document.getElementsByName('cbid.shadowsocksr.' + sid + '.plugin')[0].dispatchEvent(event);
  150. if (plugin != undefined) {
  151. document.getElementsByName('cbid.shadowsocksr.' + sid + '.plugin_opts')[0].value = pluginOpts || "";
  152. }
  153. if (param != undefined) {
  154. document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0].value = decodeURI(param);
  155. }
  156. s.innerHTML = "<font color='green'><%:Import configuration information successfully.%></font>";
  157. } else {
  158. var sstr = b64decsafe(url0);
  159. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].value = ssu[0];
  160. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].dispatchEvent(event);
  161. var team = sstr.split('@');
  162. var part1 = team[0].split(':');
  163. var part2 = team[1].split(':');
  164. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0].value = part2[0];
  165. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0].value = part2[1];
  166. document.getElementsByName('cbid.shadowsocksr.' + sid + '.password')[0].value = part1[1];
  167. document.getElementsByName('cbid.shadowsocksr.' + sid + '.encrypt_method_ss')[0].value = part1[0];
  168. if (param != undefined) {
  169. document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0].value = decodeURI(param);
  170. }
  171. s.innerHTML = "<font color='green'><%:Import configuration information successfully.%></font>";
  172. }
  173. return false;
  174. case "ssr":
  175. var sstr = b64decsafe(ssu[1]);
  176. var ploc = sstr.indexOf("/?");
  177. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].value = ssu[0];
  178. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].dispatchEvent(event);
  179. var url0, param = "";
  180. if (ploc > 0) {
  181. url0 = sstr.substr(0, ploc);
  182. param = sstr.substr(ploc + 2);
  183. }
  184. var ssm = url0.match(/^(.+):([^:]+):([^:]*):([^:]+):([^:]*):([^:]+)/);
  185. if (!ssm || ssm.length < 7) return false;
  186. var pdict = {};
  187. if (param.length > 2) {
  188. var a = param.split('&');
  189. for (var i = 0; i < a.length; i++) {
  190. var b = a[i].split('=');
  191. pdict[decodeURIComponent(b[0])] = decodeURIComponent(b[1] || '');
  192. }
  193. }
  194. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0].value = ssm[1];
  195. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0].value = ssm[2];
  196. document.getElementsByName('cbid.shadowsocksr.' + sid + '.protocol')[0].value = ssm[3];
  197. document.getElementsByName('cbid.shadowsocksr.' + sid + '.encrypt_method')[0].value = ssm[4];
  198. document.getElementsByName('cbid.shadowsocksr.' + sid + '.obfs')[0].value = ssm[5];
  199. document.getElementsByName('cbid.shadowsocksr.' + sid + '.password')[0].value = b64decsafe(ssm[6]);
  200. document.getElementsByName('cbid.shadowsocksr.' + sid + '.obfs_param')[0].value = dictvalue(pdict, 'obfsparam');
  201. document.getElementsByName('cbid.shadowsocksr.' + sid + '.protocol_param')[0].value = dictvalue(pdict, 'protoparam');
  202. var rem = pdict['remarks'];
  203. if (typeof (rem) != 'undefined' && rem != '' && rem.length > 0) document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0].value = b64decutf8safe(rem);
  204. s.innerHTML = "<font color='green'><%:Import configuration information successfully.%></font>";
  205. return false;
  206. case "trojan":
  207. try {
  208. var url = new URL("http://" + ssu[1]);
  209. } catch(e) {
  210. alert(e)
  211. return false;
  212. }
  213. document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0].value = url.hash ? decodeURIComponent(url.hash.slice(1)) : "";
  214. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].value = "v2ray";
  215. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].dispatchEvent(event);
  216. document.getElementsByName('cbid.shadowsocksr.' + sid + '.v2ray_protocol')[0].value = "trojan";
  217. document.getElementsByName('cbid.shadowsocksr.' + sid + '.v2ray_protocol')[0].dispatchEvent(event);
  218. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0].value = url.hostname;
  219. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0].value = url.port || "80";
  220. document.getElementsByName('cbid.shadowsocksr.' + sid + '.password')[0].value = decodeURIComponent(url.username);
  221. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls')[0].checked = true;
  222. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls')[0].dispatchEvent(event);
  223. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls_host')[0].value = url.searchParams.get("sni");
  224. s.innerHTML = "<font color='green'><%:Import configuration information successfully.%></font>";
  225. return false;
  226. case "vmess":
  227. var sstr = b64DecodeUnicode(ssu[1]);
  228. var ploc = sstr.indexOf("/?");
  229. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].value = "v2ray";
  230. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].dispatchEvent(event);
  231. document.getElementsByName('cbid.shadowsocksr.' + sid + '.v2ray_protocol')[0].value = "vmess";
  232. document.getElementsByName('cbid.shadowsocksr.' + sid + '.v2ray_protocol')[0].dispatchEvent(event);
  233. var url0, param = "";
  234. if (ploc > 0) {
  235. url0 = sstr.substr(0, ploc);
  236. param = sstr.substr(ploc + 2);
  237. }
  238. var ssm = JSON.parse(sstr);
  239. document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0].value = ssm.ps;
  240. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0].value = ssm.add;
  241. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0].value = ssm.port;
  242. document.getElementsByName('cbid.shadowsocksr.' + sid + '.vmess_id')[0].value = ssm.id;
  243. document.getElementsByName('cbid.shadowsocksr.' + sid + '.transport')[0].value = ssm.net;
  244. document.getElementsByName('cbid.shadowsocksr.' + sid + '.transport')[0].dispatchEvent(event);
  245. if (ssm.net == "tcp") {
  246. if (ssm.type && ssm.type != "http") {
  247. ssm.type = "none"
  248. }
  249. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tcp_guise')[0].value = ssm.type;
  250. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tcp_guise')[0].dispatchEvent(event);
  251. document.getElementsByName('cbid.shadowsocksr.' + sid + '.http_host')[0].value = ssm.host;
  252. document.getElementsByName('cbid.shadowsocksr.' + sid + '.http_path')[0].value = ssm.path;
  253. }
  254. if (ssm.net == "ws") {
  255. document.getElementsByName('cbid.shadowsocksr.' + sid + '.ws_host')[0].value = ssm.host;
  256. document.getElementsByName('cbid.shadowsocksr.' + sid + '.ws_path')[0].value = ssm.path;
  257. }
  258. if (ssm.net == "h2") {
  259. document.getElementsByName('cbid.shadowsocksr.' + sid + '.h2_host')[0].value = ssm.host;
  260. document.getElementsByName('cbid.shadowsocksr.' + sid + '.h2_path')[0].value = ssm.path;
  261. }
  262. if (ssm.net == "quic") {
  263. document.getElementsByName('cbid.shadowsocksr.' + sid + '.quic_security')[0].value = ssm.securty;
  264. document.getElementsByName('cbid.shadowsocksr.' + sid + '.quic_key')[0].value = ssm.key;
  265. }
  266. if (ssm.net == "kcp") {
  267. document.getElementsByName('cbid.shadowsocksr.' + sid + '.kcp_guise')[0].value = ssm.type;
  268. }
  269. if (ssm.tls == "tls") {
  270. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls')[0].checked = true;
  271. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls')[0].dispatchEvent(event);
  272. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls_host')[0].value = ssm.host;
  273. }
  274. document.getElementsByName('cbid.shadowsocksr.' + sid + '.mux')[0].checked = true;
  275. document.getElementsByName('cbid.shadowsocksr.' + sid + '.mux')[0].dispatchEvent(event);
  276. s.innerHTML = "<font color='green'><%:Import configuration information successfully.%></font>";
  277. return false;
  278. case "vless":
  279. try {
  280. var url = new URL("http://" + ssu[1]);
  281. var params = url.searchParams;
  282. } catch(e) {
  283. alert(e)
  284. return false;
  285. }
  286. document.getElementsByName('cbid.shadowsocksr.' + sid + '.alias')[0].value = url.hash ? decodeURIComponent(url.hash.slice(1)) : "";
  287. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].value = "v2ray";
  288. document.getElementsByName('cbid.shadowsocksr.' + sid + '.type')[0].dispatchEvent(event);
  289. document.getElementsByName('cbid.shadowsocksr.' + sid + '.v2ray_protocol')[0].value = "vless";
  290. document.getElementsByName('cbid.shadowsocksr.' + sid + '.v2ray_protocol')[0].dispatchEvent(event);
  291. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server')[0].value = url.hostname;
  292. document.getElementsByName('cbid.shadowsocksr.' + sid + '.server_port')[0].value = url.port || "80";
  293. document.getElementsByName('cbid.shadowsocksr.' + sid + '.vmess_id')[0].value = url.username;
  294. document.getElementsByName('cbid.shadowsocksr.' + sid + '.transport')[0].value = params.get("type") == "http" ? "h2" : params.get("type") || "tcp";
  295. document.getElementsByName('cbid.shadowsocksr.' + sid + '.transport')[0].dispatchEvent(event);
  296. document.getElementsByName('cbid.shadowsocksr.' + sid + '.vless_encryption')[0].value = params.get("encryption") || "none";
  297. if ([ "tls" ].includes(params.get("security"))) {
  298. document.getElementsByName('cbid.shadowsocksr.' + sid + '.' + params.get("security"))[0].checked = true;
  299. document.getElementsByName('cbid.shadowsocksr.' + sid + '.' + params.get("security"))[0].dispatchEvent(event);
  300. document.getElementsByName('cbid.shadowsocksr.' + sid + '.fingerprint')[0].value = params.get("fp") || "";
  301. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tls_host')[0].value = params.get("sni") || "";
  302. }
  303. switch (params.get("type")) {
  304. case "ws":
  305. if (params.get("security") !== "tls")
  306. document.getElementsByName('cbid.shadowsocksr.' + sid + '.ws_host')[0].value = params.get("host") ? decodeURIComponent(params.get("host")) : "";
  307. document.getElementsByName('cbid.shadowsocksr.' + sid + '.ws_path')[0].value = params.get("path") ? decodeURIComponent(params.get("path")) : "/";
  308. break;
  309. case "kcp":
  310. document.getElementsByName('cbid.shadowsocksr.' + sid + '.kcp_guise')[0].value = params.get("headerType") || "none";
  311. document.getElementsByName('cbid.shadowsocksr.' + sid + '.seed')[0].value = params.get("seed") || "";
  312. break;
  313. case "http":
  314. /* this is non-standard, bullshit */
  315. case "h2":
  316. document.getElementsByName('cbid.shadowsocksr.' + sid + '.h2_host')[0].value = params.get("host") ? decodeURIComponent(params.get("host")) : "";
  317. document.getElementsByName('cbid.shadowsocksr.' + sid + '.h2_path')[0].value = params.get("path") ? decodeURIComponent(params.get("path")) : "";
  318. break;
  319. case "quic":
  320. document.getElementsByName('cbid.shadowsocksr.' + sid + '.quic_guise')[0].value = params.get("headerType") || "none";
  321. document.getElementsByName('cbid.shadowsocksr.' + sid + '.quic_security')[0].value = params.get("quicSecurity") || "none";
  322. document.getElementsByName('cbid.shadowsocksr.' + sid + '.quic_key')[0].value = params.get("key") || "";
  323. break;
  324. case "grpc":
  325. document.getElementsByName('cbid.shadowsocksr.' + sid + '.serviceName')[0].value = params.get("serviceName") || "";
  326. document.getElementsByName('cbid.shadowsocksr.' + sid + '.grpc_mode')[0].value = params.get("mode") || "gun";
  327. break;
  328. case "tcp":
  329. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tcp_guise')[0].value = params.get("headerType") || "none";
  330. document.getElementsByName('cbid.shadowsocksr.' + sid + '.tcp_guise')[0].dispatchEvent(event);
  331. if (params.get("headerType") === "http") {
  332. document.getElementsByName('cbid.shadowsocksr.' + sid + '.http_host')[0].value = params.get("host") ? decodeURIComponent(params.get("host")) : "";
  333. document.getElementsByName('cbid.shadowsocksr.' + sid + '.http_path')[0].value = params.get("path") ? decodeURIComponent(params.get("path")) : "";
  334. }
  335. break;
  336. }
  337. s.innerHTML = "<font color='green'><%:Import configuration information successfully.%></font>";
  338. return false;
  339. default:
  340. s.innerHTML = "<font color='red'><%:Invalid format.%></font>";
  341. return false;
  342. }
  343. }
  344. //]]>
  345. </script>
  346. <input type="button" class="btn cbi-button cbi-button-apply" value="<%:Import%>" onclick="return import_ssr_url(this, '<%=self.option%>', '<%=self.value%>')" />
  347. <span id="<%=self.option%>-status"></span>
  348. <%+cbi/valuefooter%>