Align loyalty pages across admin, merchant, and store personas so each sees the same page set scoped to their access level. Admin acts as a superset of merchant with "on behalf" capabilities. New pages: - Store: Staff PINs management (CRUD) - Merchant: Cards, Card Detail, Transactions, Staff PINs (CRUD), Settings (read-only) - Admin: Merchant Cards, Card Detail, Transactions, PINs (read-only) Architecture: - 4 shared Jinja2 partials (cards-list, card-detail, transactions, pins) - 4 shared JS factory modules parameterized by apiPrefix/scope - Persona templates are thin wrappers including shared partials - PinDetailResponse schema for cross-store PIN listings API: 17 new endpoints (11 merchant, 6 admin on-behalf) Tests: 38 new integration tests, arch-check green i18n: ~130 new keys across en/fr/de/lb Docs: pages-and-navigation.md with full page matrix Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
54 lines
2.0 KiB
JavaScript
54 lines
2.0 KiB
JavaScript
// app/modules/loyalty/static/merchant/js/loyalty-merchant-settings.js
|
|
// noqa: js-006 - async init pattern is safe, loadData has try/catch
|
|
|
|
const merchantSettingsViewLog = window.LogConfig.loggers.merchantSettingsView || window.LogConfig.createLogger('merchantSettingsView');
|
|
|
|
function merchantLoyaltyMerchantSettings() {
|
|
return {
|
|
...data(),
|
|
currentPage: 'loyalty-settings',
|
|
|
|
settings: null,
|
|
loading: false,
|
|
error: null,
|
|
|
|
async init() {
|
|
merchantSettingsViewLog.info('=== MERCHANT LOYALTY SETTINGS VIEW PAGE INITIALIZING ===');
|
|
if (window._merchantLoyaltyMerchantSettingsInitialized) return;
|
|
window._merchantLoyaltyMerchantSettingsInitialized = true;
|
|
|
|
this.loadMenuConfig();
|
|
await this.loadSettings();
|
|
merchantSettingsViewLog.info('=== MERCHANT LOYALTY SETTINGS VIEW PAGE INITIALIZATION COMPLETE ===');
|
|
},
|
|
|
|
async loadSettings() {
|
|
this.loading = true;
|
|
this.error = null;
|
|
|
|
try {
|
|
const response = await apiClient.get('/merchants/loyalty/settings');
|
|
if (response) {
|
|
this.settings = response;
|
|
merchantSettingsViewLog.info('Settings loaded');
|
|
}
|
|
} catch (error) {
|
|
if (error.status === 404) {
|
|
merchantSettingsViewLog.info('No settings found');
|
|
this.settings = null;
|
|
} else {
|
|
merchantSettingsViewLog.error('Failed to load settings:', error);
|
|
this.error = error.message || 'Failed to load settings';
|
|
}
|
|
} finally {
|
|
this.loading = false;
|
|
}
|
|
},
|
|
};
|
|
}
|
|
|
|
if (!window.LogConfig.loggers.merchantSettingsView) {
|
|
window.LogConfig.loggers.merchantSettingsView = window.LogConfig.createLogger('merchantSettingsView');
|
|
}
|
|
merchantSettingsViewLog.info('Merchant loyalty settings view module loaded');
|