secubox-openwrt/package/secubox/luci-app-bandwidth-manager/htdocs/luci-static/resources/view/bandwidth-manager/quotas.js
CyberMind-FR 31a87c5d7a feat(structure): reorganize luci-app packages into package/secubox/ + appstore migration
Major structural reorganization and feature additions:

## Folder Reorganization
- Move 17 luci-app-* packages to package/secubox/ (except luci-app-secubox core hub)
- Update all tooling to support new structure:
  - secubox-tools/quick-deploy.sh: search both locations
  - secubox-tools/validate-modules.sh: validate both directories
  - secubox-tools/fix-permissions.sh: fix permissions in both locations
  - .github/workflows/test-validate.yml: build from both paths
- Update README.md links to new package/secubox/ paths

## AppStore Migration (Complete)
- Add catalog entries for all remaining luci-app packages:
  - network-tweaks.json: Network optimization tools
  - secubox-bonus.json: Documentation & demos hub
- Total: 24 apps in AppStore catalog (22 existing + 2 new)
- New category: 'documentation' for docs/demos/tutorials

## VHost Manager v2.0 Enhancements
- Add profile activation system for Internal Services and Redirects
- Implement createVHost() API wrapper for template-based deployment
- Fix Virtual Hosts view rendering with proper LuCI patterns
- Fix RPCD backend shell script errors (remove invalid local declarations)
- Extend backend validation for nginx return directives (redirect support)
- Add section_id parameter for named VHost profiles
- Add Remove button to Redirects page for feature parity
- Update README to v2.0 with comprehensive feature documentation

## Network Tweaks Dashboard
- Close button added to component details modal

Files changed: 340+ (336 renames with preserved git history)
Packages affected: 19 luci-app, 2 secubox-app, 1 theme, 4 tools

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 14:59:38 +01:00

121 lines
3.5 KiB
JavaScript

'use strict';
'require view';
'require secubox-theme/theme as Theme';
'require ui';
'require form';
'require bandwidth-manager/api as API';
return L.view.extend({
load: function() {
return API.listQuotas();
},
render: function(quotas) {
var m = new form.Map('bandwidth', _('Client Quotas'),
_('Set monthly data quotas for individual clients by MAC address'));
var s = m.section(form.GridSection, 'quota', _('Quotas'));
s.anonymous = false;
s.addremove = true;
s.sortable = true;
s.modaltitle = function(section_id) {
return _('Edit Quota: ') + section_id;
};
// Custom render to show usage progress bars
s.addModalOptions = function(s, section_id, ev) {
var mac = this.section.formvalue(section_id, 'mac');
if (!mac) {
ui.addNotification(null, E('p', _('MAC address is required')), 'error');
return;
}
// Save quota via API
var name = this.section.formvalue(section_id, 'name') || '';
var limit_mb = parseInt(this.section.formvalue(section_id, 'limit_mb')) || 0;
var action = this.section.formvalue(section_id, 'action') || 'throttle';
var reset_day = parseInt(this.section.formvalue(section_id, 'reset_day')) || 1;
API.setQuota(mac, name, limit_mb, action, reset_day).then(function(result) {
if (result.success) {
ui.addNotification(null, E('p', '✓ ' + result.message), 'info');
window.location.reload();
} else {
ui.addNotification(null, E('p', '✗ ' + result.message), 'error');
}
});
};
var o;
o = s.option(form.Value, 'mac', _('MAC Address'));
o.rmempty = false;
o.datatype = 'macaddr';
o.placeholder = 'AA:BB:CC:DD:EE:FF';
o = s.option(form.Value, 'name', _('Client Name'));
o.placeholder = 'iPhone John';
o.description = _('Friendly name for this client');
o = s.option(form.Value, 'limit_mb', _('Monthly Limit (MB)'));
o.rmempty = false;
o.datatype = 'uinteger';
o.placeholder = '10240';
o.description = _('Monthly data limit in megabytes (e.g., 10240 = 10GB)');
o = s.option(form.ListValue, 'action', _('Action When Exceeded'));
o.value('throttle', _('Throttle bandwidth'));
o.value('block', _('Block all traffic'));
o.value('notify', _('Notify only'));
o.default = 'throttle';
o = s.option(form.Value, 'reset_day', _('Reset Day'));
o.datatype = 'range(1,28)';
o.default = '1';
o.description = _('Day of month to reset quota (1-28)');
o = s.option(form.Flag, 'enabled', _('Enabled'));
o.default = o.enabled;
// Show current usage
s.renderRowActions = function(section_id) {
var config_name = this.uciconfig || this.map.config;
var mac = this.cfgvalue(section_id, 'mac');
var resetBtn = E('button', {
'class': 'cbi-button cbi-button-action',
'click': function(ev) {
ev.preventDefault();
if (confirm(_('Reset quota counter for this client?'))) {
API.resetQuota(mac).then(function(result) {
if (result.success) {
ui.addNotification(null, E('p', '✓ ' + result.message), 'info');
window.location.reload();
} else {
ui.addNotification(null, E('p', '✗ ' + result.message), 'error');
}
});
}
}
}, _('Reset Counter'));
var actions = form.GridSection.prototype.renderRowActions.call(this, section_id);
actions.appendChild(resetBtn);
return actions;
};
return m.render().then(function(rendered) {
return E('div', {}, [
E('link', { 'rel': 'stylesheet', 'href': L.resource('secubox-theme/secubox-theme.css') }),
rendered
]);
});
},
handleSaveApply: null,
handleSave: null,
handleReset: null
});