mirror of
https://github.com/screentinker/screentinker.git
synced 2026-05-15 07:32:23 -06:00
- teams.js: list, detail with members + shared devices, invite/role controls, all toasts - activity.js: page chrome, action verb/noun mapping translated through t() so the audit log reads naturally in each language - help.js: page chrome translated; guides and FAQ body content kept in English with a comment explaining why (machine-translated docs read worse than English source) - 1008 keys total, parity 100% across en/es/fr/de/pt All 16 dashboard views now use t(). index.html modal, player overlay, and Android resources still pending. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
111 lines
4.6 KiB
JavaScript
111 lines
4.6 KiB
JavaScript
import { showToast } from '../components/toast.js';
|
|
import { esc } from '../utils.js';
|
|
import { t } from '../i18n.js';
|
|
|
|
const API = (url) => fetch('/api' + url, { headers: { Authorization: `Bearer ${localStorage.getItem('token')}` }}).then(r => r.json());
|
|
|
|
export async function render(container) {
|
|
container.innerHTML = `
|
|
<div class="page-header">
|
|
<div><h1>${t('activity.title')}</h1><div class="subtitle">${t('activity.subtitle')}</div></div>
|
|
</div>
|
|
<div id="activityList"><div class="empty-state"><h3>${t('common.loading')}</h3></div></div>
|
|
<div style="text-align:center;margin-top:16px">
|
|
<button class="btn btn-secondary btn-sm" id="loadMoreBtn" style="display:none">${t('activity.load_more')}</button>
|
|
</div>
|
|
`;
|
|
|
|
let offset = 0;
|
|
const limit = 50;
|
|
|
|
async function loadActivity(append = false) {
|
|
try {
|
|
const items = await API(`/activity?limit=${limit}&offset=${offset}`);
|
|
const list = document.getElementById('activityList');
|
|
|
|
if (!append) list.innerHTML = '';
|
|
|
|
if (items.length === 0 && offset === 0) {
|
|
list.innerHTML = `<div class="empty-state"><h3>${t('activity.empty_title')}</h3><p>${t('activity.empty_desc')}</p></div>`;
|
|
return;
|
|
}
|
|
|
|
const html = items.map(item => {
|
|
const time = new Date(item.created_at * 1000);
|
|
const timeStr = time.toLocaleDateString(undefined, { month: 'short', day: 'numeric' }) + ' ' +
|
|
time.toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit' });
|
|
const icon = getActionIcon(item.action);
|
|
|
|
return `
|
|
<div style="display:flex;gap:12px;padding:12px 0;border-bottom:1px solid var(--border);align-items:flex-start">
|
|
<div style="width:32px;height:32px;border-radius:50%;background:var(--bg-card);display:flex;align-items:center;justify-content:center;flex-shrink:0;font-size:14px">${icon}</div>
|
|
<div style="flex:1;min-width:0">
|
|
<div style="font-size:13px">
|
|
<strong>${esc(item.user_name || item.user_email || t('activity.system'))}</strong>
|
|
<span style="color:var(--text-secondary)"> ${esc(formatAction(item.action))}</span>
|
|
</div>
|
|
${item.details ? `<div style="font-size:12px;color:var(--text-muted);margin-top:2px">${esc(item.details)}</div>` : ''}
|
|
</div>
|
|
<div style="font-size:11px;color:var(--text-muted);white-space:nowrap;flex-shrink:0">${timeStr}</div>
|
|
</div>
|
|
`;
|
|
}).join('');
|
|
|
|
if (append) {
|
|
list.insertAdjacentHTML('beforeend', html);
|
|
} else {
|
|
list.innerHTML = html;
|
|
}
|
|
|
|
document.getElementById('loadMoreBtn').style.display = items.length >= limit ? '' : 'none';
|
|
} catch (err) {
|
|
showToast(err.message, 'error');
|
|
}
|
|
}
|
|
|
|
document.getElementById('loadMoreBtn').onclick = () => {
|
|
offset += limit;
|
|
loadActivity(true);
|
|
};
|
|
|
|
loadActivity();
|
|
}
|
|
|
|
function getActionIcon(action) {
|
|
if (action.includes('DELETE')) return '🗑';
|
|
if (action.includes('POST') && action.includes('content')) return '📤';
|
|
if (action.includes('POST') && action.includes('provision')) return '🔗';
|
|
if (action.includes('POST') && action.includes('assignment')) return '📋';
|
|
if (action.includes('alert')) return '🔔';
|
|
if (action.includes('PUT')) return '✎';
|
|
if (action.includes('POST')) return '➕';
|
|
return '📄';
|
|
}
|
|
|
|
// Action verbs are user-visible; translate them through t() so they switch
|
|
// languages with the rest of the UI. The mapping below preserves the original
|
|
// verb-then-noun structure of the English version.
|
|
function formatAction(action) {
|
|
// Verbs
|
|
let s = action
|
|
.replace('POST /api/', t('activity.verb_created') + ' ')
|
|
.replace('PUT /api/', t('activity.verb_updated') + ' ')
|
|
.replace('DELETE /api/', t('activity.verb_deleted') + ' ');
|
|
// Specific endpoints
|
|
s = s
|
|
.replace('/provision/pair', t('activity.action_paired_device'))
|
|
.replace('/content/remote', t('activity.action_added_remote_content'))
|
|
.replace('/content', t('activity.noun_content'))
|
|
.replace('/devices/:id', t('activity.noun_device'))
|
|
.replace('/assignments/device/:deviceId', t('activity.noun_playlist_assignment'))
|
|
.replace('/assignments/:id', t('activity.noun_assignment'))
|
|
.replace('/layouts', t('activity.noun_layout'))
|
|
.replace('/widgets', t('activity.noun_widget'))
|
|
.replace('/schedules', t('activity.noun_schedule'))
|
|
.replace('/walls', t('activity.noun_video_wall'))
|
|
.replace('alert:device_offline', t('activity.alert_device_offline'));
|
|
return s;
|
|
}
|
|
|
|
export function cleanup() {}
|