refactor(shopfloor): modern redesign w/ gradients, theme-safe tokens

Shop-floor operators and managers live on these screens all day — the
old look worked but felt like a spec sheet. Pass through all 5 pages
with one design language: gradient KPI cards, hero banners, soft
shadows, rounded corners, large touch targets, friendly empty states.
Dark mode and light mode both look deliberate, not inverted.

New shared file: _fp_shopfloor_tokens.scss

  A single source of truth for radii, elevation (shadows that respect
  dark mode via color-mix on foreground), typography scale (tabular
  numerics for KPIs, 18px base for shop-floor readability), animation
  easings, semantic gradients (@mixin fp-grad), tone helpers
  (@mixin fp-tone), focus ring, and the 44px touch-min token.
  Every other SCSS file imports this — no duplicated colour math.

  Gradients are built on color-mix(in srgb, var(--bs-foo) X%, transparent)
  so they layer naturally on either the light or dark page background.
  No @media-prefers-color-scheme forks needed.

Tablet Station (fusion_plating_shopfloor.scss):

  * Hero banner with dual radial-gradient wash (brand + success), live
    station chip, gradient focus ring on the picker.
  * KPI cards (6-up on desktop, 2x3 on phone) get a subtle top accent
    line, coloured gradient overlay, 40px headline number, faded icon
    at corner. Tone variants (info/success/warning/danger/muted) drive
    colour without extra CSS.
  * Active WO banner is a green gradient pill with a breathing-dot
    pulse — unmissable when something is running.
  * Panels get top accents, queue rows get priority pills (HI/M/·),
    bake/gate/hold rows get colour-coded left accent bars.
  * Tiles have a 4px left stripe keyed to state + hover lift.
  * Status chips are uppercase, pill-shaped, tone-tinted with
    color-mix so they respect theme.
  * Empty states now have a large 36px icon + friendly copy instead
    of a one-liner.
  * Focus rings use the shared @mixin fp-focus-ring.

Manager Desk (manager_dashboard.scss):

  * Same hero treatment with radial gradient + live-dot pulse.
  * 3 panels carry a coloured top accent bar — amber (Unassigned),
    green (In Progress), blue (Team). Instant visual routing.
  * KPI strip matches tablet.
  * MO cards get a left priority stripe (red for HOT, amber for
    Urgent), lift on hover, expand cleanly.
  * Team avatars get a border + subtle tint background for depth.
  * Worker/tank pickers have custom focus rings.

Plant Overview (plant_overview.scss):

  * Header is now a gradient wash tied to the brand colour.
  * Work-centre columns get a thin gradient top-stripe and pill-style
    count badges.
  * Cards have real depth (layered shadow), lift harder on hover,
    change border colour on hover.

All three files share the same design tokens, so colours/shadows/
radii are identical across pages. Edit one place, everything updates.

Verified: backend asset bundle compiles clean (no SCSS errors), zero
warnings on module upgrade, asset cache cleared for fresh delivery.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
gsinghpal
2026-04-18 18:30:47 -04:00
parent ae03e32b5d
commit 298f5942eb
6 changed files with 1055 additions and 522 deletions

View File

@@ -0,0 +1,115 @@
// =============================================================================
// Fusion Plating — Shop-Floor Design Tokens
// Copyright 2026 Nexa Systems Inc.
// License OPL-1 (Odoo Proprietary License v1.0)
//
// Single source of truth for the look-and-feel of every shop-floor OWL page.
// All values use CSS custom properties (--bs-*, --o-*) so the tokens adapt
// automatically between Odoo's light and dark themes — no media queries or
// duplicate palettes.
//
// Gradients use `color-mix(in srgb, var(--bs-foo) X%, transparent)` because
// transparent mixes with the page background, which already obeys the theme.
// =============================================================================
// ---------- Radii ------------------------------------------------------------
$fp-radius-sm : 8px;
$fp-radius-md : 12px;
$fp-radius-lg : 16px;
$fp-radius-xl : 20px;
$fp-radius-pill : 999px;
// ---------- Elevation (shadows that respect dark mode) -----------------------
// Shadows are built on the page's foreground colour so they darken in light
// mode and deepen (but never go pure black) in dark mode.
$fp-shadow-xs : 0 1px 2px color-mix(in srgb, var(--bs-body-color) 6%, transparent);
$fp-shadow-sm : 0 2px 6px color-mix(in srgb, var(--bs-body-color) 9%, transparent);
$fp-shadow-md : 0 4px 14px color-mix(in srgb, var(--bs-body-color) 10%, transparent),
0 1px 3px color-mix(in srgb, var(--bs-body-color) 6%, transparent);
$fp-shadow-lg : 0 10px 28px color-mix(in srgb, var(--bs-body-color) 14%, transparent),
0 2px 6px color-mix(in srgb, var(--bs-body-color) 6%, transparent);
$fp-shadow-glow-primary : 0 0 0 4px color-mix(in srgb, var(--o-action) 12%, transparent);
// ---------- Surface layering -------------------------------------------------
// A "raised" surface tint — sits slightly above the base background. Uses
// the foreground colour to tint so it deepens correctly in dark mode.
$fp-surface : var(--o-view-background-color, var(--bs-body-bg));
$fp-surface-raised : color-mix(in srgb,
var(--bs-body-color) 2%,
var(--o-view-background-color, var(--bs-body-bg)));
$fp-surface-sunken : color-mix(in srgb,
var(--bs-body-color) 4%,
var(--o-view-background-color, var(--bs-body-bg)));
// ---------- Border-tints -----------------------------------------------------
$fp-border : var(--bs-border-color);
$fp-border-strong : color-mix(in srgb, var(--bs-body-color) 18%, transparent);
$fp-border-accent : color-mix(in srgb, var(--o-action) 40%, var(--bs-border-color));
// ---------- Typography -------------------------------------------------------
// Shop-floor tablets and phones get read from 18" away — bumps over Odoo's
// default body size. Numbers are tabular so KPIs don't jitter on refresh.
$fp-font-base : 1rem; // 16px
$fp-font-sm : 0.875rem; // 14px
$fp-font-xs : 0.78rem; // 12.5px
$fp-font-lg : 1.125rem; // 18px
$fp-font-xl : 1.375rem; // 22px
$fp-font-2xl : 1.75rem; // 28px
$fp-font-3xl : 2.25rem; // 36px
$fp-font-kpi : 2.5rem; // 40px — headline number
$fp-font-hero : clamp(1.5rem, 3.5vw, 2.25rem);
// ---------- Animation --------------------------------------------------------
$fp-ease : cubic-bezier(0.22, 1, 0.36, 1);
$fp-ease-snap : cubic-bezier(0.34, 1.56, 0.64, 1);
$fp-dur-fast : 140ms;
$fp-dur : 220ms;
$fp-dur-slow : 360ms;
// ---------- Semantic gradients (tone-based, theme-safe) ----------------------
// Each tone mixes the named Bootstrap token with transparent so it lays
// naturally over whatever page background it's on — light OR dark.
@mixin fp-grad($color-var, $strong: 18%, $soft: 6%) {
background-image: linear-gradient(
135deg,
color-mix(in srgb, var(#{$color-var}) #{$strong}, transparent) 0%,
color-mix(in srgb, var(#{$color-var}) #{$soft}, transparent) 100%
);
}
@mixin fp-grad-solid($color-var, $from: 100%, $to: 75%) {
background-image: linear-gradient(
135deg,
color-mix(in srgb, var(#{$color-var}) #{$from}, transparent) 0%,
color-mix(in srgb, var(#{$color-var}) #{$to}, transparent) 100%
);
color: var(--o-we-text-on-action, #fff);
}
// ---------- Tonal helpers ----------------------------------------------------
// Border + tint combo used on banners / pills — reads as "slightly
// saturated panel" in both themes.
@mixin fp-tone($color-var, $strength: 14%) {
background-color: color-mix(in srgb, var(#{$color-var}) #{$strength}, transparent);
color: var(#{$color-var});
border: 1px solid color-mix(in srgb, var(#{$color-var}) 38%, transparent);
}
// ---------- Focus ring -------------------------------------------------------
@mixin fp-focus-ring() {
outline: none;
box-shadow: 0 0 0 3px color-mix(in srgb, var(--o-action) 35%, transparent);
}
// ---------- Touch target -----------------------------------------------------
$fp-touch-min : 44px; // Apple HIG minimum

View File

@@ -1,39 +1,20 @@
// ============================================================================= // =============================================================================
// Fusion Plating — Shop Floor backend / tablet styles // Fusion Plating — Shop-Floor Tablet Station
// Copyright 2026 Nexa Systems Inc. // Copyright 2026 Nexa Systems Inc.
// License OPL-1 (Odoo Proprietary License v1.0) // License OPL-1 (Odoo Proprietary License v1.0)
// //
// THEME AWARENESS // Modernised 2026-04 design: hero banner, gradient KPI cards, tap-first
// --------------- // action rows. Works in Odoo's light + dark themes because every colour
// All colours come from CSS custom properties (Bootstrap / Odoo tokens) so // resolves from CSS custom properties or color-mix'd tokens.
// the tablet view renders correctly in BOTH light and dark mode without any
// duplication or media queries. Status tints use color-mix() against the
// theme token so green/yellow/red adapt to the surface.
//
// background: var(--bs-body-bg)
// surface: var(--o-view-background-color)
// foreground: var(--bs-body-color)
// muted text: var(--bs-secondary-color)
// border: var(--bs-border-color)
// primary: var(--o-action)
// ============================================================================= // =============================================================================
@import "./fp_shopfloor_tokens";
// -----------------------------------------------------------------------------
// Local mixin — semantic tint that respects light/dark mode
// -----------------------------------------------------------------------------
@mixin fp-shop-tint($color-var, $amount: 14%) {
background-color: color-mix(in srgb, var(#{$color-var}) #{$amount}, transparent);
color: var(#{$color-var});
border: 1px solid color-mix(in srgb, var(#{$color-var}) 35%, transparent);
}
// ----------------------------------------------------------------------------- // =============================================================================
// Global touch tweaks — apply to every shop-floor page // Global touch / hover suppression (kept from previous rev)
// ----------------------------------------------------------------------------- // =============================================================================
@media (hover: none) { @media (hover: none) {
// Hover highlights stick on tap — disable them on touch devices.
.o_fp_tablet .o_fp_queue_row:hover, .o_fp_tablet .o_fp_queue_row:hover,
.o_fp_tablet .o_fp_tile:hover, .o_fp_tablet .o_fp_tile:hover,
.o_fp_tablet .o_fp_tablet_card:hover { .o_fp_tablet .o_fp_tablet_card:hover {
@@ -44,442 +25,643 @@
} }
// ----------------------------------------------------------------------------- // =============================================================================
// Tablet root container — large touch targets, generous whitespace // Root container
// ----------------------------------------------------------------------------- // =============================================================================
.o_fp_tablet { .o_fp_tablet {
background-color: var(--o-view-background-color, var(--bs-body-bg)); background-color: $fp-surface-sunken;
color: var(--bs-body-color); color: var(--bs-body-color);
min-height: 100%; min-height: 100%;
padding: 20px 24px; padding: 24px 28px;
font-size: 1rem; font-size: $fp-font-base;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 14px; gap: 18px;
// iPad portrait / small tablet @media (max-width: 900px) { padding: 16px; gap: 14px; }
@media (max-width: 900px) { padding: 14px 12px; gap: 10px; } @media (max-width: 600px) { padding: 10px 10px 16px; gap: 10px; }
// Phone
@media (max-width: 600px) { padding: 10px 8px; gap: 8px; font-size: 0.95rem; }
// ---------- Header -------------------------------------------------------
// =========================================================================
// Hero banner — sits at the top, shows who's logged in + station
// =========================================================================
.o_fp_tablet_header { .o_fp_tablet_header {
display: flex; position: relative;
justify-content: space-between; overflow: hidden;
align-items: center; border-radius: $fp-radius-lg;
padding: 20px 24px;
background-color: $fp-surface-raised;
border: 1px solid $fp-border;
box-shadow: $fp-shadow-sm;
// Brand gradient wash — primary + success blend reads as
// "productive activity". Low opacity so text stays readable.
&::before {
content: "";
position: absolute;
inset: 0;
background-image:
radial-gradient(circle at 0% 0%,
color-mix(in srgb, var(--o-action) 22%, transparent) 0%,
transparent 55%),
radial-gradient(circle at 100% 100%,
color-mix(in srgb, var(--bs-success) 20%, transparent) 0%,
transparent 55%);
pointer-events: none;
z-index: 0;
}
display: grid;
grid-template-columns: 1fr auto;
gap: 16px; gap: 16px;
flex-wrap: wrap; align-items: center;
> * { position: relative; z-index: 1; }
@media (max-width: 600px) { @media (max-width: 600px) {
flex-direction: column; padding: 14px 14px 12px;
align-items: stretch; grid-template-columns: 1fr;
gap: 8px; gap: 10px;
.o_fp_tablet_header_actions {
width: 100%;
flex-wrap: wrap;
> * { flex: 1; min-width: 0; }
}
} }
} }
.o_fp_tablet_title { .o_fp_tablet_title {
font-size: 1.5rem; font-size: $fp-font-hero;
font-weight: 600; font-weight: 700;
@media (max-width: 600px) { font-size: 1.15rem; } line-height: 1.1;
letter-spacing: -0.01em;
color: var(--bs-body-color);
display: flex; align-items: center; gap: 10px;
} }
.o_fp_tablet_subtitle { .o_fp_tablet_subtitle {
font-size: 0.95rem; margin-top: 4px;
font-size: $fp-font-sm;
color: var(--bs-secondary-color); color: var(--bs-secondary-color);
display: flex; flex-wrap: wrap; gap: 10px;
} }
.o_fp_tablet_chip { .o_fp_tablet_chip {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
gap: 4px; gap: 6px;
padding: 2px 10px; padding: 4px 12px;
background-color: color-mix(in srgb, var(--o-action) 12%, transparent); border-radius: $fp-radius-pill;
border: 1px solid color-mix(in srgb, var(--o-action) 35%, transparent); font-size: $fp-font-xs;
color: var(--o-action); font-weight: 600;
border-radius: 999px; letter-spacing: 0.02em;
font-size: 0.9rem; @include fp-tone(--o-action, 14%);
} }
.o_fp_tablet_header_actions { .o_fp_tablet_header_actions {
display: flex; display: flex; gap: 8px; align-items: center;
gap: 8px; @media (max-width: 600px) {
align-items: center; width: 100%; flex-wrap: wrap;
> * { flex: 1; min-width: 0; }
}
} }
.o_fp_station_picker { .o_fp_station_picker {
min-width: 240px; min-width: 240px;
max-width: 320px; max-width: 340px;
min-height: 38px; min-height: $fp-touch-min;
@media (max-width: 600px) { padding: 8px 14px;
min-width: 0; border-radius: $fp-radius-md;
max-width: 100%; background-color: $fp-surface;
} border: 1px solid $fp-border;
color: var(--bs-body-color);
font-size: $fp-font-base;
transition: border-color $fp-dur $fp-ease, box-shadow $fp-dur $fp-ease;
&:focus { @include fp-focus-ring; border-color: var(--o-action); }
@media (max-width: 600px) { min-width: 0; max-width: 100%; }
} }
.o_fp_scan_toggle { .o_fp_scan_toggle {
white-space: nowrap; min-height: $fp-touch-min;
min-height: 44px; padding: 8px 18px;
border-radius: $fp-radius-md;
font-weight: 600;
transition: transform $fp-dur-fast $fp-ease;
&:active { transform: scale(0.97); }
} }
// ---------- Scan drawer --------------------------------------------------
// =========================================================================
// Scan drawer
// =========================================================================
.o_fp_scan_drawer { .o_fp_scan_drawer {
display: flex; display: flex; gap: 10px;
gap: 10px; padding: 14px;
padding: 12px; border-radius: $fp-radius-lg;
background-color: color-mix(in srgb, var(--o-action) 6%, var(--o-view-background-color, var(--bs-body-bg))); background-color: $fp-surface-raised;
border: 1px dashed color-mix(in srgb, var(--o-action) 40%, transparent); border: 1px dashed $fp-border-accent;
border-radius: 10px; box-shadow: $fp-shadow-xs;
} }
// ---------- Flash message ------------------------------------------------
// =========================================================================
// Flash message
// =========================================================================
.o_fp_tablet_message { .o_fp_tablet_message {
padding: 10px 14px; padding: 12px 16px;
border-radius: 8px; border-radius: $fp-radius-md;
font-size: 1rem; font-size: $fp-font-base;
&.o_fp_msg_info { @include fp-shop-tint(--bs-info); } font-weight: 500;
&.o_fp_msg_success { @include fp-shop-tint(--bs-success); } display: flex; align-items: center; gap: 10px;
&.o_fp_msg_warning { @include fp-shop-tint(--bs-warning); } box-shadow: $fp-shadow-xs;
&.o_fp_msg_danger { @include fp-shop-tint(--bs-danger); }
&.o_fp_msg_info { @include fp-tone(--bs-info); }
&.o_fp_msg_success { @include fp-tone(--bs-success); }
&.o_fp_msg_warning { @include fp-tone(--bs-warning); }
&.o_fp_msg_danger { @include fp-tone(--bs-danger); }
} }
// ---------- KPI strip ----------------------------------------------------
// =========================================================================
// KPI strip — gradient tiles, big numbers
// =========================================================================
.o_fp_kpi_strip { .o_fp_kpi_strip {
display: grid; display: grid;
grid-template-columns: repeat(auto-fit, minmax(150px, 1fr)); grid-template-columns: repeat(auto-fit, minmax(170px, 1fr));
gap: 10px; gap: 12px;
@media (max-width: 600px) { @media (max-width: 600px) {
// 2 columns on phone so 6 tiles wrap to 3 rows instead of 6 stacked
grid-template-columns: repeat(2, 1fr); grid-template-columns: repeat(2, 1fr);
gap: 6px; gap: 8px;
} }
} }
.o_fp_kpi { .o_fp_kpi {
position: relative; position: relative;
padding: 12px 14px; overflow: hidden;
border: 1px solid var(--bs-border-color); padding: 18px 18px 16px;
border-radius: 10px; border-radius: $fp-radius-lg;
background-color: var(--o-view-background-color, var(--bs-body-bg)); background-color: $fp-surface-raised;
border: 1px solid $fp-border;
box-shadow: $fp-shadow-sm;
transition: transform $fp-dur $fp-ease,
box-shadow $fp-dur $fp-ease,
border-color $fp-dur $fp-ease;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 2px; gap: 4px;
transition: border-color 120ms ease, box-shadow 120ms ease; min-height: 104px;
@media (max-width: 600px) {
padding: 8px 10px; // Accent bar along the top edge
.o_fp_kpi_value { font-size: 1.4rem !important; } &::before {
.o_fp_kpi_label { font-size: 0.72rem !important; } content: "";
> .fa { font-size: 1rem !important; top: 8px; right: 8px; } position: absolute;
top: 0; left: 0; right: 0; height: 3px;
background-image: linear-gradient(90deg, transparent, currentColor, transparent);
opacity: 0.7;
} }
// Soft tone overlay
&::after {
content: "";
position: absolute;
inset: 0;
opacity: 0.35;
pointer-events: none;
z-index: 0;
}
@media (hover: hover) {
&:hover {
transform: translateY(-2px);
box-shadow: $fp-shadow-md;
}
}
> * { position: relative; z-index: 1; }
> .fa { > .fa {
position: absolute; position: absolute;
right: 12px; right: 14px; top: 14px;
top: 12px; font-size: 1.5rem;
font-size: 1.4rem; opacity: 0.28;
opacity: 0.45; color: currentColor;
} }
.o_fp_kpi_value { font-size: 1.8rem; font-weight: 700; line-height: 1; }
.o_fp_kpi_label { font-size: 0.85rem; color: var(--bs-secondary-color); text-transform: uppercase; letter-spacing: 0.03em; }
&.o_fp_kpi_info { border-left: 4px solid var(--bs-info); } .o_fp_kpi_value {
&.o_fp_kpi_success { border-left: 4px solid var(--bs-success); } font-size: $fp-font-kpi;
&.o_fp_kpi_warning { border-left: 4px solid var(--bs-warning); } font-weight: 800;
line-height: 1;
letter-spacing: -0.02em;
font-variant-numeric: tabular-nums;
color: var(--bs-body-color);
}
.o_fp_kpi_label {
font-size: $fp-font-xs;
color: var(--bs-secondary-color);
text-transform: uppercase;
letter-spacing: 0.06em;
font-weight: 600;
}
&.o_fp_kpi_info { color: var(--bs-info); &::after { @include fp-grad(--bs-info); } }
&.o_fp_kpi_success { color: var(--bs-success); &::after { @include fp-grad(--bs-success); } }
&.o_fp_kpi_warning { color: var(--bs-warning); &::after { @include fp-grad(--bs-warning); } }
&.o_fp_kpi_danger { &.o_fp_kpi_danger {
border-left: 4px solid var(--bs-danger); color: var(--bs-danger);
background-color: color-mix(in srgb, var(--bs-danger) 5%, var(--o-view-background-color, var(--bs-body-bg))); border-color: color-mix(in srgb, var(--bs-danger) 45%, var(--bs-border-color));
&::after { @include fp-grad(--bs-danger, 22%, 8%); }
.o_fp_kpi_value { color: var(--bs-danger); } .o_fp_kpi_value { color: var(--bs-danger); }
} }
&.o_fp_kpi_muted { border-left: 4px solid var(--bs-border-color); opacity: 0.8; } &.o_fp_kpi_muted { color: var(--bs-secondary-color); &::after { opacity: 0; } }
@media (max-width: 600px) {
min-height: 84px;
padding: 12px 12px 10px;
.o_fp_kpi_value { font-size: 1.6rem; }
.o_fp_kpi_label { font-size: 0.68rem; }
> .fa { font-size: 1rem; top: 10px; right: 10px; }
}
} }
// ---------- Active WO banner --------------------------------------------
// =========================================================================
// Active WO banner — green gradient "you're running this right now"
// =========================================================================
.o_fp_active_wo { .o_fp_active_wo {
position: relative;
overflow: hidden;
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
gap: 14px; gap: 14px;
padding: 12px 16px; padding: 16px 20px;
border: 1px solid color-mix(in srgb, var(--bs-success) 40%, var(--bs-border-color)); border-radius: $fp-radius-lg;
background-color: color-mix(in srgb, var(--bs-success) 7%, var(--o-view-background-color, var(--bs-body-bg))); border: 1px solid color-mix(in srgb, var(--bs-success) 45%, $fp-border);
border-radius: 10px; box-shadow: $fp-shadow-sm;
@include fp-grad(--bs-success, 14%, 4%);
@media (max-width: 600px) { @media (max-width: 600px) {
flex-direction: column; flex-direction: column; align-items: stretch; gap: 10px;
align-items: stretch; padding: 14px; border-radius: $fp-radius-md;
gap: 8px; > .btn { width: 100%; min-height: $fp-touch-min; }
padding: 10px 12px;
> .btn { width: 100%; min-height: 44px; }
} }
} }
.o_fp_active_wo_left { .o_fp_active_wo_left {
display: flex; display: flex; align-items: center; gap: 14px;
align-items: center; }
gap: 12px; .o_fp_active_wo_title {
font-weight: 700; font-size: $fp-font-lg; letter-spacing: -0.01em;
}
.o_fp_active_wo_meta {
color: var(--bs-secondary-color); font-size: $fp-font-sm; margin-top: 2px;
} }
.o_fp_active_wo_title { font-weight: 600; font-size: 1.05rem; }
.o_fp_active_wo_meta { color: var(--bs-secondary-color); font-size: 0.9rem; }
.o_fp_active_wo_pulse { .o_fp_active_wo_pulse {
width: 12px; height: 12px; border-radius: 50%; flex-shrink: 0;
width: 14px; height: 14px; border-radius: 50%;
background-color: var(--bs-success); background-color: var(--bs-success);
box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 60%, transparent); animation: o_fp_pulse 1.4s ease-in-out infinite;
animation: o_fp_pulse 1.4s infinite;
} }
@keyframes o_fp_pulse { @keyframes o_fp_pulse {
0% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 60%, transparent); } 0% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 60%, transparent); }
70% { box-shadow: 0 0 0 10px color-mix(in srgb, var(--bs-success) 0%, transparent); } 70% { box-shadow: 0 0 0 12px color-mix(in srgb, var(--bs-success) 0%, transparent); }
100% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 0%, transparent); } 100% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 0%, transparent); }
} }
// ---------- Dashboard layout --------------------------------------------
// =========================================================================
// Dashboard layout
// =========================================================================
.o_fp_tablet_dashboard { .o_fp_tablet_dashboard {
display: grid; display: grid;
grid-template-columns: minmax(0, 1.1fr) minmax(0, 1fr); grid-template-columns: minmax(0, 1.15fr) minmax(0, 1fr);
gap: 14px; gap: 16px;
@media (max-width: 1100px) { grid-template-columns: 1fr; } @media (max-width: 1100px) { grid-template-columns: 1fr; }
@media (max-width: 600px) { gap: 8px; } @media (max-width: 600px) { gap: 10px; }
} }
.o_fp_right_col { .o_fp_right_col {
display: flex; display: flex; flex-direction: column; gap: 16px;
flex-direction: column; @media (max-width: 600px) { gap: 10px; }
gap: 14px;
@media (max-width: 600px) { gap: 8px; }
} }
// ---------- Panel (reusable card) ---------------------------------------
// =========================================================================
// Panels (reusable card)
// =========================================================================
.o_fp_panel { .o_fp_panel {
background-color: var(--o-view-background-color, var(--bs-body-bg)); background-color: $fp-surface-raised;
border: 1px solid var(--bs-border-color); border: 1px solid $fp-border;
border-radius: 12px; border-radius: $fp-radius-lg;
padding: 14px 16px; padding: 18px 20px;
box-shadow: $fp-shadow-xs;
@media (max-width: 600px) { @media (max-width: 600px) {
padding: 10px 12px; padding: 12px 14px; border-radius: $fp-radius-md;
border-radius: 10px;
} }
} }
.o_fp_panel_head { .o_fp_panel_head {
display: flex; display: flex; justify-content: space-between; align-items: center;
justify-content: space-between; margin-bottom: 14px;
align-items: center; padding-bottom: 10px;
margin-bottom: 10px; border-bottom: 1px solid $fp-border;
padding-bottom: 8px;
border-bottom: 1px dashed var(--bs-border-color); h3 {
h3 { font-size: 1.05rem; font-weight: 600; margin: 0; } font-size: $fp-font-lg;
font-weight: 700;
margin: 0;
display: inline-flex; align-items: center; gap: 10px;
letter-spacing: -0.01em;
}
} }
.o_fp_panel_count { .o_fp_panel_count {
background-color: color-mix(in srgb, var(--bs-body-color) 6%, transparent); min-width: 34px;
color: var(--bs-body-color); padding: 3px 12px;
border-radius: 999px; border-radius: $fp-radius-pill;
padding: 1px 10px; font-weight: 700;
font-size: 0.85rem; font-size: $fp-font-sm;
font-weight: 600; font-variant-numeric: tabular-nums;
} @include fp-tone(--bs-body-color, 8%);
.o_fp_empty {
padding: 14px;
text-align: center;
color: var(--bs-secondary-color);
font-size: 0.95rem;
} }
// ---------- My Queue list -----------------------------------------------
// =========================================================================
// Empty state
// =========================================================================
.o_fp_empty {
padding: 32px 16px;
text-align: center;
color: var(--bs-secondary-color);
font-size: $fp-font-base;
i.fa {
display: block;
font-size: 2.25rem;
margin-bottom: 10px;
opacity: 0.55;
}
}
// =========================================================================
// My Queue
// =========================================================================
.o_fp_queue_list { .o_fp_queue_list {
list-style: none; list-style: none; margin: 0; padding: 0;
margin: 0; padding: 0; display: flex; flex-direction: column; gap: 10px;
display: flex;
flex-direction: column;
gap: 8px;
} }
.o_fp_queue_row { .o_fp_queue_row {
display: grid; display: grid;
grid-template-columns: 40px 1fr auto; grid-template-columns: 44px 1fr auto;
align-items: center; align-items: center;
gap: 10px; gap: 12px;
padding: 10px 12px; padding: 12px 14px;
border: 1px solid var(--bs-border-color); border: 1px solid $fp-border;
border-radius: 8px; border-radius: $fp-radius-md;
transition: background-color 120ms ease, border-color 120ms ease; background-color: $fp-surface;
min-height: 56px; // comfortable tap zone min-height: 64px;
transition: background-color $fp-dur $fp-ease,
border-color $fp-dur $fp-ease,
transform $fp-dur-fast $fp-ease,
box-shadow $fp-dur $fp-ease;
@media (hover: hover) { @media (hover: hover) {
&:hover { &:hover {
background-color: color-mix(in srgb, var(--o-action) 7%, var(--o-view-background-color, var(--bs-body-bg))); transform: translateY(-1px);
border-color: color-mix(in srgb, var(--o-action) 40%, var(--bs-border-color)); background-color: color-mix(in srgb, var(--o-action) 4%, $fp-surface);
border-color: $fp-border-accent;
box-shadow: $fp-shadow-sm;
} }
} }
@media (max-width: 600px) { @media (max-width: 600px) {
grid-template-columns: 32px 1fr; grid-template-columns: 36px 1fr;
// Action buttons drop to their own row padding: 10px 12px;
.o_fp_queue_actions { .o_fp_queue_actions {
grid-column: 1 / -1; grid-column: 1 / -1;
justify-content: flex-end; justify-content: stretch;
margin-top: 4px; margin-top: 6px;
> .btn { flex: 1; } > .btn { flex: 1; }
} }
} }
} }
.o_fp_queue_body { cursor: pointer; } .o_fp_queue_body { cursor: pointer; min-width: 0; }
.o_fp_queue_actions { .o_fp_queue_label {
display: flex;
gap: 6px;
align-items: center;
.btn { min-height: 44px; } // touch-friendly
}
.o_fp_queue_label { font-weight: 600; }
.o_fp_queue_desc { font-size: 0.88rem; color: var(--bs-secondary-color); }
.o_fp_queue_pri {
display: flex;
align-items: center;
justify-content: center;
width: 36px; height: 36px;
border-radius: 8px;
font-weight: 700; font-weight: 700;
font-size: 0.82rem; font-size: $fp-font-base;
&[data-priority="high"] { background-color: color-mix(in srgb, var(--bs-danger) 15%, transparent); color: var(--bs-danger); } color: var(--bs-body-color);
&[data-priority="med"] { background-color: color-mix(in srgb, var(--bs-warning) 15%, transparent); color: var(--bs-warning); }
&[data-priority="low"] { background-color: color-mix(in srgb, var(--bs-body-color) 8%, transparent); color: var(--bs-secondary-color); }
} }
.o_fp_queue_desc {
font-size: $fp-font-sm;
color: var(--bs-secondary-color);
margin-top: 2px;
overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
}
.o_fp_queue_actions {
display: flex; gap: 6px; align-items: center;
.btn { min-height: $fp-touch-min; font-weight: 600; border-radius: $fp-radius-md; }
}
.o_fp_queue_pri {
display: flex; align-items: center; justify-content: center;
width: 40px; height: 40px; border-radius: $fp-radius-md;
font-weight: 800;
font-size: 0.78rem;
letter-spacing: 0.02em;
// ---------- Bath tiles -------------------------------------------------- &[data-priority="high"] {
.o_fp_tile_grid { @include fp-grad(--bs-danger, 30%, 12%);
display: grid; color: var(--bs-danger);
grid-template-columns: repeat(auto-fit, minmax(180px, 1fr)); border: 1px solid color-mix(in srgb, var(--bs-danger) 40%, transparent);
gap: 10px; }
@media (max-width: 600px) { &[data-priority="med"] {
grid-template-columns: 1fr 1fr; @include fp-grad(--bs-warning, 28%, 10%);
gap: 6px; color: var(--bs-warning);
border: 1px solid color-mix(in srgb, var(--bs-warning) 40%, transparent);
}
&[data-priority="low"] {
background-color: color-mix(in srgb, var(--bs-body-color) 5%, transparent);
color: var(--bs-secondary-color);
border: 1px solid $fp-border;
} }
} }
.o_fp_tile {
border: 1px solid var(--bs-border-color);
border-radius: 10px;
padding: 10px 12px;
cursor: pointer;
transition: border-color 120ms ease, background-color 120ms ease;
&:hover { border-color: color-mix(in srgb, var(--o-action) 40%, var(--bs-border-color)); }
&[data-tone="danger"] { border-left: 4px solid var(--bs-danger); } // =========================================================================
&[data-tone="warning"] { border-left: 4px solid var(--bs-warning); } // Bath / Gate / Hold tiles + rows
&[data-tone="success"] { border-left: 4px solid var(--bs-success); } // =========================================================================
&[data-tone="info"] { border-left: 4px solid var(--bs-info); } .o_fp_tile_grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 12px;
@media (max-width: 600px) { grid-template-columns: 1fr 1fr; gap: 8px; }
} }
.o_fp_tile_title { font-weight: 600; margin-bottom: 2px; } .o_fp_tile {
.o_fp_tile_meta { font-size: 0.85rem; color: var(--bs-secondary-color); margin-bottom: 6px; } position: relative; overflow: hidden;
.o_fp_tile_chips { display: flex; flex-wrap: wrap; gap: 4px; } border: 1px solid $fp-border;
border-radius: $fp-radius-md;
padding: 14px 16px;
background-color: $fp-surface;
cursor: pointer;
transition: border-color $fp-dur $fp-ease,
transform $fp-dur-fast $fp-ease,
box-shadow $fp-dur $fp-ease;
// ---------- Chips ------------------------------------------------------- @media (hover: hover) {
&:hover {
transform: translateY(-2px);
border-color: $fp-border-accent;
box-shadow: $fp-shadow-sm;
}
}
// Accent stripe by tone
&::before {
content: "";
position: absolute;
left: 0; top: 0; bottom: 0; width: 4px;
}
&[data-tone="success"]::before { background: var(--bs-success); }
&[data-tone="info"]::before { background: var(--bs-info); }
&[data-tone="warning"]::before { background: var(--bs-warning); }
&[data-tone="danger"]::before { background: var(--bs-danger); }
&[data-tone="muted"]::before { background: $fp-border-strong; }
}
.o_fp_tile_title {
font-weight: 700; font-size: $fp-font-base;
margin-bottom: 4px; padding-left: 6px;
}
.o_fp_tile_meta {
font-size: $fp-font-sm; color: var(--bs-secondary-color);
margin-bottom: 8px; padding-left: 6px;
}
.o_fp_tile_chips {
display: flex; flex-wrap: wrap; gap: 6px; padding-left: 6px;
}
// =========================================================================
// Status chips
// =========================================================================
.o_fp_chip { .o_fp_chip {
display: inline-block; display: inline-flex;
padding: 1px 8px; align-items: center;
border-radius: 999px; gap: 4px;
font-size: 0.78rem; padding: 3px 10px;
font-weight: 600; border-radius: $fp-radius-pill;
font-size: 0.72rem;
font-weight: 700;
text-transform: uppercase; text-transform: uppercase;
letter-spacing: 0.03em; letter-spacing: 0.04em;
line-height: 1.4;
&.o_fp_chip_info { @include fp-shop-tint(--bs-info); } &.o_fp_chip_info { @include fp-tone(--bs-info); }
&.o_fp_chip_success { @include fp-shop-tint(--bs-success); } &.o_fp_chip_success { @include fp-tone(--bs-success); }
&.o_fp_chip_warning { @include fp-shop-tint(--bs-warning); } &.o_fp_chip_warning { @include fp-tone(--bs-warning); }
&.o_fp_chip_danger { @include fp-shop-tint(--bs-danger); } &.o_fp_chip_danger { @include fp-tone(--bs-danger); }
&.o_fp_chip_muted { &.o_fp_chip_muted {
background-color: color-mix(in srgb, var(--bs-body-color) 6%, transparent); background-color: color-mix(in srgb, var(--bs-body-color) 6%, transparent);
color: var(--bs-secondary-color); color: var(--bs-secondary-color);
border: 1px solid var(--bs-border-color); border: 1px solid $fp-border;
} }
} }
// ---------- Bake / Gate / Hold rows -------------------------------------
// =========================================================================
// Bake / Gate / Hold rows — accent-left banding by state
// =========================================================================
.o_fp_bake_list { .o_fp_bake_list {
list-style: none; list-style: none; margin: 0; padding: 0;
margin: 0; padding: 0; display: flex; flex-direction: column; gap: 10px;
display: flex;
flex-direction: column;
gap: 8px;
} }
.o_fp_bake_row { .o_fp_bake_row {
position: relative;
display: grid; display: grid;
grid-template-columns: 1fr auto auto; grid-template-columns: 1fr auto auto;
align-items: center; align-items: center;
gap: 10px; gap: 12px;
padding: 8px 10px; padding: 12px 14px 12px 18px;
border: 1px solid var(--bs-border-color); border: 1px solid $fp-border;
border-radius: 8px; border-radius: $fp-radius-md;
min-height: 56px; background-color: $fp-surface;
min-height: 64px;
transition: transform $fp-dur-fast $fp-ease, box-shadow $fp-dur $fp-ease;
@media (hover: hover) { &:hover { box-shadow: $fp-shadow-sm; } }
// Left accent bar by state
&::before {
content: "";
position: absolute;
top: 8px; bottom: 8px; left: 6px;
width: 4px; border-radius: 4px;
background-color: $fp-border-strong;
}
&[data-state="awaiting_bake"]::before, &[data-state="pending"]::before {
background-color: var(--bs-warning);
}
&[data-state="bake_in_progress"]::before, &[data-state="under_review"]::before {
background-color: var(--bs-info);
}
&[data-state="missed_window"]::before, &[data-state="fail"]::before, &[data-state="on_hold"]::before {
background-color: var(--bs-danger);
}
&[data-state="missed_window"], &[data-state="fail"], &[data-state="on_hold"] {
background-color: color-mix(in srgb, var(--bs-danger) 6%, $fp-surface);
}
&[data-state="baked"]::before, &[data-state="pass"]::before {
background-color: var(--bs-success);
}
@media (max-width: 600px) { @media (max-width: 600px) {
grid-template-columns: 1fr; grid-template-columns: 1fr;
gap: 6px; gap: 8px;
.o_fp_bake_time, .o_fp_bake_actions { .o_fp_bake_time, .o_fp_bake_actions { justify-self: stretch; }
justify-self: stretch;
}
.o_fp_bake_actions { .o_fp_bake_actions {
display: flex; display: flex; gap: 6px;
gap: 6px; .btn { flex: 1; min-height: $fp-touch-min; }
.btn { flex: 1; min-height: 44px; }
} }
} }
&[data-state="awaiting_bake"], &[data-state="pending"] {
border-left: 4px solid var(--bs-warning);
}
&[data-state="bake_in_progress"], &[data-state="under_review"] {
border-left: 4px solid var(--bs-info);
}
&[data-state="missed_window"], &[data-state="fail"], &[data-state="on_hold"] {
border-left: 4px solid var(--bs-danger);
background-color: color-mix(in srgb, var(--bs-danger) 4%, transparent);
}
&[data-state="baked"], &[data-state="pass"] {
border-left: 4px solid var(--bs-success);
}
} }
.o_fp_bake_name { font-weight: 600; } .o_fp_bake_name { font-weight: 700; font-size: $fp-font-base; }
.o_fp_bake_meta { font-size: 0.85rem; } .o_fp_bake_meta { font-size: $fp-font-sm; margin-top: 2px; }
.o_fp_bake_actions { display: flex; gap: 6px; } .o_fp_bake_actions {
display: flex; gap: 6px;
.btn {
min-height: $fp-touch-min; font-weight: 600;
border-radius: $fp-radius-md; padding: 6px 14px;
}
}
// ---------- Footer ------------------------------------------------------
// =========================================================================
// Footer
// =========================================================================
.o_fp_tablet_footer { .o_fp_tablet_footer {
text-align: right; text-align: right;
padding-top: 8px; padding-top: 10px;
border-top: 1px dashed var(--bs-border-color); border-top: 1px dashed $fp-border;
color: var(--bs-secondary-color);
font-size: $fp-font-xs;
} }
} }
// ----------------------------------------------------------------------------- // =============================================================================
// Large QR scan input — friendly to tablet keyboards / wedge scanners // Big QR scan input + action button — shared outside .o_fp_tablet scope too
// ----------------------------------------------------------------------------- // =============================================================================
.o_fp_scan_input { .o_fp_scan_input {
flex: 1 1 auto; flex: 1 1 auto;
min-height: 44px; min-height: 48px;
padding: 8px 14px; padding: 10px 16px;
font-size: 1.05rem; font-size: $fp-font-base;
border: 2px solid var(--bs-border-color); border: 2px solid var(--bs-border-color);
border-radius: 8px; border-radius: $fp-radius-md;
background-color: var(--bs-body-bg); background-color: var(--bs-body-bg);
color: var(--bs-body-color); color: var(--bs-body-color);
transition: border-color $fp-dur $fp-ease, box-shadow $fp-dur $fp-ease;
&:focus { &:focus { @include fp-focus-ring; border-color: var(--o-action); }
outline: none;
border-color: var(--o-action);
box-shadow: 0 0 0 3px color-mix(in srgb, var(--o-action) 25%, transparent);
}
&::placeholder { color: var(--bs-secondary-color); } &::placeholder { color: var(--bs-secondary-color); }
} }
// -----------------------------------------------------------------------------
// Big touch-friendly action button
// -----------------------------------------------------------------------------
.o_fp_big_button { .o_fp_big_button {
min-height: 44px; min-height: 48px;
min-width: 100px; min-width: 110px;
padding: 8px 18px; padding: 10px 22px;
font-size: 1rem; font-size: $fp-font-base;
font-weight: 500; font-weight: 600;
border-radius: 8px; border-radius: $fp-radius-md;
border: 1px solid var(--o-action); border: 1px solid var(--o-action);
background-color: var(--o-action); background-color: var(--o-action);
color: var(--o-we-text-on-action, #fff); color: var(--o-we-text-on-action, #fff);
cursor: pointer; cursor: pointer;
transition: filter 120ms ease, transform 80ms ease; transition: filter $fp-dur-fast $fp-ease, transform $fp-dur-fast $fp-ease;
&:hover:not(:disabled) { filter: brightness(1.05); } &:hover:not(:disabled) { filter: brightness(1.06); }
&:active:not(:disabled) { transform: translateY(1px); } &:active:not(:disabled) { transform: scale(0.97); }
&:disabled { opacity: 0.6; cursor: not-allowed; } &:disabled { opacity: 0.55; cursor: not-allowed; }
} }

View File

@@ -1,269 +1,483 @@
// ============================================================================= // =============================================================================
// Fusion Plating — Manager Dashboard styles // Fusion Plating — Manager Desk
// Copyright 2026 Nexa Systems Inc. // Copyright 2026 Nexa Systems Inc.
// License OPL-1 (Odoo Proprietary License v1.0) // License OPL-1 (Odoo Proprietary License v1.0)
// //
// Inherits the tablet's theme variables (--bs-*, --o-action) so it flips // Shares design tokens + panel / KPI / chip classes from the Tablet Station
// light/dark without media queries. Shares .o_fp_kpi, .o_fp_chip, // SCSS. Only the manager-specific components live here: hero banner with
// .o_fp_panel, .o_fp_empty, .o_fp_tablet_message from the tablet SCSS. // live dot, 3-column workload grid, richer MO cards, gradient avatars.
// ============================================================================= // =============================================================================
// Touch device — stop hover states from sticking on tap @import "./fp_shopfloor_tokens";
// Touch-device hover suppression
@media (hover: none) { @media (hover: none) {
.o_fp_manager .o_fp_mgr_card:hover, .o_fp_manager .o_fp_mgr_card:hover,
.o_fp_manager .o_fp_team_card:hover { .o_fp_manager .o_fp_team_card:hover {
background-color: inherit !important; background-color: inherit !important;
border-color: var(--bs-border-color) !important; border-color: var(--bs-border-color) !important;
transform: none !important;
} }
} }
.o_fp_manager { .o_fp_manager {
background-color: var(--o-view-background-color, var(--bs-body-bg)); background-color: $fp-surface-sunken;
color: var(--bs-body-color); color: var(--bs-body-color);
min-height: 100%; min-height: 100%;
padding: 20px 24px; padding: 24px 28px;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 14px; gap: 18px;
// iPad portrait & small tablet @media (max-width: 900px) { padding: 16px; gap: 14px; }
@media (max-width: 900px) { @media (max-width: 600px) { padding: 10px 10px 16px; gap: 10px; }
padding: 14px 12px;
gap: 10px;
}
// Phone
@media (max-width: 600px) {
padding: 10px 8px;
gap: 8px;
}
// =========================================================================
// Hero banner — gradient wash, live dot, action cluster
// =========================================================================
.o_fp_manager_header { .o_fp_manager_header {
display: flex; position: relative;
justify-content: space-between; overflow: hidden;
align-items: center; border-radius: $fp-radius-lg;
padding: 20px 24px;
background-color: $fp-surface-raised;
border: 1px solid $fp-border;
box-shadow: $fp-shadow-sm;
&::before {
content: "";
position: absolute;
inset: 0;
background-image:
radial-gradient(circle at 0% 50%,
color-mix(in srgb, var(--o-action) 22%, transparent) 0%,
transparent 55%),
radial-gradient(circle at 100% 0%,
color-mix(in srgb, var(--bs-info) 18%, transparent) 0%,
transparent 55%);
pointer-events: none;
}
display: grid;
grid-template-columns: 1fr auto;
gap: 16px; gap: 16px;
flex-wrap: wrap; align-items: center;
> * { position: relative; z-index: 1; }
@media (max-width: 600px) { @media (max-width: 600px) {
gap: 8px; grid-template-columns: 1fr;
// Stack title and actions on phone gap: 10px;
flex-direction: column; padding: 14px;
align-items: stretch;
.o_fp_manager_head_actions {
width: 100%;
> .btn { flex: 1; }
}
} }
} }
.o_fp_manager_title { .o_fp_manager_title {
font-size: 1.5rem; font-size: $fp-font-hero;
font-weight: 600; font-weight: 700;
display: inline-flex; letter-spacing: -0.01em;
align-items: center; display: inline-flex; align-items: center; gap: 10px;
@media (max-width: 600px) { font-size: 1.15rem; }
} }
// Small breathing dot that pulses while a poll is in flight. // Small breathing dot — green at rest, brighter pulse while polling
// At rest: soft green. While fetching: brighter, animating.
.o_fp_live_dot { .o_fp_live_dot {
display: inline-block; display: inline-block;
width: 10px; width: 11px; height: 11px; border-radius: 50%;
height: 10px;
border-radius: 50%;
background-color: color-mix(in srgb, var(--bs-success) 75%, transparent); background-color: color-mix(in srgb, var(--bs-success) 75%, transparent);
transition: background-color 160ms ease; transition: background-color $fp-dur $fp-ease;
box-shadow: 0 0 0 0 transparent;
&[data-active="y"] { &[data-active="y"] {
background-color: var(--bs-success); background-color: var(--bs-success);
animation: o_fp_live_pulse 1.0s ease-in-out infinite; animation: o_fp_live_pulse 1.1s ease-in-out infinite;
} }
} }
@keyframes o_fp_live_pulse { @keyframes o_fp_live_pulse {
0% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 55%, transparent); } 0% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 60%, transparent); }
70% { box-shadow: 0 0 0 8px color-mix(in srgb, var(--bs-success) 0%, transparent); } 70% { box-shadow: 0 0 0 10px color-mix(in srgb, var(--bs-success) 0%, transparent); }
100% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 0%, transparent); } 100% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--bs-success) 0%, transparent); }
} }
.o_fp_manager_subtitle { .o_fp_manager_subtitle {
font-size: 0.95rem; font-size: $fp-font-sm;
color: var(--bs-secondary-color); color: var(--bs-secondary-color);
margin-top: 4px;
}
.o_fp_manager_head_actions {
display: flex; gap: 8px; align-items: center;
.btn {
min-height: $fp-touch-min;
padding: 8px 16px;
border-radius: $fp-radius-md;
font-weight: 600;
}
@media (max-width: 600px) {
width: 100%; flex-wrap: wrap;
> .btn { flex: 1; }
}
} }
// 3-column grid: Unassigned | In Progress | Team
// =========================================================================
// KPI strip — same token system as the tablet
// =========================================================================
.o_fp_kpi_strip {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(170px, 1fr));
gap: 12px;
@media (max-width: 600px) {
grid-template-columns: repeat(2, 1fr);
gap: 8px;
}
}
.o_fp_kpi {
position: relative; overflow: hidden;
padding: 18px 18px 16px;
border-radius: $fp-radius-lg;
background-color: $fp-surface-raised;
border: 1px solid $fp-border;
box-shadow: $fp-shadow-sm;
transition: transform $fp-dur $fp-ease, box-shadow $fp-dur $fp-ease;
display: flex; flex-direction: column; gap: 4px;
min-height: 104px;
&::before {
content: "";
position: absolute;
top: 0; left: 0; right: 0; height: 3px;
background-image: linear-gradient(90deg, transparent, currentColor, transparent);
opacity: 0.7;
}
&::after {
content: "";
position: absolute; inset: 0; opacity: 0.35; pointer-events: none;
}
> * { position: relative; z-index: 1; }
@media (hover: hover) {
&:hover { transform: translateY(-2px); box-shadow: $fp-shadow-md; }
}
> .fa {
position: absolute;
right: 14px; top: 14px;
font-size: 1.5rem;
opacity: 0.28;
}
.o_fp_kpi_value {
font-size: $fp-font-kpi;
font-weight: 800;
line-height: 1;
letter-spacing: -0.02em;
font-variant-numeric: tabular-nums;
color: var(--bs-body-color);
}
.o_fp_kpi_label {
font-size: $fp-font-xs;
color: var(--bs-secondary-color);
text-transform: uppercase;
letter-spacing: 0.06em;
font-weight: 600;
}
&.o_fp_kpi_info { color: var(--bs-info); &::after { @include fp-grad(--bs-info); } }
&.o_fp_kpi_success { color: var(--bs-success); &::after { @include fp-grad(--bs-success); } }
&.o_fp_kpi_warning { color: var(--bs-warning); &::after { @include fp-grad(--bs-warning); } }
&.o_fp_kpi_danger {
color: var(--bs-danger);
border-color: color-mix(in srgb, var(--bs-danger) 45%, $fp-border);
&::after { @include fp-grad(--bs-danger, 22%, 8%); }
.o_fp_kpi_value { color: var(--bs-danger); }
}
&.o_fp_kpi_muted { color: var(--bs-secondary-color); &::after { opacity: 0; } }
@media (max-width: 600px) {
min-height: 84px; padding: 12px 12px 10px;
.o_fp_kpi_value { font-size: 1.6rem; }
.o_fp_kpi_label { font-size: 0.68rem; }
> .fa { font-size: 1rem; top: 10px; right: 10px; }
}
}
// =========================================================================
// Flash message (shares tablet styling)
// =========================================================================
.o_fp_tablet_message {
padding: 12px 16px;
border-radius: $fp-radius-md;
font-size: $fp-font-base;
font-weight: 500;
display: flex; align-items: center; gap: 10px;
box-shadow: $fp-shadow-xs;
&.o_fp_msg_info { @include fp-tone(--bs-info); }
&.o_fp_msg_success { @include fp-tone(--bs-success); }
&.o_fp_msg_warning { @include fp-tone(--bs-warning); }
&.o_fp_msg_danger { @include fp-tone(--bs-danger); }
}
// =========================================================================
// 3-column workload grid
// =========================================================================
.o_fp_manager_grid { .o_fp_manager_grid {
display: grid; display: grid;
grid-template-columns: minmax(0, 1.2fr) minmax(0, 1.2fr) minmax(0, 0.8fr); grid-template-columns: minmax(0, 1.15fr) minmax(0, 1.15fr) minmax(0, 0.85fr);
gap: 14px; gap: 16px;
}
// iPad landscape @media (max-width: 1280px) {
@media (max-width: 1280px) {
.o_fp_manager_grid {
grid-template-columns: 1fr 1fr; grid-template-columns: 1fr 1fr;
.o_fp_panel_team { grid-column: span 2; }
} }
.o_fp_panel_team { grid-column: span 2; } @media (max-width: 900px) {
} grid-template-columns: 1fr; gap: 10px;
// iPad portrait / phone landscape .o_fp_panel_team { grid-column: auto; }
@media (max-width: 900px) {
.o_fp_manager_grid {
grid-template-columns: 1fr;
gap: 10px;
} }
.o_fp_panel_team { grid-column: auto; }
} }
.o_fp_panel_unassigned {
border-left: 4px solid var(--bs-warning); // =========================================================================
// Panels with coloured top accent
// =========================================================================
.o_fp_panel {
position: relative;
overflow: hidden;
background-color: $fp-surface-raised;
border: 1px solid $fp-border;
border-radius: $fp-radius-lg;
padding: 18px 20px;
box-shadow: $fp-shadow-sm;
@media (max-width: 600px) { padding: 12px 14px; border-radius: $fp-radius-md; }
&::before {
content: "";
position: absolute; top: 0; left: 0; right: 0; height: 4px;
background-color: $fp-border-strong;
}
} }
.o_fp_panel_active { .o_fp_panel_unassigned::before {
border-left: 4px solid var(--bs-success); background-image: linear-gradient(90deg,
var(--bs-warning), color-mix(in srgb, var(--bs-warning) 50%, transparent));
} }
.o_fp_panel_team { .o_fp_panel_active::before {
border-left: 4px solid var(--bs-info); background-image: linear-gradient(90deg,
var(--bs-success), color-mix(in srgb, var(--bs-success) 50%, transparent));
}
.o_fp_panel_team::before {
background-image: linear-gradient(90deg,
var(--bs-info), color-mix(in srgb, var(--bs-info) 50%, transparent));
} }
.o_fp_panel_head {
display: flex; justify-content: space-between; align-items: center;
margin-bottom: 14px; padding-bottom: 12px;
border-bottom: 1px solid $fp-border;
h3 {
font-size: $fp-font-lg;
font-weight: 700;
margin: 0;
display: inline-flex; align-items: center; gap: 10px;
letter-spacing: -0.01em;
}
}
.o_fp_panel_count {
min-width: 34px;
padding: 3px 12px;
border-radius: $fp-radius-pill;
font-weight: 700;
font-size: $fp-font-sm;
font-variant-numeric: tabular-nums;
background-color: color-mix(in srgb, var(--bs-body-color) 8%, transparent);
color: var(--bs-body-color);
border: 1px solid $fp-border;
}
.o_fp_empty {
padding: 32px 16px;
text-align: center;
color: var(--bs-secondary-color);
i.fa { display: block; font-size: 2.25rem; opacity: 0.55; margin-bottom: 8px; }
}
// =========================================================================
// MO card list (Unassigned + In Progress columns)
// =========================================================================
.o_fp_mgr_card_list { .o_fp_mgr_card_list {
display: flex; display: flex; flex-direction: column; gap: 10px;
flex-direction: column;
gap: 10px;
} }
.o_fp_mgr_card { .o_fp_mgr_card {
border: 1px solid var(--bs-border-color); position: relative;
border-radius: 10px; overflow: hidden;
transition: border-color 120ms ease, box-shadow 120ms ease; border: 1px solid $fp-border;
border-radius: $fp-radius-md;
background-color: $fp-surface;
transition: border-color $fp-dur $fp-ease,
box-shadow $fp-dur $fp-ease,
transform $fp-dur-fast $fp-ease;
@media (hover: hover) {
&:hover {
border-color: $fp-border-accent;
box-shadow: $fp-shadow-sm;
transform: translateY(-1px);
}
}
&[data-priority="2"] { &[data-priority="2"] {
border-left: 4px solid var(--bs-danger); border-color: color-mix(in srgb, var(--bs-danger) 50%, $fp-border);
background-color: color-mix(in srgb, var(--bs-danger) 4%, transparent); &::before {
content: "";
position: absolute; top: 0; left: 0; bottom: 0; width: 4px;
background: var(--bs-danger);
}
} }
&[data-priority="1"] { &[data-priority="1"] {
border-left: 4px solid var(--bs-warning); &::before {
} content: "";
position: absolute; top: 0; left: 0; bottom: 0; width: 4px;
&:hover { background: var(--bs-warning);
border-color: color-mix(in srgb, var(--o-action) 40%, var(--bs-border-color)); }
} }
} }
.o_fp_mgr_card_head { .o_fp_mgr_card_head {
display: flex; display: flex; justify-content: space-between; align-items: center;
justify-content: space-between; padding: 12px 14px; cursor: pointer; gap: 10px;
align-items: center; min-height: 60px;
padding: 10px 14px; @media (max-width: 600px) { flex-wrap: wrap; padding: 12px; }
cursor: pointer;
gap: 10px;
min-height: 56px; // comfortable touch zone
@media (max-width: 600px) {
flex-wrap: wrap;
padding: 10px 12px;
}
} }
.o_fp_mgr_card_title { .o_fp_mgr_card_title {
font-weight: 600; font-weight: 700; font-size: $fp-font-base;
font-size: 1rem; letter-spacing: -0.01em;
} }
.o_fp_mgr_card_sub { .o_fp_mgr_card_sub {
color: var(--bs-secondary-color); color: var(--bs-secondary-color); font-size: $fp-font-sm;
font-size: 0.88rem; margin-top: 3px;
margin-top: 2px;
} }
.o_fp_mgr_card_chips { .o_fp_mgr_card_chips {
display: flex; display: flex; gap: 6px; flex-wrap: wrap;
gap: 6px;
flex-wrap: wrap;
} }
.o_fp_mgr_card_body { .o_fp_mgr_card_body {
border-top: 1px dashed var(--bs-border-color); border-top: 1px dashed $fp-border;
padding: 10px 14px; padding: 12px 14px;
display: flex; display: flex; flex-direction: column; gap: 10px;
flex-direction: column;
gap: 8px;
background-color: color-mix(in srgb, var(--bs-body-color) 2%, transparent); background-color: color-mix(in srgb, var(--bs-body-color) 2%, transparent);
border-bottom-left-radius: 10px;
border-bottom-right-radius: 10px;
} }
// Per-WO row inside the expanded card
.o_fp_mgr_wo_row { .o_fp_mgr_wo_row {
display: grid; display: grid;
grid-template-columns: 1fr auto auto auto auto; grid-template-columns: 1fr auto auto auto auto;
gap: 8px; gap: 8px;
align-items: center; align-items: center;
padding: 6px 0; padding: 8px 10px;
font-size: 0.92rem; background-color: $fp-surface;
border-bottom: 1px dashed color-mix(in srgb, var(--bs-body-color) 6%, transparent); border: 1px solid $fp-border;
border-radius: $fp-radius-sm;
font-size: $fp-font-sm;
&:last-child { border-bottom: 0; } @media (max-width: 1400px) {
}
@media (max-width: 1400px) {
.o_fp_mgr_wo_row {
grid-template-columns: 1fr auto auto; grid-template-columns: 1fr auto auto;
// Worker picker takes its own row
.o_fp_mgr_picker:nth-of-type(1) { grid-column: 1 / -1; } .o_fp_mgr_picker:nth-of-type(1) { grid-column: 1 / -1; }
// Tank picker also takes its own row
.o_fp_mgr_picker:nth-of-type(2) { grid-column: 1 / -1; } .o_fp_mgr_picker:nth-of-type(2) { grid-column: 1 / -1; }
} }
} @media (max-width: 600px) {
// Phone — everything stacks full-width
@media (max-width: 600px) {
.o_fp_mgr_wo_row {
grid-template-columns: 1fr; grid-template-columns: 1fr;
padding: 10px 0;
gap: 6px; gap: 6px;
.o_fp_mgr_picker { max-width: 100% !important; width: 100%; } .o_fp_mgr_picker { max-width: 100% !important; width: 100%; }
.btn { min-height: 44px; } .btn { min-height: $fp-touch-min; }
} }
} }
.o_fp_mgr_wo_info { .o_fp_mgr_wo_info {
min-width: 0; min-width: 0;
overflow: hidden; overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
text-overflow: ellipsis; font-weight: 600;
white-space: nowrap;
} }
.o_fp_mgr_picker { .o_fp_mgr_picker {
min-width: 120px; min-width: 130px; max-width: 200px;
max-width: 180px; min-height: 38px;
min-height: 38px; // readable tap target padding: 4px 10px;
border-radius: $fp-radius-sm;
background-color: $fp-surface;
border: 1px solid $fp-border;
color: var(--bs-body-color);
font-size: $fp-font-sm;
&:focus { @include fp-focus-ring; border-color: var(--o-action); }
} }
// Team grid
// =========================================================================
// Status chips (shared styles in tablet scss, but redefined in case
// manager is loaded standalone)
// =========================================================================
.o_fp_chip {
display: inline-flex;
align-items: center;
padding: 3px 10px;
border-radius: $fp-radius-pill;
font-size: 0.72rem;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.04em;
&.o_fp_chip_info { @include fp-tone(--bs-info); }
&.o_fp_chip_success { @include fp-tone(--bs-success); }
&.o_fp_chip_warning { @include fp-tone(--bs-warning); }
&.o_fp_chip_danger { @include fp-tone(--bs-danger); }
&.o_fp_chip_muted {
background-color: color-mix(in srgb, var(--bs-body-color) 6%, transparent);
color: var(--bs-secondary-color);
border: 1px solid $fp-border;
}
}
// =========================================================================
// Team column — avatar grid
// =========================================================================
.o_fp_team_grid { .o_fp_team_grid {
display: grid; display: flex; flex-direction: column; gap: 10px;
grid-template-columns: 1fr;
gap: 8px;
} }
.o_fp_team_card { .o_fp_team_card {
display: flex; position: relative;
display: grid;
grid-template-columns: 56px 1fr;
align-items: center; align-items: center;
gap: 12px; gap: 14px;
padding: 10px 12px; padding: 12px 14px;
border: 1px solid var(--bs-border-color); border: 1px solid $fp-border;
border-radius: 10px; border-radius: $fp-radius-md;
background-color: $fp-surface;
cursor: pointer; cursor: pointer;
transition: border-color 120ms ease, background-color 120ms ease; min-height: 72px;
transition: border-color $fp-dur $fp-ease,
box-shadow $fp-dur $fp-ease,
transform $fp-dur-fast $fp-ease;
&:hover { @media (hover: hover) {
border-color: color-mix(in srgb, var(--o-action) 40%, var(--bs-border-color)); &:hover {
background-color: color-mix(in srgb, var(--o-action) 6%, transparent); border-color: $fp-border-accent;
box-shadow: $fp-shadow-sm;
transform: translateY(-1px);
}
} }
} }
.o_fp_team_avatar { .o_fp_team_avatar {
width: 44px; width: 48px; height: 48px;
height: 44px;
border-radius: 50%; border-radius: 50%;
object-fit: cover; object-fit: cover;
border: 2px solid var(--bs-border-color); border: 2px solid $fp-border;
} background-color: color-mix(in srgb, var(--o-action) 15%, $fp-surface);
.o_fp_team_info {
flex: 1;
min-width: 0;
} }
.o_fp_team_info { flex: 1; min-width: 0; }
.o_fp_team_name { .o_fp_team_name {
font-weight: 600; font-weight: 700;
font-size: 0.95rem; font-size: $fp-font-base;
letter-spacing: -0.01em;
} }
.o_fp_team_load { .o_fp_team_load {
display: flex; display: flex; gap: 6px; margin-top: 6px;
gap: 6px;
margin-top: 4px;
} }
} }

View File

@@ -3,61 +3,62 @@
// Copyright 2026 Nexa Systems Inc. // Copyright 2026 Nexa Systems Inc.
// License OPL-1 (Odoo Proprietary License v1.0) // License OPL-1 (Odoo Proprietary License v1.0)
// //
// THEME AWARENESS // Modernised 2026-04: gradient column headers, card depth, theme-safe
// --------------- // using shared design tokens.
// All colours come from CSS custom properties (Bootstrap / Odoo tokens) so
// the dashboard renders correctly in BOTH light and dark mode.
//
// background: var(--bs-body-bg)
// surface: var(--o-view-background-color)
// foreground: var(--bs-body-color)
// muted text: var(--bs-secondary-color)
// border: var(--bs-border-color)
// primary: var(--o-action)
// ============================================================================= // =============================================================================
@import "./fp_shopfloor_tokens";
.o_fp_plant_overview { .o_fp_plant_overview {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
height: 100%; height: 100%;
min-height: 0; min-height: 0;
background: var(--o-view-background-color, var(--bs-body-bg)); background: $fp-surface-sunken;
padding: 0; padding: 0;
} }
// ---- Header ----------------------------------------------------------------- // ---- Header -----------------------------------------------------------------
.o_fp_po_header { .o_fp_po_header {
position: relative;
overflow: hidden;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between; justify-content: space-between;
flex-wrap: wrap; flex-wrap: wrap;
gap: 12px; gap: 12px;
padding: 16px 20px; padding: 20px 24px;
background: var(--bs-body-bg); background-color: $fp-surface-raised;
border-bottom: 1px solid var(--bs-border-color); border-bottom: 1px solid $fp-border;
box-shadow: 0 1px 3px color-mix(in srgb, var(--bs-body-color) 6%, transparent); box-shadow: $fp-shadow-xs;
.o_fp_po_header_left { &::before {
display: flex; content: "";
align-items: center; position: absolute; inset: 0;
background-image:
radial-gradient(circle at 0% 50%,
color-mix(in srgb, var(--o-action) 18%, transparent) 0%,
transparent 55%);
pointer-events: none;
} }
> * { position: relative; z-index: 1; }
.o_fp_po_header_left { display: flex; align-items: center; }
.o_fp_po_title { .o_fp_po_title {
margin: 0; margin: 0;
font-size: 1.3rem; font-size: $fp-font-xl;
font-weight: 700; font-weight: 700;
letter-spacing: -0.01em;
color: var(--bs-body-color); color: var(--bs-body-color);
} }
.o_fp_po_refresh_ts { .o_fp_po_refresh_ts { font-size: $fp-font-xs; }
font-size: 0.8rem;
}
.o_fp_po_header_right { .o_fp_po_header_right {
display: flex; display: flex; align-items: center; gap: 10px;
align-items: center;
gap: 10px;
} }
} }
@@ -137,35 +138,49 @@
max-width: 320px; max-width: 320px;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
background: var(--bs-body-bg); background-color: $fp-surface-raised;
border-radius: 10px; border: 1px solid $fp-border;
box-shadow: 0 1px 4px color-mix(in srgb, var(--bs-body-color) 8%, transparent); border-radius: $fp-radius-lg;
max-height: calc(100vh - 140px); box-shadow: $fp-shadow-sm;
max-height: calc(100vh - 160px);
overflow: hidden;
} }
.o_fp_po_col_header { .o_fp_po_col_header {
position: relative;
overflow: hidden;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between; justify-content: space-between;
padding: 12px 14px; padding: 14px 16px;
border-bottom: 2px solid var(--bs-border-color); border-bottom: 1px solid $fp-border;
background: var(--bs-tertiary-bg); background-color: $fp-surface-sunken;
border-radius: 10px 10px 0 0;
// Subtle gradient stripe at the top — turns the header into a "tab"
&::before {
content: "";
position: absolute; top: 0; left: 0; right: 0; height: 3px;
background-image: linear-gradient(90deg,
var(--o-action),
color-mix(in srgb, var(--o-action) 30%, transparent));
}
> * { position: relative; z-index: 1; }
.o_fp_po_col_name { .o_fp_po_col_name {
font-weight: 700; font-weight: 700;
font-size: 0.9rem; font-size: $fp-font-sm;
color: var(--bs-body-color); color: var(--bs-body-color);
text-transform: uppercase; text-transform: uppercase;
letter-spacing: 0.3px; letter-spacing: 0.05em;
} }
.o_fp_po_col_count { .o_fp_po_col_count {
background: var(--bs-secondary-color); @include fp-tone(--o-action, 14%);
color: #fff; font-weight: 700;
font-size: 0.75rem; font-size: 0.72rem;
min-width: 24px; min-width: 26px;
text-align: center; padding: 2px 10px;
border-radius: $fp-radius-pill;
} }
} }
@@ -187,20 +202,22 @@
// ---- Card ------------------------------------------------------------------- // ---- Card -------------------------------------------------------------------
.o_fp_po_card { .o_fp_po_card {
background: var(--bs-body-bg); background-color: $fp-surface;
border-width: 1px; border: 1px solid $fp-border;
border-style: solid; border-radius: $fp-radius-md;
border-color: $border-color; padding: 12px 14px;
border-radius: 8px; margin-bottom: 10px;
padding: 10px 12px;
margin-bottom: 8px;
cursor: grab; cursor: grab;
box-shadow: 0 1px 4px rgba(0, 0, 0, 0.08); box-shadow: $fp-shadow-xs;
transition: box-shadow 0.15s, transform 0.1s, opacity 0.15s; transition: box-shadow $fp-dur $fp-ease,
transform $fp-dur-fast $fp-ease,
opacity $fp-dur $fp-ease,
border-color $fp-dur $fp-ease;
&:hover { &:hover {
box-shadow: 0 3px 10px rgba(0, 0, 0, 0.15); box-shadow: $fp-shadow-md;
transform: translateY(-1px); transform: translateY(-2px);
border-color: $fp-border-accent;
border-color: darken($border-color, 10%); border-color: darken($border-color, 10%);
} }

View File

@@ -88,8 +88,8 @@
<span class="o_fp_panel_count"><t t-esc="state.overview.unassigned.length"/></span> <span class="o_fp_panel_count"><t t-esc="state.overview.unassigned.length"/></span>
</div> </div>
<div t-if="!state.overview.unassigned.length" class="o_fp_empty"> <div t-if="!state.overview.unassigned.length" class="o_fp_empty">
<i class="fa fa-check-circle text-success me-2"/> <i class="fa fa-check-circle text-success"/>
Every active WO has a worker assigned. <div>Every active WO has a worker assigned.</div>
</div> </div>
<div class="o_fp_mgr_card_list" t-if="state.overview.unassigned.length"> <div class="o_fp_mgr_card_list" t-if="state.overview.unassigned.length">
<t t-foreach="state.overview.unassigned" t-as="card" t-key="card.mo_id"> <t t-foreach="state.overview.unassigned" t-as="card" t-key="card.mo_id">
@@ -176,7 +176,8 @@
<span class="o_fp_panel_count"><t t-esc="state.overview.active.length"/></span> <span class="o_fp_panel_count"><t t-esc="state.overview.active.length"/></span>
</div> </div>
<div t-if="!state.overview.active.length" class="o_fp_empty"> <div t-if="!state.overview.active.length" class="o_fp_empty">
Nothing running right now. <i class="fa fa-moon-o"/>
<div>Nothing running right now.</div>
</div> </div>
<div class="o_fp_mgr_card_list" t-if="state.overview.active.length"> <div class="o_fp_mgr_card_list" t-if="state.overview.active.length">
<t t-foreach="state.overview.active" t-as="card" t-key="card.mo_id"> <t t-foreach="state.overview.active" t-as="card" t-key="card.mo_id">
@@ -242,7 +243,8 @@
<span class="o_fp_panel_count"><t t-esc="state.overview.team.length"/></span> <span class="o_fp_panel_count"><t t-esc="state.overview.team.length"/></span>
</div> </div>
<div t-if="!state.overview.team.length" class="o_fp_empty"> <div t-if="!state.overview.team.length" class="o_fp_empty">
No operators configured. <i class="fa fa-users"/>
<div>No operators configured yet.</div>
</div> </div>
<div class="o_fp_team_grid" t-if="state.overview.team.length"> <div class="o_fp_team_grid" t-if="state.overview.team.length">
<t t-foreach="state.overview.team" t-as="member" t-key="member.user_id"> <t t-foreach="state.overview.team" t-as="member" t-key="member.user_id">

View File

@@ -122,8 +122,8 @@
</div> </div>
<div t-if="!state.overview.my_queue.length" <div t-if="!state.overview.my_queue.length"
class="o_fp_empty"> class="o_fp_empty">
<i class="fa fa-check-circle text-success me-2"/> <i class="fa fa-check-circle text-success"/>
All caught up. <div>All caught up — nothing waiting on you.</div>
</div> </div>
<ul class="o_fp_queue_list" t-if="state.overview.my_queue.length"> <ul class="o_fp_queue_list" t-if="state.overview.my_queue.length">
<t t-foreach="state.overview.my_queue" t-as="row" t-key="row.id"> <t t-foreach="state.overview.my_queue" t-as="row" t-key="row.id">
@@ -167,7 +167,8 @@
<span class="o_fp_panel_count"><t t-esc="state.overview.baths.length"/></span> <span class="o_fp_panel_count"><t t-esc="state.overview.baths.length"/></span>
</div> </div>
<div t-if="!state.overview.baths.length" class="o_fp_empty"> <div t-if="!state.overview.baths.length" class="o_fp_empty">
No baths configured. <i class="fa fa-flask"/>
<div>No baths configured.</div>
</div> </div>
<div class="o_fp_tile_grid" t-if="state.overview.baths.length"> <div class="o_fp_tile_grid" t-if="state.overview.baths.length">
<t t-foreach="state.overview.baths" t-as="b" t-key="b.id"> <t t-foreach="state.overview.baths" t-as="b" t-key="b.id">
@@ -202,7 +203,8 @@
<span class="o_fp_panel_count"><t t-esc="state.overview.bake_windows.length"/></span> <span class="o_fp_panel_count"><t t-esc="state.overview.bake_windows.length"/></span>
</div> </div>
<div t-if="!state.overview.bake_windows.length" class="o_fp_empty"> <div t-if="!state.overview.bake_windows.length" class="o_fp_empty">
No bakes pending. <i class="fa fa-fire"/>
<div>No bakes pending.</div>
</div> </div>
<ul class="o_fp_bake_list" t-if="state.overview.bake_windows.length"> <ul class="o_fp_bake_list" t-if="state.overview.bake_windows.length">
<t t-foreach="state.overview.bake_windows" t-as="bw" t-key="bw.id"> <t t-foreach="state.overview.bake_windows" t-as="bw" t-key="bw.id">
@@ -251,7 +253,8 @@
<span class="o_fp_panel_count"><t t-esc="state.overview.gates.length"/></span> <span class="o_fp_panel_count"><t t-esc="state.overview.gates.length"/></span>
</div> </div>
<div t-if="!state.overview.gates.length" class="o_fp_empty"> <div t-if="!state.overview.gates.length" class="o_fp_empty">
No pending first-piece inspections. <i class="fa fa-flag-checkered"/>
<div>No pending first-piece inspections.</div>
</div> </div>
<ul class="o_fp_bake_list" t-if="state.overview.gates.length"> <ul class="o_fp_bake_list" t-if="state.overview.gates.length">
<t t-foreach="state.overview.gates" t-as="g" t-key="g.id"> <t t-foreach="state.overview.gates" t-as="g" t-key="g.id">