8 'require tools.widgets as widgets';
13 function handleAction(ev
) {
15 L
.ui
.showModal(_('Refresh Timer'), [
16 E('p', _('To keep your adblock lists up-to-date, you should setup an automatic update job for these lists.')),
17 E('div', { 'class': 'left', 'style': 'display:flex; flex-direction:column' }, [
18 E('h5', _('Existing job(s)')),
21 'style': 'width: 100% !important; padding: 5px; font-family: monospace',
22 'readonly': 'readonly',
27 E('div', { 'class': 'left', 'style': 'display:flex; flex-direction:column' }, [
28 E('label', { 'class': 'cbi-input-select', 'style': 'padding-top:.5em' }, [
29 E('h5', _('Set a new adblock job')),
30 E('select', { 'class': 'cbi-input-select', 'id': 'timerA' }, [
31 E('option', { 'value': 'start' }, 'Start'),
32 E('option', { 'value': 'reload' }, 'Reload'),
33 E('option', { 'value': 'restart' }, 'Restart'),
34 E('option', { 'value': 'suspend' }, 'Suspend'),
35 E('option', { 'value': 'resume' }, 'Resume'),
36 E('option', { 'value': 'report gen' }, 'Report'),
37 E('option', { 'value': 'report mail' }, 'Report & Mail')
42 E('label', { 'class': 'cbi-input-text', 'style': 'padding-top:.5em' }, [
43 E('input', { 'class': 'cbi-input-text', 'id': 'timerH', 'maxlength': '2' }, [
46 _('The hours portition (req., range: 0-23)')
48 E('label', { 'class': 'cbi-input-text', 'style': 'padding-top:.5em' }, [
49 E('input', { 'class': 'cbi-input-text', 'id': 'timerM', 'maxlength': '2' }),
51 _('The minutes portion (opt., range: 0-59)')
53 E('label', { 'class': 'cbi-input-text', 'style': 'padding-top:.5em' }, [
54 E('input', { 'class': 'cbi-input-text', 'id': 'timerD', 'maxlength': '13' }),
56 _('The day of the week (opt., values: 1-7 possibly sep. by , or -)')
59 E('div', { 'class': 'left', 'style': 'display:flex; flex-direction:column' }, [
60 E('label', { 'class': 'cbi-input-select', 'style': 'padding-top:.5em' }, [
61 E('h5', _('Remove an existing job')),
62 E('input', { 'class': 'cbi-input-text', 'id': 'lineno', 'maxlength': '2' }, [
65 _('Line number to remove')
68 E('div', { 'class': 'right' }, [
70 'class': 'btn cbi-button',
75 'class': 'btn cbi-button-action',
76 'click': ui
.createHandlerFn(this, function(ev
) {
77 var lineno
= document
.getElementById('lineno').value
;
78 var action
= document
.getElementById('timerA').value
;
79 var hours
= document
.getElementById('timerH').value
;
80 var minutes
= document
.getElementById('timerM').value
|| '0';
81 var days
= document
.getElementById('timerD').value
|| '*';
83 L
.resolveDefault(fs
.exec_direct('/etc/init.d/adblock', ['timer', 'add', action
, hours
, minutes
, days
]))
86 ui
.addNotification(null, E('p', _('The Refresh Timer could not been updated.')), 'error');
88 ui
.addNotification(null, E('p', _('The Refresh Timer has been updated.')), 'info');
92 L
.resolveDefault(fs
.exec_direct('/etc/init.d/adblock', ['timer', 'remove', lineno
]))
95 ui
.addNotification(null, E('p', _('The Refresh Timer could not been updated.')), 'error');
97 ui
.addNotification(null, E('p', _('The Refresh Timer has been updated.')), 'info');
101 document
.getElementById('timerH').focus();
109 L
.resolveDefault(fs
.exec_direct('/etc/init.d/adblock', ['timer', 'list']))
110 .then(function(res
) {
111 document
.getElementById('cronView').value
= res
.trim();
113 document
.getElementById('timerH').focus();
117 if (document
.getElementById('status') && document
.getElementById('status').textContent
.substr(0,6) === 'paused') {
121 fs
.exec_direct('/etc/init.d/adblock', [ev
])
127 L
.resolveDefault(fs
.exec_direct('/etc/init.d/adblock', ['list']), {}),
128 L
.resolveDefault(fs
.read_direct('/etc/adblock/adblock.categories'), ''),
133 render: function(result
) {
136 m
= new form
.Map('adblock', 'Adblock', _('Configuration of the adblock package to block ad/abuse domains by using DNS. \
137 For further information <a href="https://github.com/openwrt/packages/blob/master/net/adblock/files/README.md" target="_blank" rel="noreferrer noopener" >check the online documentation</a>'));
140 poll runtime information
142 pollData
: poll
.add(function() {
143 return L
.resolveDefault(fs
.read_direct('/tmp/adb_runtime.json'), 'null').then(function(res
) {
144 var info
= JSON
.parse(res
);
145 var status
= document
.getElementById('status');
146 if (status
&& info
) {
147 status
.textContent
= (info
.adblock_status
|| '-') + ' / ' + (info
.adblock_version
|| '-');
148 if (info
.adblock_status
=== "running") {
149 if (!status
.classList
.contains("spinning")) {
150 status
.classList
.add("spinning");
153 if (status
.classList
.contains("spinning")) {
154 status
.classList
.remove("spinning");
155 if (document
.getElementById('btn_suspend')) {
156 if (status
.textContent
.substr(0,6) === 'paused') {
157 document
.querySelector('#btn_suspend').textContent
= 'Resume';
159 if (document
.getElementById('status').textContent
.substr(0,7) === 'enabled') {
160 document
.querySelector('#btn_suspend').textContent
= 'Suspend';
165 if (status
.textContent
.substr(0,6) === 'paused' && document
.getElementById('btn_suspend')) {
166 document
.querySelector('#btn_suspend').textContent
= 'Resume';
169 status
.textContent
= '-';
170 if (status
.classList
.contains("spinning")) {
171 status
.classList
.remove("spinning");
174 var domains
= document
.getElementById('domains');
175 if (domains
&& info
) {
176 domains
.textContent
= parseInt(info
.blocked_domains
, 10).toLocaleString() || '-';
178 var sources
= document
.getElementById('sources');
180 if (sources
&& info
) {
181 for (var i
= 0; i
< info
.active_sources
.length
; i
++) {
182 if (i
< info
.active_sources
.length
-1) {
183 src_array
+= info
.active_sources
[i
].source
+ ', ';
185 src_array
+= info
.active_sources
[i
].source
188 sources
.textContent
= src_array
|| '-';
190 var backend
= document
.getElementById('backend');
191 if (backend
&& info
) {
192 backend
.textContent
= info
.dns_backend
|| '-';
194 var utils
= document
.getElementById('utils');
196 utils
.textContent
= info
.run_utils
|| '-';
198 var ifaces
= document
.getElementById('ifaces');
199 if (ifaces
&& info
) {
200 ifaces
.textContent
= info
.run_ifaces
|| '-';
202 var dirs
= document
.getElementById('dirs');
204 dirs
.textContent
= info
.run_directories
|| '-';
206 var flags
= document
.getElementById('flags');
208 flags
.textContent
= info
.run_flags
|| '-';
210 var run
= document
.getElementById('run');
212 run
.textContent
= info
.last_run
|| '-';
218 runtime information and buttons
220 s
= m
.section(form
.NamedSection
, 'global');
221 s
.render
= L
.bind(function(view
, section_id
) {
222 return E('div', { 'class': 'cbi-section' }, [
223 E('h3', _('Information')),
224 E('div', { 'class': 'cbi-value' }, [
225 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Status / Version')),
226 E('div', { 'class': 'cbi-value-field spinning', 'id': 'status', 'style': 'color:#37c' },'\xa0')
228 E('div', { 'class': 'cbi-value' }, [
229 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Blocked Domains')),
230 E('div', { 'class': 'cbi-value-field', 'id': 'domains', 'style': 'color:#37c' },'-')
232 E('div', { 'class': 'cbi-value' }, [
233 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Active Sources')),
234 E('div', { 'class': 'cbi-value-field', 'id': 'sources', 'style': 'color:#37c' },'-')
236 E('div', { 'class': 'cbi-value' }, [
237 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('DNS Backend')),
238 E('div', { 'class': 'cbi-value-field', 'id': 'backend', 'style': 'color:#37c' },'-')
240 E('div', { 'class': 'cbi-value' }, [
241 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Run Utils')),
242 E('div', { 'class': 'cbi-value-field', 'id': 'utils', 'style': 'color:#37c' },'-')
244 E('div', { 'class': 'cbi-value' }, [
245 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Run Interfaces')),
246 E('div', { 'class': 'cbi-value-field', 'id': 'ifaces', 'style': 'color:#37c' },'-')
248 E('div', { 'class': 'cbi-value' }, [
249 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Run Directories')),
250 E('div', { 'class': 'cbi-value-field', 'id': 'dirs', 'style': 'color:#37c' },'-')
252 E('div', { 'class': 'cbi-value' }, [
253 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Run Flags')),
254 E('div', { 'class': 'cbi-value-field', 'id': 'flags', 'style': 'color:#37c' },'-')
256 E('div', { 'class': 'cbi-value' }, [
257 E('label', { 'class': 'cbi-value-title', 'style': 'padding-top:0rem' }, _('Last Run')),
258 E('div', { 'class': 'cbi-value-field', 'id': 'run', 'style': 'color:#37c' },'-')
260 E('div', { class: 'right' }, [
262 'class': 'btn cbi-button cbi-button-apply',
263 'click': ui
.createHandlerFn(this, function() {
264 return handleAction('timer');
266 }, [ _('Refresh Timer...') ]),
269 'class': 'btn cbi-button cbi-button-apply',
271 'click': ui
.createHandlerFn(this, function() {
272 return handleAction('suspend');
274 }, [ _('Suspend') ]),
277 'class': 'btn cbi-button cbi-button-positive',
278 'click': ui
.createHandlerFn(this, function() {
279 return handleAction('reload');
284 'class': 'btn cbi-button cbi-button-negative',
285 'click': ui
.createHandlerFn(this, function() {
286 return handleAction('restart');
295 tabbed config section
297 s
= m
.section(form
.NamedSection
, 'global', 'adblock', _('Settings'));
299 s
.tab('general', _('General Settings'));
300 s
.tab('additional', _('Additional Settings'));
301 s
.tab('adv_dns', _('Advanced DNS Settings'));
302 s
.tab('adv_report', _('Advanced Report Settings'));
303 s
.tab('adv_email', _('Advanced E-Mail Settings'));
304 s
.tab('sources', _('Blocklist Sources'));
309 o
= s
.taboption('general', form
.Flag
, 'adb_enabled', _('Enabled'), _('Enable the adblock service.'));
312 o
= s
.taboption('general', widgets
.NetworkSelect
, 'adb_trigger', _('Startup Trigger Interface'), _('List of available network interfaces to trigger the adblock start. \
313 Choose \'unspecified\' to use a classic startup timeout instead of a network trigger.'));
314 o
.unspecified
= true;
318 o
= s
.taboption('general', form
.Flag
, 'adb_forcedns', _('Force Local DNS'), _('Redirect all DNS queries from specified zones to the local DNS resolver, applies to UDP and TCP protocol.'));
321 o
= s
.taboption('general', widgets
.ZoneSelect
, 'adb_zonelist', _('Forced Zones'), _('Firewall source zones that should be forced locally.'));
322 o
.depends('adb_forcedns', '1');
323 o
.unspecified
= true;
328 o
= s
.taboption('general', form
.DynamicList
, 'adb_portlist', _('Forced Ports'), _('Firewall ports that should be forced locally.'));
329 o
.depends('adb_forcedns', '1');
330 o
.unspecified
= true;
339 o
= s
.taboption('general', form
.Flag
, 'adb_safesearch', _('Enable SafeSearch'), _('Enforcing SafeSearch for google, bing, duckduckgo, yandex, youtube and pixabay.'));
342 o
= s
.taboption('general', form
.MultiValue
, 'adb_safesearchlist', _('Limit SafeSearch'), _('Limit SafeSearch to certain providers.'));
343 o
.depends('adb_safesearch', '1');
351 o
= s
.taboption('general', form
.Flag
, 'adb_safesearchmod', _('Relax SafeSearch'), _('Enable moderate SafeSearch filters for youtube.'));
352 o
.depends('adb_safesearch', '1');
355 o
= s
.taboption('general', form
.Flag
, 'adb_report', _('DNS Report'), _('Gather DNS related network traffic via tcpdump and provide a DNS Report on demand. \
356 Please note: this needs additional \'tcpdump-mini\' package installation and a full adblock service restart to take effect.'));
359 o
= s
.taboption('general', form
.Flag
, 'adb_mail', _('E-Mail Notification'), _('Send adblock related notification e-mails. \
360 Please note: this needs additional \'msmtp\' package installation.'));
363 o
= s
.taboption('general', form
.Value
, 'adb_mailreceiver', _('E-Mail Receiver Address'), _('Receiver address for adblock notification e-mails.'));
364 o
.depends('adb_mail', '1');
365 o
.placeholder
= 'name@example.com';
369 additional settings tab
371 o
= s
.taboption('additional', form
.Flag
, 'adb_debug', _('Verbose Debug Logging'), _('Enable verbose debug logging in case of any processing errors.'));
374 o
= s
.taboption('additional', form
.Flag
, 'adb_nice', _('Low Priority Service'), _('Reduce the priority of the adblock background processing to take fewer resources from the system. \
375 Please note: This change requires a full adblock service restart to take effect.'));
379 o
= s
.taboption('additional', form
.Value
, 'adb_triggerdelay', _('Trigger Delay'), _('Additional trigger delay in seconds before adblock processing begins.'));
381 o
.datatype
= 'range(1,300)';
384 o
= s
.taboption('additional', form
.ListValue
, 'adb_maxqueue', _('Download Queue'), _('Size of the download queue for download processing (incl. sorting, merging etc.) in parallel.'));
393 o
= s
.taboption('additional', form
.Value
, 'adb_tmpbase', _('Base Temp Directory'), _('Base Temp Directory for all adblock related runtime operations, \
394 e.g. downloading, sorting, merging etc.'));
395 o
.placeholder
= '/tmp';
398 o
= s
.taboption('additional', form
.Flag
, 'adb_backup', _('Blocklist Backup'), _('Create compressed blocklist backups, they will be used in case of download errors or during startup.'));
402 o
= s
.taboption('additional', form
.Value
, 'adb_backupdir', _('Backup Directory'), _('Target directory for blocklist backups.'));
403 o
.depends('adb_backup', '1');
404 o
.placeholder
= '/tmp/adblock-Backup';
407 o
= s
.taboption('additional', form
.ListValue
, 'adb_fetchutil', _('Download Utility'), _('List of supported and fully pre-configured download utilities.'));
408 o
.value('uclient-fetch');
415 o
= s
.taboption('additional', form
.Flag
, 'adb_fetchinsecure', _('Download Insecure'), _('Don\'t check SSL server certificates during download.'));
419 o
= s
.taboption('additional', form
.Value
, 'adb_fetchparm', _('Download Parameters'), _('Manually override the pre-configured download options for the selected download utility.'));
424 advanced dns settings tab
426 o
= s
.taboption('adv_dns', form
.ListValue
, 'adb_dns', _('DNS Backend'), _('List of supported DNS backends with their default list directory. \
427 To overwrite the default path use the \'DNS Directory\' option.'));
428 o
.value('dnsmasq', _('dnsmasq (/tmp/dnsmasq.d)'));
429 o
.value('unbound', _('unbound (/var/lib/unbound)'));
430 o
.value('named', _('bind (/var/lib/bind)'));
431 o
.value('kresd', _('kresd (/etc/kresd)'));
432 o
.value('raw', _('raw (/tmp)'));
436 o
= s
.taboption('adv_dns', form
.Value
, 'adb_dnsdir', _('DNS Directory'), _('Target directory for the generated blocklist \'adb_list.overall\'.'));
437 o
.placeholder
= '/tmp';
440 o
= s
.taboption('adv_dns', form
.Value
, 'adb_dnstimeout', _('DNS Restart Timeout'), _('Timeout to wait for a successful DNS backend restart.'));
441 o
.placeholder
= '20';
442 o
.datatype
= 'range(1,60)';
445 o
= s
.taboption('adv_dns', form
.Value
, 'adb_lookupdomain', _('External DNS Lookup Domain'), _('External domain to check for a successful DNS backend restart. \
446 Please note: To disable this check set this option to \'false\'.'));
447 o
.placeholder
= 'example.com';
450 o
= s
.taboption('adv_dns', form
.Flag
, 'adb_dnsflush', _('Flush DNS Cache'), _('Flush the DNS Cache before adblock processing as well.'));
453 o
= s
.taboption('adv_dns', form
.Flag
, 'adb_dnsallow', _('Disable DNS Allow'), _('Disable selective DNS whitelisting (RPZ-PASSTHRU).'));
456 o
= s
.taboption('adv_dns', form
.DynamicList
, 'adb_denyip', _('Block Local Client IPs'), _('Block all requests of certain DNS clients based on their IP address (RPZ-CLIENT-IP). \
457 Please note: This feature is currently only supported by bind DNS backend.'));
458 o
.datatype
= 'or(ip4addr("nomask"),ip6addr("nomask"))';
462 o
= s
.taboption('adv_dns', form
.DynamicList
, 'adb_allowip', _('Allow Local Client IPs'), _('Allow all requests of certain DNS clients based on their IP address (RPZ-CLIENT-IP). \
463 Please note: This feature is currently only supported by bind DNS backend.'));
464 o
.datatype
= 'or(ip4addr("nomask"),ip6addr("nomask"))';
468 o
= s
.taboption('adv_dns', form
.Flag
, 'adb_jail', _('Additional Jail Blocklist'), _('Builds an additional DNS blocklist to block access to all domains except those listed in the whitelist. \
469 Please note: You can use this restrictive blocklist e.g. for guest wifi or kidsafe configurations.'));
472 o
= s
.taboption('adv_dns', form
.Value
, 'adb_jaildir', _('Jail Directory'), _('Target directory for the generated jail blocklist \'adb_list.jail\'.'));
473 o
.depends('adb_jail', '1');
474 o
.placeholder
= '/tmp';
477 o
= s
.taboption('adv_dns', form
.Flag
, 'adb_dnsinotify', _('Disable DNS Restarts'), _('Disable adblock triggered restarts for dns backends with autoload/inotify functions.'));
478 o
.depends('adb_dnsflush', '0');
482 advanced report settings tab
484 o
= s
.taboption('adv_report', form
.DummyValue
, '_sub');
486 o
.default = '<em><b>Changes on this tab needs a full adblock service restart to take effect.</b></em>';
488 o
= s
.taboption('adv_report', widgets
.DeviceSelect
, 'adb_repiface', _('Report Interface'), _('List of available network devices used by tcpdump.'));
489 o
.unspecified
= true;
493 o
= s
.taboption('adv_report', form
.Value
, 'adb_reportdir', _('Report Directory'), _('Target directory for DNS related report files.'));
494 o
.placeholder
= '/tmp/adblock-Report';
497 o
= s
.taboption('adv_report', form
.Value
, 'adb_repchunkcnt', _('Report Chunk Count'), _('Report chunk count used by tcpdump.'));
499 o
.datatype
= 'range(1,10)';
502 o
= s
.taboption('adv_report', form
.Value
, 'adb_repchunksize', _('Report Chunk Size'), _('Report chunk size used by tcpdump in MByte.'));
504 o
.datatype
= 'range(1,10)';
507 o
= s
.taboption('adv_report', form
.Value
, 'adb_replisten', _('Report Ports'), _('Space separated list of ports used by tcpdump.'));
508 o
.placeholder
= '53';
511 o
= s
.taboption('adv_report', form
.Flag
, 'adb_represolve', _('Resolve IPs'), _('Resolve reporting IP addresses by using reverse DNS (PTR) lookups.'));
515 advanced email settings tab
517 o
= s
.taboption('adv_email', form
.Value
, 'adb_mailsender', _('E-Mail Sender Address'), _('Sender address for adblock notification E-Mails.'));
518 o
.placeholder
= 'no-reply@adblock';
521 o
= s
.taboption('adv_email', form
.Value
, 'adb_mailtopic', _('E-Mail Topic'), _('Topic for adblock notification E-Mails.'));
522 o
.placeholder
= 'adblock notification';
525 o
= s
.taboption('adv_email', form
.Value
, 'adb_mailprofile', _('E-Mail Profile'), _('Profile used by \'msmtp\' for adblock notification E-Mails.'));
526 o
.placeholder
= 'adb_notify';
529 o
= s
.taboption('adv_email', form
.Value
, 'adb_mailcnt', _('E-Mail Notification Count'), _('Raise the notification count, to get E-Mails if the overall blocklist count is less or equal to the given limit.'));
531 o
.datatype
= 'min(0)';
535 blocklist sources tab
537 o
= s
.taboption('sources', form
.DummyValue
, '_sub');
539 o
.default = '<em><b>List of supported and fully pre-configured adblock sources.</b></em><br /> \
540 List size information with the respective domain ranges as follows:<br /> \
541 • <b>S</b> (-10k), <b>M</b> (10k-30k) and <b>L</b> (30k-80k) should work for 128 MByte devices,<br /> \
542 • <b>XL</b> (80k-200k) should work for 256-512 MByte devices,<br /> \
543 • <b>XXL</b> (200k-) needs more RAM and Multicore support, e.g. x86 or raspberry devices.<br /> \
544 • <b>VAR</b> (50k-500k) variable size depending on the selection.<br />';
546 var name
, size
, focus
, sources
= [];
548 sources
= result
[0].trim().split('\n');
551 o
= s
.taboption('sources', form
.MultiValue
, 'adb_sources', _('Sources (Size, Focus)'));
552 for (var i
= 0; i
< sources
.length
; i
++) {
553 if (sources
[i
].match(/^\s+\+/)) {
554 name
= sources
[i
].match(/^\s+\+\s(\w+)\s/)[1] || '-';
555 size
= sources
[i
].match(/^\s+\+\s\w+[\sx]+(\w+)/)[1] || '-';
556 focus
= sources
[i
].match(/^\s+\+\s\w+[\sx]+\w+\s+([\w\+]+)/)[1] || '-';
557 o
.value(name
, name
+ ' (' + size
+ ', ' + focus
+ ')');
564 prepare category data
566 var code
, category
, list
, path
, categories
= [];
568 categories
= result
[1].trim().split('\n');
571 o
= s
.taboption('sources', form
.DummyValue
, '_sub');
573 o
.default = '<em><b>Shallalist Archive Selection</b></em>';
575 o
= s
.taboption('sources', form
.DynamicList
, 'adb_sha_sources', _('Categories'));
576 for (var i
= 0; i
< categories
.length
; i
++) {
577 code
= categories
[i
].match(/^(\w+);/)[1].trim();
578 if (code
=== 'sha') {
579 category
= categories
[i
].match(/^\w+;(.*$)/)[1].trim();
586 o
= s
.taboption('sources', form
.DummyValue
, '_sub');
588 o
.default = '<em><b>UTCapitole Archive Selection</b></em>';
590 o
= s
.taboption('sources', form
.DynamicList
, 'adb_utc_sources', _('Categories'));
591 for (var i
= 0; i
< categories
.length
; i
++) {
592 code
= categories
[i
].match(/^(\w+);/)[1].trim();
593 if (code
=== 'utc') {
594 category
= categories
[i
].match(/^\w+;(.*$)/)[1].trim();
601 o
= s
.taboption('sources', form
.DummyValue
, '_sub');
603 o
.default = '<em><b>Energized List Selection</b></em>';
605 o
= s
.taboption('sources', form
.DynamicList
, 'adb_eng_sources', _('Variants'));
606 for (var i
= 0; i
< categories
.length
; i
++) {
607 code
= categories
[i
].match(/^(\w+);/)[1].trim();
608 if (code
=== 'eng') {
609 list
= categories
[i
].match(/^\w+;(.*);/)[1].trim();
610 path
= categories
[i
].match(/^.*;(.*$)/)[1].trim();
617 o
= s
.taboption('sources', form
.DummyValue
, '_sub');
619 o
.default = '<em><b>StevenBlack List Selection</b></em>';
621 o
= s
.taboption('sources', form
.DynamicList
, 'adb_stb_sources', _('Variants'));
622 for (var i
= 0; i
< categories
.length
; i
++) {
623 code
= categories
[i
].match(/^(\w+);/)[1].trim();
624 if (code
=== 'stb') {
625 list
= categories
[i
].match(/^\w+;(.*);/)[1].trim();
626 path
= categories
[i
].match(/^.*;(.*$)/)[1].trim();