ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
// modal-new-recorder.jsx - New Recorder dialog (SRT / RTMP / SDI / Deltacast)
|
2026-05-28 19:18:55 -04:00
|
|
|
|
|
|
|
|
|
|
/**
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
* DevicePortPicker - groups a flat per-port API response by node_id and
|
2026-05-28 19:18:55 -04:00
|
|
|
|
* renders one button per actual port. Replaces the old code that iterated
|
|
|
|
|
|
* over entries and synthesised port counts, which caused duplicate groups.
|
|
|
|
|
|
*
|
|
|
|
|
|
* props:
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
* ports - flat array from /cluster/devices/blackmagic or /deltacast
|
2026-05-28 19:18:55 -04:00
|
|
|
|
* each entry: { node_id, hostname, model, index, device, present? }
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
* selectedIdx - currently selected device_index
|
|
|
|
|
|
* selectedNode - currently selected node_id
|
2026-05-28 19:18:55 -04:00
|
|
|
|
* onSelect(idx, nodeId)
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
* portLabel - e.g. "SDI" or "Port"
|
|
|
|
|
|
* showTestBadge - show TEST CARD badge when present===false
|
2026-05-28 19:18:55 -04:00
|
|
|
|
*/
|
|
|
|
|
|
function DevicePortPicker({ ports, selectedIdx, selectedNode, onSelect, portLabel = 'Port', showTestBadge = false }) {
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
// Group by node_id (stable - one group per physical node)
|
2026-05-28 19:18:55 -04:00
|
|
|
|
const groups = React.useMemo(() => {
|
|
|
|
|
|
const map = new Map();
|
|
|
|
|
|
for (const p of ports) {
|
|
|
|
|
|
const key = p.node_id || p.hostname || 'unknown';
|
|
|
|
|
|
if (!map.has(key)) map.set(key, { nodeId: p.node_id || p.hostname || '', hostname: p.hostname || key, model: p.model || '', ports: [] });
|
|
|
|
|
|
map.get(key).ports.push(p);
|
|
|
|
|
|
}
|
|
|
|
|
|
// Sort ports within each group by index
|
|
|
|
|
|
for (const g of map.values()) g.ports.sort((a, b) => a.index - b.index);
|
|
|
|
|
|
return Array.from(map.values());
|
|
|
|
|
|
}, [ports]);
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div className="sdi-port-mini">
|
|
|
|
|
|
{groups.map(group => (
|
|
|
|
|
|
<div key={group.nodeId} style={{ marginBottom: groups.length > 1 ? 12 : 4 }}>
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
{/* Node header: only show when multiple groups, or always for clarity */}
|
2026-05-28 19:18:55 -04:00
|
|
|
|
<div style={{ fontSize: 10, fontWeight: 700, letterSpacing: '0.08em', color: 'var(--text-3)', textTransform: 'uppercase', padding: '0 0 6px' }}>
|
|
|
|
|
|
{group.model ? group.model.toUpperCase() + ' · ' : ''}{group.hostname}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
{group.ports.map(port => {
|
|
|
|
|
|
const active = selectedIdx === port.index && selectedNode === group.nodeId;
|
|
|
|
|
|
return (
|
|
|
|
|
|
<button key={port.index}
|
|
|
|
|
|
className={`sdi-mini-port${active ? ' active' : ''}`}
|
|
|
|
|
|
onClick={() => onSelect(port.index, group.nodeId)}>
|
|
|
|
|
|
<span className="sdi-mini-radio"><span className="sdi-mini-radio-dot" /></span>
|
|
|
|
|
|
<span style={{ fontSize: 12, fontWeight: 600 }}>
|
|
|
|
|
|
{portLabel} {port.index + 1}
|
|
|
|
|
|
{showTestBadge && port.present === false && (
|
|
|
|
|
|
<span style={{ fontSize: 9, fontWeight: 700, letterSpacing: '0.06em', color: 'var(--accent)', background: 'var(--accent-soft)', borderRadius: 3, padding: '1px 5px', marginLeft: 7 }}>
|
|
|
|
|
|
TEST CARD
|
|
|
|
|
|
</span>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
<span style={{ fontSize: 11, color: 'var(--text-3)', marginLeft: 'auto', fontFamily: 'var(--font-mono)' }}>
|
|
|
|
|
|
{port.device ? port.device.split('/').pop() : `index ${port.index}`}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</button>
|
|
|
|
|
|
);
|
|
|
|
|
|
})}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
* ManualDevicePicker - fallback when no devices detected. Lets the operator
|
2026-05-28 19:18:55 -04:00
|
|
|
|
* pick node + index from dropdowns.
|
|
|
|
|
|
*/
|
|
|
|
|
|
function ManualDevicePicker({ nodes, nodeId, deviceIdx, portLabel, portCount, onNodeChange, onIdxChange, emptyNote }) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div style={{ padding: '4px 0' }}>
|
|
|
|
|
|
<div style={{ fontSize: 12, color: 'var(--text-3)', marginBottom: 10 }}>
|
|
|
|
|
|
{emptyNote || `No ${portLabel} devices auto-detected. Configure manually:`}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div style={{ display: 'grid', gridTemplateColumns: '1fr 1fr', gap: 10 }}>
|
|
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Capture node</label>
|
|
|
|
|
|
<select className="field-input" value={nodeId}
|
|
|
|
|
|
onChange={e => onNodeChange(e.target.value)} style={{ appearance: 'auto' }}>
|
|
|
|
|
|
{nodes.length === 0
|
|
|
|
|
|
? <option value="">No cluster nodes</option>
|
|
|
|
|
|
: nodes.map(n => {
|
|
|
|
|
|
const id = n.id || n.hostname || n.name || '';
|
|
|
|
|
|
return <option key={id} value={id}>{n.hostname || n.name || id}</option>;
|
|
|
|
|
|
})}
|
|
|
|
|
|
</select>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">{portLabel} index</label>
|
|
|
|
|
|
<select className="field-input" value={deviceIdx}
|
|
|
|
|
|
onChange={e => onIdxChange(Number(e.target.value))} style={{ appearance: 'auto' }}>
|
|
|
|
|
|
{Array.from({ length: portCount }, (_, i) =>
|
|
|
|
|
|
<option key={i} value={i}>{portLabel} {i + 1} (index {i})</option>)}
|
|
|
|
|
|
</select>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
2026-05-22 08:19:03 -04:00
|
|
|
|
|
2026-05-22 11:33:45 -04:00
|
|
|
|
function ProbeResult({ result }) {
|
|
|
|
|
|
if (!result.ok) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div style={{ marginTop: 6, padding: '8px 10px', background: 'var(--danger-soft)', border: '1px solid var(--danger)', borderRadius: 5, fontSize: 11.5, color: 'var(--danger)' }}>
|
|
|
|
|
|
Probe failed: {result.error}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
const d = result.data || {};
|
|
|
|
|
|
const entries = Object.entries(d).filter(([, v]) => v !== null && typeof v !== 'object');
|
|
|
|
|
|
if (entries.length === 0) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div style={{ marginTop: 6, padding: '8px 10px', background: 'var(--success-soft)', border: '1px solid var(--success)', borderRadius: 5, fontSize: 11.5, color: 'var(--success)' }}>
|
|
|
|
|
|
✓ Source reachable
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div style={{ marginTop: 6, padding: '10px 12px', background: 'var(--bg-2)', border: '1px solid var(--success)', borderRadius: 5, fontSize: 11.5 }}>
|
|
|
|
|
|
{entries.map(([k, v]) => (
|
|
|
|
|
|
<div key={k} style={{ display: 'flex', gap: 8, padding: '2px 0' }}>
|
|
|
|
|
|
<span style={{ color: 'var(--text-3)', minWidth: 100, flexShrink: 0 }}>{k}</span>
|
|
|
|
|
|
<span style={{ fontFamily: 'var(--font-mono)', fontSize: 11 }}>{String(v)}</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-22 08:19:03 -04:00
|
|
|
|
function NewRecorderModal({ open, onClose }) {
|
test: deploy/api-smoke.sh — exercises every API surface
Walks GET endpoints for auth, projects, assets, recorders, jobs, bins,
users, groups, cluster, settings, metrics, schedules, sdk, and the
freshly added comments routes. Deep-links one asset + one recorder by
ID so per-asset endpoints (stream, thumbnail, comments) get coverage.
Prints HTTP codes inline and exits non-zero on any failure. Treats
2xx/3xx as pass; 400/401 also pass since they indicate the route
exists and auth/validation is working as designed.
Usage:
deploy/api-smoke.sh # localhost:47432
API=http://10.0.0.25:47432 deploy/api-smoke.sh
NewRecorderModal: hardened ZAMPP_DATA hydration with defensive
defaults so first-load timing doesn't blow up the modal.
2026-05-23 00:24:10 -04:00
|
|
|
|
const PROJECTS = window.ZAMPP_DATA?.PROJECTS || [];
|
|
|
|
|
|
const NODES = window.ZAMPP_DATA?.NODES || [];
|
2026-05-22 10:55:22 -04:00
|
|
|
|
const [name, setName] = React.useState('');
|
|
|
|
|
|
const [sourceType, setSourceType] = React.useState('SRT');
|
|
|
|
|
|
const [srtUrl, setSrtUrl] = React.useState('srt://10.0.4.18:4200');
|
|
|
|
|
|
const [rtmpUrl, setRtmpUrl] = React.useState('rtmp://stream.local/live/cam_a');
|
2026-05-22 11:10:01 -04:00
|
|
|
|
const [sdiDeviceIdx, setSdiDeviceIdx] = React.useState(0);
|
|
|
|
|
|
const [sdiNodeId, setSdiNodeId] = React.useState(() => {
|
test: deploy/api-smoke.sh — exercises every API surface
Walks GET endpoints for auth, projects, assets, recorders, jobs, bins,
users, groups, cluster, settings, metrics, schedules, sdk, and the
freshly added comments routes. Deep-links one asset + one recorder by
ID so per-asset endpoints (stream, thumbnail, comments) get coverage.
Prints HTTP codes inline and exits non-zero on any failure. Treats
2xx/3xx as pass; 400/401 also pass since they indicate the route
exists and auth/validation is working as designed.
Usage:
deploy/api-smoke.sh # localhost:47432
API=http://10.0.0.25:47432 deploy/api-smoke.sh
NewRecorderModal: hardened ZAMPP_DATA hydration with defensive
defaults so first-load timing doesn't blow up the modal.
2026-05-23 00:24:10 -04:00
|
|
|
|
const n = NODES[0];
|
2026-05-22 11:10:01 -04:00
|
|
|
|
return n ? (n.id || n.hostname || '') : '';
|
|
|
|
|
|
});
|
2026-05-22 10:55:22 -04:00
|
|
|
|
const [sdiDevices, setSdiDevices] = React.useState(null);
|
2026-05-28 19:12:40 -04:00
|
|
|
|
const [dcDeviceIdx, setDcDeviceIdx] = React.useState(0);
|
|
|
|
|
|
const [dcNodeId, setDcNodeId] = React.useState(() => {
|
|
|
|
|
|
const n = NODES[0];
|
|
|
|
|
|
return n ? (n.id || n.hostname || '') : '';
|
|
|
|
|
|
});
|
|
|
|
|
|
const [dcDevices, setDcDevices] = React.useState(null);
|
2026-05-22 10:55:22 -04:00
|
|
|
|
const [recTab, setRecTab] = React.useState('video');
|
2026-05-29 17:04:00 -04:00
|
|
|
|
// All-Intra HEVC (NVENC) is the default master — GPU-encoded, growing-file
|
|
|
|
|
|
// capable in fragmented MOV. ProRes stays selectable for 4:2:2 mezzanine.
|
|
|
|
|
|
const [recCodec, setRecCodec] = React.useState('hevc_nvenc');
|
|
|
|
|
|
// Custom target bitrate in Mbps for bitrate-controlled codecs (NVENC / x264 /
|
|
|
|
|
|
// x265 / DNxHD). ProRes ignores bitrate (quality is profile-driven).
|
2026-06-01 18:41:36 -04:00
|
|
|
|
const [recBitrate, setRecBitrate] = React.useState('25');
|
2026-05-29 17:04:00 -04:00
|
|
|
|
// Container is derived from the codec, not manually chosen: HEVC/ProRes/DNxHR
|
|
|
|
|
|
// → MOV (fragmented, growing-capable); H.264 → MP4.
|
|
|
|
|
|
const recContainer = (recCodec === 'h264' || recCodec === 'h264_nvenc' || recCodec === 'libx264') ? 'mp4' : 'mov';
|
|
|
|
|
|
// Codecs whose bitrate is operator-controlled (everything except ProRes).
|
|
|
|
|
|
const BITRATE_CODECS = new Set(['hevc_nvenc', 'h264_nvenc', 'libx264', 'libx265', 'dnxhd', 'dnxhr_hq']);
|
|
|
|
|
|
const codecUsesBitrate = BITRATE_CODECS.has(recCodec);
|
2026-05-22 08:19:03 -04:00
|
|
|
|
const [proxyOn, setProxyOn] = React.useState(true);
|
2026-05-31 14:50:31 -04:00
|
|
|
|
const [growingOn, setGrowingOn] = React.useState(false);
|
2026-05-31 22:14:52 -04:00
|
|
|
|
// Growing-files mode forces the master to XDCAM HD422 / MXF OP1a in the capture
|
2026-05-31 22:13:01 -04:00
|
|
|
|
// backend (the only growing format Premiere can import live), but the target
|
|
|
|
|
|
// bitrate is still operator-controlled and applied via -b:v. Keep the bitrate
|
|
|
|
|
|
// input visible/editable whenever growing is on, even if the selected (and
|
|
|
|
|
|
// soon-to-be-overridden) codec would normally be quality-driven (ProRes).
|
|
|
|
|
|
const showBitrate = codecUsesBitrate || growingOn;
|
2026-05-22 10:55:22 -04:00
|
|
|
|
const [projectId, setProjectId] = React.useState(PROJECTS[0]?.id || '');
|
|
|
|
|
|
const [submitting, setSubmitting] = React.useState(false);
|
|
|
|
|
|
const [submitErr, setSubmitErr] = React.useState(null);
|
2026-05-22 11:33:45 -04:00
|
|
|
|
const [probing, setProbing] = React.useState(false);
|
|
|
|
|
|
const [probeResult, setProbeResult] = React.useState(null);
|
2026-05-22 10:55:22 -04:00
|
|
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
|
|
if (sourceType !== 'SDI' || sdiDevices !== null) return;
|
|
|
|
|
|
window.ZAMPP_API.fetch('/cluster/devices/blackmagic')
|
|
|
|
|
|
.then(d => setSdiDevices(Array.isArray(d) ? d : []))
|
|
|
|
|
|
.catch(() => setSdiDevices([]));
|
|
|
|
|
|
}, [sourceType]);
|
|
|
|
|
|
|
2026-05-28 19:12:40 -04:00
|
|
|
|
React.useEffect(() => {
|
|
|
|
|
|
if (sourceType !== 'DELTACAST' || dcDevices !== null) return;
|
|
|
|
|
|
window.ZAMPP_API.fetch('/cluster/devices/deltacast')
|
|
|
|
|
|
.then(d => setDcDevices(Array.isArray(d) ? d : []))
|
|
|
|
|
|
.catch(() => setDcDevices([]));
|
|
|
|
|
|
}, [sourceType]);
|
|
|
|
|
|
|
2026-05-22 11:33:45 -04:00
|
|
|
|
React.useEffect(() => { setProbeResult(null); }, [sourceType, srtUrl, rtmpUrl]);
|
|
|
|
|
|
|
|
|
|
|
|
const handleProbe = () => {
|
|
|
|
|
|
setProbing(true);
|
|
|
|
|
|
setProbeResult(null);
|
|
|
|
|
|
const body = sourceType === 'SRT'
|
|
|
|
|
|
? { source_type: 'srt', url: srtUrl }
|
|
|
|
|
|
: { source_type: 'rtmp', url: rtmpUrl };
|
|
|
|
|
|
window.ZAMPP_API.fetch('/recorders/probe', { method: 'POST', body: JSON.stringify(body) })
|
|
|
|
|
|
.then(r => { setProbing(false); setProbeResult({ ok: true, data: r }); })
|
|
|
|
|
|
.catch(e => { setProbing(false); setProbeResult({ ok: false, error: e.message }); });
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2026-05-22 10:55:22 -04:00
|
|
|
|
const handleCreate = () => {
|
|
|
|
|
|
if (!name.trim()) { setSubmitErr('Recorder name is required.'); return; }
|
2026-05-22 11:10:01 -04:00
|
|
|
|
if (sourceType === 'SDI' && !sdiNodeId) { setSubmitErr('Select a capture node for SDI.'); return; }
|
2026-05-28 19:12:40 -04:00
|
|
|
|
if (sourceType === 'DELTACAST' && !dcNodeId) { setSubmitErr('Select a capture node for Deltacast.'); return; }
|
2026-05-22 10:55:22 -04:00
|
|
|
|
setSubmitting(true);
|
|
|
|
|
|
setSubmitErr(null);
|
2026-05-22 11:10:01 -04:00
|
|
|
|
|
2026-05-22 10:55:22 -04:00
|
|
|
|
const body = {
|
|
|
|
|
|
name: name.trim(),
|
|
|
|
|
|
source_type: sourceType.toLowerCase(),
|
|
|
|
|
|
project_id: projectId || undefined,
|
|
|
|
|
|
generate_proxy: proxyOn,
|
2026-05-31 14:50:31 -04:00
|
|
|
|
growing_enabled: growingOn,
|
2026-05-22 17:20:01 -04:00
|
|
|
|
recording_codec: recCodec,
|
|
|
|
|
|
recording_container: recContainer,
|
2026-05-29 17:04:00 -04:00
|
|
|
|
// Framerate + resolution are auto-detected from the source signal/stream.
|
|
|
|
|
|
recording_framerate: '', // empty = match source
|
|
|
|
|
|
recording_resolution: 'native',
|
2026-05-22 10:55:22 -04:00
|
|
|
|
};
|
2026-05-31 22:13:01 -04:00
|
|
|
|
// Custom bitrate applies to bitrate-controlled codecs AND to growing-files
|
|
|
|
|
|
// mode (which forces H.264/TS in capture but still honors -b:v). ProRes
|
|
|
|
|
|
// without growing ignores bitrate, so we omit it there.
|
|
|
|
|
|
if ((codecUsesBitrate || growingOn) && recBitrate) {
|
2026-05-29 17:04:00 -04:00
|
|
|
|
body.recording_video_bitrate = `${recBitrate}M`;
|
|
|
|
|
|
}
|
2026-05-22 11:10:01 -04:00
|
|
|
|
|
|
|
|
|
|
if (sourceType === 'SRT') {
|
|
|
|
|
|
body.source_config = { url: srtUrl };
|
|
|
|
|
|
} else if (sourceType === 'RTMP') {
|
|
|
|
|
|
body.source_config = { url: rtmpUrl };
|
2026-05-28 19:12:40 -04:00
|
|
|
|
} else if (sourceType === 'DELTACAST') {
|
2026-06-01 15:00:48 -04:00
|
|
|
|
// One Deltacast board (index 0) exposes 8 channels. The picker's selected
|
|
|
|
|
|
// index IS the capture channel, so persist it as source_config.port; the
|
|
|
|
|
|
// capture sidecar maps that to the bridge's --port. device_index is kept
|
|
|
|
|
|
// for backward-compatible display/fallback.
|
|
|
|
|
|
body.source_config = { port: dcDeviceIdx };
|
2026-05-28 19:12:40 -04:00
|
|
|
|
body.device_index = dcDeviceIdx;
|
|
|
|
|
|
body.node_id = dcNodeId || undefined;
|
2026-05-22 11:10:01 -04:00
|
|
|
|
} else {
|
2026-05-28 19:12:40 -04:00
|
|
|
|
// SDI (DeckLink): device_index and node_id are top-level fields
|
2026-05-22 11:10:01 -04:00
|
|
|
|
body.source_config = {};
|
|
|
|
|
|
body.device_index = sdiDeviceIdx;
|
|
|
|
|
|
body.node_id = sdiNodeId || undefined;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-22 10:55:22 -04:00
|
|
|
|
window.ZAMPP_API.fetch('/recorders', { method: 'POST', body: JSON.stringify(body) })
|
2026-05-23 14:52:04 -04:00
|
|
|
|
.then(() => {
|
|
|
|
|
|
setSubmitting(false);
|
|
|
|
|
|
// Recorders list listens for this and re-fetches; otherwise the
|
|
|
|
|
|
// operator has to wait for the next 10s poll tick to see the new row.
|
|
|
|
|
|
window.dispatchEvent(new CustomEvent('df:recorders-changed'));
|
|
|
|
|
|
onClose();
|
|
|
|
|
|
})
|
2026-05-22 10:55:22 -04:00
|
|
|
|
.catch(e => { setSubmitting(false); setSubmitErr(e.message || 'Failed to create recorder'); });
|
|
|
|
|
|
};
|
2026-05-22 08:19:03 -04:00
|
|
|
|
|
|
|
|
|
|
if (!open) return null;
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div className="modal-backdrop" onClick={onClose}>
|
|
|
|
|
|
<div className="modal" onClick={e => e.stopPropagation()}>
|
|
|
|
|
|
<div className="modal-head">
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<div style={{ fontSize: 15, fontWeight: 600 }}>New recorder</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div style={{ fontSize: 12, color: 'var(--text-3)', marginTop: 2 }}>Configure source, codec, and destination</div>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
chore: 1.2 ship-prep sweep — close 38 issues
Frontend / UX / a11y
- Sidebar collapse/expand toggle with localStorage persistence (#142)
- Settings sections wrap inputs in <form> with Enter-to-submit + native
validation; password autocomplete=new-password (#141, #138)
- Asset thumbnails get descriptive alt text (#140)
- Production deploy now precompiles JSX via esbuild and loads the
production React UMD instead of dev builds + in-browser Babel (#139,
#122)
- Search wrapper gets role=search; global search input gets aria-label,
role=combobox, aria-controls/aria-expanded/aria-activedescendant
wiring (#137, #135)
- Dashboard and Library no longer share the same nav icon (#136)
- Sidebar collapses off-canvas with a topbar menu button below 768 px;
mobile default is collapsed (#134)
- --text-3 bumped to #8B92A0 for WCAG AA contrast on --bg-0 (#133)
- Schedule and Library routes were rendering empty inside the .main
flex container — switched to flex:1 + min-height:0 (#131, #132,
editor + asset detail get the same fix)
- Jobs nav badge now polls /jobs?status=active every 10 s and reflects
the live count (#130, #113)
- aria-label sweep on every icon-only button (#126)
- Premiere panel release list moved to window.PREMIERE_RELEASES in
data.jsx; Editor + Settings read from the same source (#125)
- Typo setPgMclips → setPgmClips (#124)
- Stray console.error / console.warn calls gated behind
window.DF_LOG.{warn,error} (#123)
- Hardcoded /api/v1 paths route through window.ZAMPP_API_PREFIX (#115)
- Schedule rows no longer crash on null recorder_id (#117)
- EditorKeyboard guards against document.activeElement === null (#116)
- Unmount-safe timers for PasswordResetModal, Containers, Editor (#111)
- Player seek clamps below totalMs, server-side range clamping +
uncached 416 on EOF, client-side EOF-stall watchdog (#143)
- Duration badge overlap fix on narrow asset cards (#52)
Backend / security / reliability
- GET /recorders fixed N+1: single LATERAL JOIN for live_asset_id;
Docker inspects bounded to actually-recording rows (#121)
- Upload disk-storage (multer.diskStorage) streams parts to S3 instead
of buffering 500 MB in RAM (#120)
- /assets list clamps limit to MAX_LIMIT=500 to prevent OOM (#119)
- SDK upload archive listing + post-extract sanitize block zip-slip /
tar-slip and symlink escapes (#118)
- Migrations track applied state in schema_migrations, run in a
transaction, and exit non-zero on failure (#107)
- node-agent BMD_COUNT override uses BMD_DEVICE_PREFIX; filesystem
detection wins (#109, #127)
- GPU_COUNT override now merges with nvidia-smi enrichment (#108)
- /cluster/heartbeat requires a node-bound token or admin user;
tokens carry bound_hostname (#106)
- /recorders/:id/start error responses no longer echo the Docker
create payload — env vars stay out of client responses (#105)
- /recorders/probe restricts schemes (srt/rtmp/rtsp/udp/rtp), blocks
private + loopback hosts for non-admins, denies common service
ports (#104)
- Scheduler tick guarded by a Postgres advisory lock; pending/running
rows claimed via UPDATE...RETURNING + FOR UPDATE SKIP LOCKED to
survive multi-node deploys (#103)
- UUID validateUuid('id') param middleware on every /:id route (#102)
- Error handler scrubs Postgres error messages and 5xx detail (#101)
- Graceful SIGTERM/SIGINT shutdown — stops scheduler, drains the HTTP
server, ends the pool, 25 s force-exit watchdog (#100)
- AMPP sync moved from fire-and-forget to a persisted retry queue
(ampp_sync_status / attempts / next_attempt_at + scheduler retry
loop with exponential backoff) (#77)
Migrations
- 019: api_tokens.bound_hostname (#106)
- 020: assets.ampp_sync_status + retry bookkeeping (#77)
Other
- Defer #92 Growing-files per-upload toggle, #80 Audio tab, #57
Dashboard redesign, #56 Editor SPA polish phase 3, #114 S3
migration tool to v1.3
2026-05-26 22:06:14 -04:00
|
|
|
|
<button className="icon-btn" aria-label="Close" onClick={onClose}><Icon name="x" /></button>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div className="modal-body">
|
|
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Recorder name</label>
|
2026-05-22 11:10:01 -04:00
|
|
|
|
<input className="field-input" placeholder="e.g. Studio A Stage Cam"
|
|
|
|
|
|
value={name} onChange={e => setName(e.target.value)} />
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Source type</label>
|
|
|
|
|
|
<div className="source-type-grid">
|
|
|
|
|
|
{[
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
{ id: 'SRT', label: 'SRT', desc: 'Secure Reliable Transport · pull caller', icon: 'signal' },
|
2026-05-28 19:12:40 -04:00
|
|
|
|
{ id: 'RTMP', label: 'RTMP', desc: 'Real-Time Messaging Protocol', icon: 'globe' },
|
|
|
|
|
|
{ id: 'SDI', label: 'SDI', desc: 'Blackmagic DeckLink hardware', icon: 'video' },
|
|
|
|
|
|
{ id: 'DELTACAST', label: 'Deltacast', desc: 'Deltacast VideoMaster SDI', icon: 'video' },
|
2026-05-22 08:19:03 -04:00
|
|
|
|
].map(t => (
|
2026-05-22 11:10:01 -04:00
|
|
|
|
<button key={t.id}
|
2026-05-22 10:55:22 -04:00
|
|
|
|
className={`source-type-card ${sourceType === t.id ? 'active' : ''}`}
|
2026-05-22 11:10:01 -04:00
|
|
|
|
onClick={() => setSourceType(t.id)}>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div className="source-type-icon"><Icon name={t.icon} size={16} /></div>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
<div>
|
|
|
|
|
|
<div style={{ fontWeight: 600, fontSize: 13 }}>{t.label}</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div style={{ fontSize: 11, color: 'var(--text-3)', marginTop: 2 }}>{t.desc}</div>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
</button>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{sourceType === 'SRT' && (
|
2026-05-22 08:19:03 -04:00
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Source URL</label>
|
2026-05-22 11:33:45 -04:00
|
|
|
|
<div style={{ display: 'flex', gap: 6 }}>
|
|
|
|
|
|
<input className="field-input mono" placeholder="srt://192.168.1.100:4200"
|
|
|
|
|
|
value={srtUrl} onChange={e => setSrtUrl(e.target.value)} style={{ flex: 1 }} />
|
|
|
|
|
|
<button className="btn ghost sm" onClick={handleProbe} disabled={probing}
|
|
|
|
|
|
style={{ flexShrink: 0, minWidth: 64 }}>
|
|
|
|
|
|
{probing ? '…' : 'Probe'}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div style={{ fontSize: 11, color: 'var(--text-3)', marginTop: 4 }}>
|
2026-05-22 11:10:01 -04:00
|
|
|
|
Recorder connects out to this URL (caller mode).
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
2026-05-22 11:33:45 -04:00
|
|
|
|
{probeResult && <ProbeResult result={probeResult} />}
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{sourceType === 'RTMP' && (
|
2026-05-22 08:19:03 -04:00
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Source URL</label>
|
2026-05-22 11:33:45 -04:00
|
|
|
|
<div style={{ display: 'flex', gap: 6 }}>
|
|
|
|
|
|
<input className="field-input mono" placeholder="rtmp://server/live/streamkey"
|
|
|
|
|
|
value={rtmpUrl} onChange={e => setRtmpUrl(e.target.value)} style={{ flex: 1 }} />
|
|
|
|
|
|
<button className="btn ghost sm" onClick={handleProbe} disabled={probing}
|
|
|
|
|
|
style={{ flexShrink: 0, minWidth: 64 }}>
|
|
|
|
|
|
{probing ? '…' : 'Probe'}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div style={{ fontSize: 11, color: 'var(--text-3)', marginTop: 4 }}>
|
2026-05-22 11:10:01 -04:00
|
|
|
|
Recorder pulls this RTMP stream.
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
2026-05-22 11:33:45 -04:00
|
|
|
|
{probeResult && <ProbeResult result={probeResult} />}
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{sourceType === 'SDI' && (
|
|
|
|
|
|
<div className="field">
|
2026-05-22 11:10:01 -04:00
|
|
|
|
<label className="field-label">Capture device</label>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{sdiDevices === null && (
|
2026-05-22 11:10:01 -04:00
|
|
|
|
<div style={{ padding: '10px 0', color: 'var(--text-3)', fontSize: 12 }}>Detecting DeckLink devices…</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
)}
|
|
|
|
|
|
{sdiDevices !== null && sdiDevices.length > 0 && (
|
2026-05-28 19:18:55 -04:00
|
|
|
|
<DevicePortPicker
|
|
|
|
|
|
ports={sdiDevices}
|
|
|
|
|
|
selectedIdx={sdiDeviceIdx}
|
|
|
|
|
|
selectedNode={sdiNodeId}
|
|
|
|
|
|
onSelect={(idx, nodeId) => { setSdiDeviceIdx(idx); setSdiNodeId(nodeId); }}
|
|
|
|
|
|
portLabel="SDI"
|
|
|
|
|
|
/>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
)}
|
2026-05-22 11:10:01 -04:00
|
|
|
|
{sdiDevices !== null && sdiDevices.length === 0 && (
|
2026-05-28 19:18:55 -04:00
|
|
|
|
<ManualDevicePicker
|
|
|
|
|
|
nodes={NODES}
|
|
|
|
|
|
nodeId={sdiNodeId}
|
|
|
|
|
|
deviceIdx={sdiDeviceIdx}
|
|
|
|
|
|
portLabel="SDI"
|
|
|
|
|
|
portCount={4}
|
|
|
|
|
|
onNodeChange={setSdiNodeId}
|
|
|
|
|
|
onIdxChange={setSdiDeviceIdx}
|
|
|
|
|
|
/>
|
2026-05-22 11:10:01 -04:00
|
|
|
|
)}
|
2026-05-22 10:55:22 -04:00
|
|
|
|
</div>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
)}
|
|
|
|
|
|
|
2026-05-28 19:12:40 -04:00
|
|
|
|
{sourceType === 'DELTACAST' && (
|
|
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Capture device</label>
|
|
|
|
|
|
{dcDevices === null && (
|
|
|
|
|
|
<div style={{ padding: '10px 0', color: 'var(--text-3)', fontSize: 12 }}>Detecting Deltacast devices…</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
{dcDevices !== null && dcDevices.length > 0 && (
|
2026-05-28 19:18:55 -04:00
|
|
|
|
<DevicePortPicker
|
|
|
|
|
|
ports={dcDevices}
|
|
|
|
|
|
selectedIdx={dcDeviceIdx}
|
|
|
|
|
|
selectedNode={dcNodeId}
|
|
|
|
|
|
onSelect={(idx, nodeId) => { setDcDeviceIdx(idx); setDcNodeId(nodeId); }}
|
|
|
|
|
|
portLabel="Port"
|
|
|
|
|
|
showTestBadge
|
|
|
|
|
|
/>
|
2026-05-28 19:12:40 -04:00
|
|
|
|
)}
|
|
|
|
|
|
{dcDevices !== null && dcDevices.length === 0 && (
|
2026-05-28 19:18:55 -04:00
|
|
|
|
<ManualDevicePicker
|
|
|
|
|
|
nodes={NODES}
|
|
|
|
|
|
nodeId={dcNodeId}
|
|
|
|
|
|
deviceIdx={dcDeviceIdx}
|
|
|
|
|
|
portLabel="Port"
|
|
|
|
|
|
portCount={8}
|
|
|
|
|
|
onNodeChange={setDcNodeId}
|
|
|
|
|
|
onIdxChange={setDcDeviceIdx}
|
|
|
|
|
|
emptyNote="No Deltacast devices detected. Configure manually (test-card mode):"
|
|
|
|
|
|
/>
|
2026-05-28 19:12:40 -04:00
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
2026-05-22 08:19:03 -04:00
|
|
|
|
<div className="modal-section">
|
|
|
|
|
|
<div className="modal-section-head">
|
|
|
|
|
|
<span>Master recording</span>
|
|
|
|
|
|
<span style={{ flex: 1 }} />
|
|
|
|
|
|
<div className="tab-group">
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{['video', 'audio', 'container'].map(t => (
|
|
|
|
|
|
<button key={t} className={recTab === t ? 'active' : ''} onClick={() => setRecTab(t)}>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
{t[0].toUpperCase() + t.slice(1)}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="modal-section-body">
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{recTab === 'video' && (
|
2026-05-31 18:34:36 -04:00
|
|
|
|
<>
|
|
|
|
|
|
{/* Codec presets — one click fills codec + bitrate with a known-good
|
|
|
|
|
|
combo that passes the server-side validateRecorderConfig guard.
|
|
|
|
|
|
Container is derived from the codec (HEVC/ProRes/DNxHR → MOV,
|
|
|
|
|
|
H.264 → MP4), and master audio is always PCM (valid in MOV). */}
|
|
|
|
|
|
<div style={{ display: 'flex', flexWrap: 'wrap', gap: 6, marginBottom: 12 }}>
|
|
|
|
|
|
{[
|
2026-06-01 18:41:36 -04:00
|
|
|
|
{ id: 'hevc', label: 'HEVC Master (MOV)', codec: 'hevc_nvenc', bitrate: '25' },
|
2026-05-31 18:34:36 -04:00
|
|
|
|
{ id: 'h264', label: 'H.264 Proxy-friendly (MP4)', codec: 'h264_nvenc', bitrate: '25' },
|
|
|
|
|
|
{ id: 'dnxhr', label: 'DNxHR HQ (MOV)', codec: 'dnxhr_hq', bitrate: '145' },
|
|
|
|
|
|
].map(p => (
|
|
|
|
|
|
<button key={p.id}
|
|
|
|
|
|
className={`btn ghost sm${recCodec === p.codec ? ' active' : ''}`}
|
|
|
|
|
|
onClick={() => { setRecCodec(p.codec); setRecBitrate(p.bitrate); }}
|
|
|
|
|
|
style={{ flexShrink: 0 }}>
|
|
|
|
|
|
{p.label}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div style={{ display: 'grid', gridTemplateColumns: '1fr 1fr', gap: 10 }}>
|
2026-05-22 17:20:01 -04:00
|
|
|
|
<div className="field">
|
2026-05-31 22:13:01 -04:00
|
|
|
|
<label className="field-label">
|
|
|
|
|
|
Video codec{growingOn && <span style={{ marginLeft: 6, fontSize: 10, fontWeight: 700, letterSpacing: '0.06em', color: 'var(--warn, #d9a441)' }}>· LOCKED BY GROWING</span>}
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<select className="field-input" value={growingOn ? 'h264_growing' : recCodec}
|
|
|
|
|
|
onChange={e => setRecCodec(e.target.value)} disabled={growingOn}
|
|
|
|
|
|
style={{ appearance: 'auto', opacity: growingOn ? 0.6 : 1 }}>
|
design: overhaul pass — amber accent, home recomposition, motion layer
Lever 1 (color): Replace #5B7CFA AI-blue with electric amber #E8821C across
all accent tokens, tile tones, logo glows, and hardcoded rgba values. Dark
text on amber primary buttons for WCAG AA contrast.
Lever 2 (home): Collapse centered logo hero into compact left-aligned header.
Split tile grid into primary ops row (Library, Recorders, Playout) + secondary
4-col row (Downloads, Jobs, Dashboard, Settings) with reduced visual weight.
Lever 3 (typography): Remove v1.2.0 from sidebar. Fix em-dashes to hyphens or
periods across all visible UI strings (option labels, body copy, error messages).
Topbar height 56px -> 48px.
Lever 4 (motion): Staggered entry animation for launcher tiles
(prefers-reduced-motion gated). Tactile scale(0.97) on primary/record buttons.
Smooth 150ms nav active-item transitions.
Lever 5 (blocks): Jobs stats row semantic card variants (amber glow when
active, red border when failed, quiet muted style for Total).
Lever 6 (spacing): Topbar 48px, launcher inner gap tightened, status left-aligned.
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
2026-06-02 08:20:15 -04:00
|
|
|
|
{growingOn && <option value="h264_growing">XDCAM HD422 (MXF OP1a, growing)</option>}
|
|
|
|
|
|
<option value="hevc_nvenc">All-Intra HEVC (NVENC, GPU, growing)</option>
|
|
|
|
|
|
<option value="h264_nvenc">H.264 (NVENC, GPU)</option>
|
|
|
|
|
|
<option value="prores_hq">ProRes 422 HQ (4:2:2, CPU)</option>
|
2026-05-22 17:20:01 -04:00
|
|
|
|
<option value="prores">ProRes 422</option>
|
|
|
|
|
|
<option value="prores_lt">ProRes 422 LT</option>
|
|
|
|
|
|
<option value="prores_proxy">ProRes 422 Proxy</option>
|
|
|
|
|
|
<option value="dnxhr_hq">DNxHR HQ</option>
|
2026-05-29 17:04:00 -04:00
|
|
|
|
<option value="libx264">H.264 (x264, CPU)</option>
|
|
|
|
|
|
<option value="libx265">H.265 (x265, CPU)</option>
|
2026-05-22 17:20:01 -04:00
|
|
|
|
</select>
|
|
|
|
|
|
</div>
|
2026-05-31 22:13:01 -04:00
|
|
|
|
{showBitrate ? (
|
2026-05-29 17:04:00 -04:00
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Target bitrate (Mbps)</label>
|
|
|
|
|
|
<input
|
|
|
|
|
|
className="field-input"
|
|
|
|
|
|
type="number" min="1" max="400" step="1"
|
|
|
|
|
|
value={recBitrate}
|
|
|
|
|
|
onChange={e => setRecBitrate(e.target.value)}
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<Field label="Bitrate" value="Quality-based (profile)" select />
|
|
|
|
|
|
)}
|
design: overhaul pass — amber accent, home recomposition, motion layer
Lever 1 (color): Replace #5B7CFA AI-blue with electric amber #E8821C across
all accent tokens, tile tones, logo glows, and hardcoded rgba values. Dark
text on amber primary buttons for WCAG AA contrast.
Lever 2 (home): Collapse centered logo hero into compact left-aligned header.
Split tile grid into primary ops row (Library, Recorders, Playout) + secondary
4-col row (Downloads, Jobs, Dashboard, Settings) with reduced visual weight.
Lever 3 (typography): Remove v1.2.0 from sidebar. Fix em-dashes to hyphens or
periods across all visible UI strings (option labels, body copy, error messages).
Topbar height 56px -> 48px.
Lever 4 (motion): Staggered entry animation for launcher tiles
(prefers-reduced-motion gated). Tactile scale(0.97) on primary/record buttons.
Smooth 150ms nav active-item transitions.
Lever 5 (blocks): Jobs stats row semantic card variants (amber glow when
active, red border when failed, quiet muted style for Total).
Lever 6 (spacing): Topbar 48px, launcher inner gap tightened, status left-aligned.
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
2026-06-02 08:20:15 -04:00
|
|
|
|
<Field label="Resolution" value="Auto (from source)" select />
|
|
|
|
|
|
<Field label="Framerate" value="Auto (from source)" select />
|
2026-05-29 17:04:00 -04:00
|
|
|
|
{/* #3: warn when the configured bitrate exceeds the probed source
|
|
|
|
|
|
bitrate — re-encoding above source adds storage, not quality. */}
|
|
|
|
|
|
{codecUsesBitrate && (() => {
|
|
|
|
|
|
const d = probeResult && probeResult.ok ? (probeResult.data || {}) : null;
|
|
|
|
|
|
const raw = d && (d.bitrate ?? d.bit_rate ?? d.video_bitrate ?? (d.video && d.video.bit_rate));
|
|
|
|
|
|
const srcMbps = raw ? (Number(raw) > 100000 ? Number(raw) / 1e6 : Number(raw)) : null;
|
|
|
|
|
|
const cfg = parseFloat(recBitrate);
|
|
|
|
|
|
if (srcMbps && cfg && cfg > srcMbps * 1.05) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<div style={{ gridColumn: '1 / -1', fontSize: 11.5, color: 'var(--warn, #d9a441)', border: '1px solid var(--warn, #d9a441)', borderRadius: 6, padding: '8px 10px', background: 'rgba(217,164,65,0.08)' }}>
|
|
|
|
|
|
⚠ Target {cfg} Mbps exceeds the source stream (~{srcMbps.toFixed(1)} Mbps). Encoding above the source bitrate increases file size without adding quality.
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
return null;
|
|
|
|
|
|
})()}
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
2026-05-31 18:34:36 -04:00
|
|
|
|
</>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
)}
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{recTab === 'audio' && (
|
|
|
|
|
|
<div style={{ display: 'grid', gridTemplateColumns: '1fr 1fr', gap: 10 }}>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
<Field label="Audio codec" value="PCM" select />
|
|
|
|
|
|
<Field label="Sample rate" value="48 kHz" select />
|
|
|
|
|
|
<Field label="Channels" value="2.0 stereo" select />
|
|
|
|
|
|
<Field label="Bit depth" value="24-bit" select />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{recTab === 'container' && (
|
|
|
|
|
|
<div style={{ display: 'grid', gridTemplateColumns: '1fr 1fr', gap: 10 }}>
|
2026-05-31 22:13:01 -04:00
|
|
|
|
<Field label="Container"
|
2026-05-31 22:14:52 -04:00
|
|
|
|
value={growingOn ? 'MXF OP1a (growing, auto)' : (recContainer === 'mp4' ? 'MP4 (auto)' : 'Fragmented MOV (auto)')} select />
|
2026-05-31 22:13:01 -04:00
|
|
|
|
<Field label="Growing-file"
|
|
|
|
|
|
value={growingOn ? 'On (edit-while-record, locked)' : (recContainer === 'mov' ? 'Supported (edit-while-record)' : 'No')} select />
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div className="modal-toggle-row">
|
|
|
|
|
|
<label className="switch">
|
|
|
|
|
|
<input type="checkbox" checked={proxyOn} onChange={e => setProxyOn(e.target.checked)} />
|
|
|
|
|
|
<span className="switch-track"><span className="switch-knob" /></span>
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<div style={{ fontWeight: 500, fontSize: 13 }}>Generate proxy</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div style={{ fontSize: 11, color: 'var(--text-3)' }}>
|
|
|
|
|
|
SDI sources record proxy in parallel. Network sources generate proxy after stop.
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-05-31 14:50:31 -04:00
|
|
|
|
<div className="modal-toggle-row">
|
|
|
|
|
|
<label className="switch">
|
|
|
|
|
|
<input type="checkbox" checked={growingOn} onChange={e => setGrowingOn(e.target.checked)} />
|
|
|
|
|
|
<span className="switch-track"><span className="switch-knob" /></span>
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<div style={{ fontWeight: 500, fontSize: 13 }}>Growing-files mode</div>
|
|
|
|
|
|
<div style={{ fontSize: 11, color: 'var(--text-3)' }}>
|
|
|
|
|
|
Write the live master to the SMB share so editors can cut while it's still recording.
|
|
|
|
|
|
Requires the SMB share to be configured in Settings → Storage.
|
|
|
|
|
|
</div>
|
2026-05-31 21:56:58 -04:00
|
|
|
|
{growingOn && (
|
|
|
|
|
|
<div style={{ fontSize: 11, color: 'var(--warn, #d9a441)', marginTop: 6 }}>
|
2026-05-31 22:14:52 -04:00
|
|
|
|
Growing-files mode records XDCAM HD422 (MPEG-2 4:2:2 CBR) in MXF OP1a — the format Premiere supports for edit-while-record growing files. Bitrate below still applies.
|
2026-05-31 21:56:58 -04:00
|
|
|
|
Premiere can import while it's still being written. The codec and container above
|
|
|
|
|
|
are overridden for this recorder (the target bitrate still applies). Turn growing
|
|
|
|
|
|
off to record your selected master codec/container.
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
2026-05-31 14:50:31 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-05-22 08:19:03 -04:00
|
|
|
|
{proxyOn && (
|
|
|
|
|
|
<div className="modal-section">
|
2026-05-22 17:20:01 -04:00
|
|
|
|
<div className="modal-section-head"><span>Proxy</span></div>
|
|
|
|
|
|
<div className="modal-section-body">
|
|
|
|
|
|
<div style={{ display: 'flex', flexWrap: 'wrap', gap: 6, alignItems: 'center' }}>
|
|
|
|
|
|
{['H.264', '2 Mbps', 'MP4', '1920×1080', 'AAC 128 kbps'].map(tag => (
|
|
|
|
|
|
<span key={tag} className="mono" style={{ background: 'var(--bg-3)', borderRadius: 4, padding: '2px 8px', fontSize: 12 }}>{tag}</span>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
ui: full audit pass (fixes #146, #147, #148, #149, #151, #152, #153, #154, #155)
Sweep of 9 web-ui audit findings from tracker #156. Issue #150 (modal
codec stubs) deferred per user request.
## #146 sweep em-dashes (186 to 0)
- Replace placeholder '—' with '·' across all jsx
- Convert ' — ' to ': ' or '. ' in copy where context permits
- Comment-only em-dashes converted to ASCII dash
- Sweep css files too (16 comments)
## #147 remove glassmorphism + accent gradients
- Strip 8 backdrop-filter declarations from styles-screens.css and
styles-asset.css. Only legit modal scrim in styles-modal.css remains.
- Replace .job-progress-fill gradient with solid var(--accent)
- Replace .monitor-tile.audio gradient with flat var(--bg-1)
## #148 extract Jobs inline styles to CSS
- Cut 19 inline style={{...}} blocks in screens-jobs.jsx to 1 (dynamic
width on progress bar). Live DOM was 487 inline-styled elements due
to per-row repetition; now ~0.
- Added job-row-kind, job-row-asset, job-row-node, job-row-time,
job-row-actions, job-row-status-* utility classes in styles-screens.css
## #149 sidebar IA reorganized
- Replace flat NAV_TREE + ADMIN_TREE with NAV_SECTIONS:
Workspace / Ingest / Operations / Admin
- Move Capture out of Ingest into Operations (it's a live-signal monitor,
not an ingest action)
- Drop the 0/N capture badge from nav (belongs in topbar)
- Add BETA badge to Editor
## #151 redesign Editor 'Coming Soon' bumper
- Replace fullscreen glassmorphism + gradient + glow overlay with a flat
beta banner across the top of the editor area
- New .editor-beta-banner CSS class (flat, accent-soft tint, no blur)
## #152 hide Tokens parody, restore real API token mgmt
- New top-level Tokens admin page wraps existing ApiTokensSection
- Old parody renamed to TokensParody, accessible at /tokens-parody route
- Add window-level df:nav event for cross-component routing
## #153 make Home actually useful
- New activity strip below the launcher grid: 'Recording now' tiles for
live recorders, 'Last 24 hours' tiles for newly created assets, plus
an attention strip when there are failed jobs or errored recorders
- Each item is clickable and routes to the relevant screen
## #154 aria-labels on icon-only buttons
- Projects + Library grid/list view toggles now have aria-label + title
## #155 page-header pattern
- Dashboard now renders a proper .page-header h1 with subtitle + alert
badge + cluster status pip
- Library toolbar-title promoted to h1 for screen-reader hierarchy
- Document Home/Library/Editor full-bleed exceptions in DESIGN.md
- Editor's chrome is the beta banner (covered by #151)
2026-05-28 19:50:07 -04:00
|
|
|
|
<div style={{ fontSize: 11, color: 'var(--text-3)', marginTop: 6 }}>Fixed proxy profile. Not configurable.</div>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
<div className="modal-section">
|
|
|
|
|
|
<div className="modal-section-head"><span>Destination</span></div>
|
|
|
|
|
|
<div className="modal-section-body">
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<div className="field">
|
|
|
|
|
|
<label className="field-label">Project</label>
|
2026-05-22 11:10:01 -04:00
|
|
|
|
<select className="field-input" value={projectId}
|
|
|
|
|
|
onChange={e => setProjectId(e.target.value)} style={{ appearance: 'auto' }}>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
{PROJECTS.length === 0
|
|
|
|
|
|
? <option value="">No projects</option>
|
|
|
|
|
|
: PROJECTS.map(p => <option key={p.id} value={p.id}>{p.name}</option>)}
|
|
|
|
|
|
</select>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
|
|
|
|
|
|
{submitErr && (
|
2026-05-22 11:10:01 -04:00
|
|
|
|
<div style={{
|
|
|
|
|
|
padding: '10px 14px', background: 'var(--danger-soft)',
|
|
|
|
|
|
border: '1px solid var(--danger)', borderRadius: 6,
|
|
|
|
|
|
fontSize: 12.5, color: 'var(--danger)', marginTop: 4,
|
|
|
|
|
|
}}>{submitErr}</div>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
)}
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div className="modal-foot">
|
|
|
|
|
|
<button className="btn ghost" onClick={onClose}>Cancel</button>
|
|
|
|
|
|
<span style={{ flex: 1 }} />
|
2026-05-22 10:55:22 -04:00
|
|
|
|
<button className="btn primary" onClick={handleCreate} disabled={submitting}>
|
|
|
|
|
|
{submitting ? 'Creating…' : 'Create recorder'}
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</button>
|
2026-05-22 10:55:22 -04:00
|
|
|
|
</div>
|
2026-05-22 08:19:03 -04:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
window.NewRecorderModal = NewRecorderModal;
|