Restyling light pastel: tema blanco/azul/verde + Inter self-hosted
- theme.css reescrito: paleta light (sky/emerald accents, slate neutrals), tokens de spacing/typography/radii/shadows, @font-face Inter + JetBrains Mono variable woff2 self-hosted. - ops-shell: header blanco con nav active=accent-soft + status pill pastel. - run-timeline: bubbles emerald-100 (user) / blue-100 (bot) sobre blanco. - home-dashboard: helpers cssVar + withAlpha, 6 charts coordinados a paleta pastel (--chart-blue/green/purple/orange/pink/gray). - 8 CRUDs (users, products, orders, conversations, aliases, recommendations, quantities, takeovers, settings, debug) migrados de hex hardcoded oscuros a var(--*). - modal.js + toast.js refactor a vars con fallbacks; modal blanco con shadow-lg y soft icon backgrounds. - test-panel: aliases :host apuntan a globals en vez de override dark. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -19,46 +19,46 @@ class ProductsCrud extends HTMLElement {
|
||||
:host { display:block; height:100%; padding:16px; overflow:hidden; }
|
||||
* { box-sizing:border-box; font-family:system-ui,Segoe UI,Roboto,Arial; }
|
||||
.container { display:grid; grid-template-columns:1fr 1fr; gap:16px; height:100%; }
|
||||
.panel { background:#121823; border:1px solid #1e2a3a; border-radius:10px; padding:16px; overflow:hidden; display:flex; flex-direction:column; }
|
||||
.panel-title { font-size:14px; font-weight:700; color:#8aa0b5; text-transform:uppercase; letter-spacing:.4px; margin-bottom:12px; }
|
||||
.panel { background:var(--panel); border:1px solid var(--border); border-radius:10px; padding:16px; overflow:hidden; display:flex; flex-direction:column; }
|
||||
.panel-title { font-size:14px; font-weight:700; color:var(--text-muted); text-transform:uppercase; letter-spacing:.4px; margin-bottom:12px; }
|
||||
|
||||
.toolbar { display:flex; gap:8px; margin-bottom:12px; }
|
||||
input, select { background:#0f1520; color:#e7eef7; border:1px solid #253245; border-radius:8px; padding:8px 12px; font-size:13px; }
|
||||
input:focus, select:focus { outline:none; border-color:#1f6feb; }
|
||||
input, select { background:var(--panel-2); color:var(--text); border:1px solid var(--border-hi); border-radius:8px; padding:8px 12px; font-size:13px; }
|
||||
input:focus, select:focus { outline:none; border-color:var(--accent); }
|
||||
input { flex:1; }
|
||||
button { cursor:pointer; background:#1f6feb; color:#fff; border:none; border-radius:8px; padding:8px 16px; font-size:13px; }
|
||||
button:hover { background:#1a5fd0; }
|
||||
button { cursor:pointer; background:var(--accent); color:#fff; border:none; border-radius:8px; padding:8px 16px; font-size:13px; }
|
||||
button:hover { background:var(--accent-hover); }
|
||||
button:disabled { opacity:.5; cursor:not-allowed; }
|
||||
button.secondary { background:#253245; }
|
||||
button.secondary:hover { background:#2d3e52; }
|
||||
button.secondary { background:var(--border-hi); }
|
||||
button.secondary:hover { background:var(--border-hi); }
|
||||
|
||||
.list { flex:1; overflow-y:auto; }
|
||||
.item { background:#0f1520; border:1px solid #253245; border-radius:8px; padding:12px; margin-bottom:8px; cursor:pointer; transition:all .15s; user-select:none; }
|
||||
.item:hover { border-color:#1f6feb; }
|
||||
.item.active { border-color:#1f6feb; background:#111b2a; }
|
||||
.item.selected { border-color:#2ecc71; background:#0f2a1a; }
|
||||
.item-name { font-weight:600; color:#e7eef7; margin-bottom:4px; }
|
||||
.item-meta { font-size:12px; color:#8aa0b5; }
|
||||
.item-price { color:#2ecc71; font-weight:600; }
|
||||
.item { background:var(--panel-2); border:1px solid var(--border-hi); border-radius:8px; padding:12px; margin-bottom:8px; cursor:pointer; transition:all .15s; user-select:none; }
|
||||
.item:hover { border-color:var(--accent); }
|
||||
.item.active { border-color:var(--accent); background:var(--accent-soft); }
|
||||
.item.selected { border-color:var(--ok); background:var(--ok-soft); }
|
||||
.item-name { font-weight:600; color:var(--text); margin-bottom:4px; }
|
||||
.item-meta { font-size:12px; color:var(--text-muted); }
|
||||
.item-price { color:var(--ok); font-weight:600; }
|
||||
|
||||
.detail { flex:1; overflow-y:auto; }
|
||||
.detail-empty { color:#8aa0b5; text-align:center; padding:40px; }
|
||||
.detail-empty { color:var(--text-muted); text-align:center; padding:40px; }
|
||||
.field { margin-bottom:16px; }
|
||||
.field label { display:block; font-size:12px; color:#8aa0b5; margin-bottom:4px; text-transform:uppercase; letter-spacing:.4px; }
|
||||
.field-value { background:#0f1520; border:1px solid #253245; border-radius:8px; padding:10px 12px; color:#e7eef7; font-size:13px; }
|
||||
.field label { display:block; font-size:12px; color:var(--text-muted); margin-bottom:4px; text-transform:uppercase; letter-spacing:.4px; }
|
||||
.field-value { background:var(--panel-2); border:1px solid var(--border-hi); border-radius:8px; padding:10px 12px; color:var(--text); font-size:13px; }
|
||||
.field-value.json { font-family:monospace; font-size:11px; white-space:pre-wrap; max-height:200px; overflow-y:auto; }
|
||||
|
||||
.stats { display:flex; gap:16px; margin-bottom:16px; }
|
||||
.stat { background:#0f1520; border:1px solid #253245; border-radius:8px; padding:12px; flex:1; text-align:center; cursor:pointer; transition:all .15s; }
|
||||
.stat:hover { border-color:#1f6feb; }
|
||||
.stat.active { border-color:#1f6feb; background:#111b2a; }
|
||||
.stat-value { font-size:24px; font-weight:700; color:#1f6feb; }
|
||||
.stat-label { font-size:11px; color:#8aa0b5; text-transform:uppercase; margin-top:4px; }
|
||||
.stat { background:var(--panel-2); border:1px solid var(--border-hi); border-radius:8px; padding:12px; flex:1; text-align:center; cursor:pointer; transition:all .15s; }
|
||||
.stat:hover { border-color:var(--accent); }
|
||||
.stat.active { border-color:var(--accent); background:var(--accent-soft); }
|
||||
.stat-value { font-size:24px; font-weight:700; color:var(--accent); }
|
||||
.stat-label { font-size:11px; color:var(--text-muted); text-transform:uppercase; margin-top:4px; }
|
||||
|
||||
.loading { text-align:center; padding:40px; color:#8aa0b5; }
|
||||
.badge { display:inline-block; padding:2px 8px; border-radius:999px; font-size:11px; background:#253245; color:#8aa0b5; margin-left:8px; }
|
||||
.badge.stock { background:#0f2a1a; color:#2ecc71; }
|
||||
.badge.nostock { background:#241214; color:#e74c3c; }
|
||||
.loading { text-align:center; padding:40px; color:var(--text-muted); }
|
||||
.badge { display:inline-block; padding:2px 8px; border-radius:999px; font-size:11px; background:var(--border-hi); color:var(--text-muted); margin-left:8px; }
|
||||
.badge.stock { background:var(--ok-soft); color:var(--ok); }
|
||||
.badge.nostock { background:var(--err-soft); color:var(--err); }
|
||||
</style>
|
||||
|
||||
<div class="container">
|
||||
@@ -272,7 +272,7 @@ class ProductsCrud extends HTMLElement {
|
||||
|
||||
// Mostrar unidad actual si está definida
|
||||
const unit = item.sell_unit || item.payload?._sell_unit_override;
|
||||
const unitBadge = unit ? `<span class="badge" style="background:#1a3a5c;color:#7eb8e7;">${unit === 'unit' ? 'Unidad' : 'Kg'}</span>` : '';
|
||||
const unitBadge = unit ? `<span class="badge" style="background:var(--accent-soft);color:var(--accent-hover);">${unit === 'unit' ? 'Unidad' : 'Kg'}</span>` : '';
|
||||
|
||||
el.innerHTML = `
|
||||
<div class="item-name">${item.name || "Sin nombre"} ${stockBadge} ${unitBadge}</div>
|
||||
@@ -328,7 +328,7 @@ class ProductsCrud extends HTMLElement {
|
||||
} catch (err) {
|
||||
console.error("[products-crud] Error in renderDetail:", err);
|
||||
const detail = this.shadowRoot.getElementById("detail");
|
||||
detail.innerHTML = `<div class="detail-empty" style="color:#e74c3c;">Error: ${err.message}</div>`;
|
||||
detail.innerHTML = `<div class="detail-empty" style="color:var(--err);">Error: ${err.message}</div>`;
|
||||
}
|
||||
|
||||
// Scroll detail panel to top
|
||||
@@ -449,12 +449,12 @@ class ProductsCrud extends HTMLElement {
|
||||
<div class="field">
|
||||
<label>Unidad de venta</label>
|
||||
<div style="display:flex;gap:8px;align-items:center;">
|
||||
<select id="sellUnit" style="background:#0f1520;color:#e7eef7;border:1px solid #253245;border-radius:8px;padding:8px 12px;font-size:13px;">
|
||||
<select id="sellUnit" style="background:var(--panel-2);color:var(--text);border:1px solid var(--border-hi);border-radius:8px;padding:8px 12px;font-size:13px;">
|
||||
<option value="kg" ${currentUnit === "kg" ? "selected" : ""}>Por peso (kg)</option>
|
||||
<option value="unit" ${currentUnit === "unit" ? "selected" : ""}>Por unidad</option>
|
||||
</select>
|
||||
</div>
|
||||
<div style="font-size:11px;color:#8aa0b5;margin-top:4px;">
|
||||
<div style="font-size:11px;color:var(--text-muted);margin-top:4px;">
|
||||
Define si este producto se vende por peso o por unidad
|
||||
</div>
|
||||
</div>
|
||||
@@ -464,16 +464,16 @@ class ProductsCrud extends HTMLElement {
|
||||
${categoriesArray.length > 0
|
||||
? categoriesArray.map(cat => `
|
||||
<span class="category-tag" data-category="${this.escapeHtml(cat)}"
|
||||
style="display:inline-flex;align-items:center;gap:4px;background:#1a3a5c;color:#7eb8e7;padding:4px 8px;border-radius:6px;font-size:12px;">
|
||||
style="display:inline-flex;align-items:center;gap:4px;background:var(--accent-soft);color:var(--accent-hover);padding:4px 8px;border-radius:6px;font-size:12px;">
|
||||
${this.escapeHtml(cat)}
|
||||
<span class="remove-cat" style="cursor:pointer;font-weight:bold;opacity:0.7;">×</span>
|
||||
</span>
|
||||
`).join("")
|
||||
: '<span style="color:#8aa0b5;font-size:12px;">Sin categorías</span>'
|
||||
: '<span style="color:var(--text-muted);font-size:12px;">Sin categorías</span>'
|
||||
}
|
||||
</div>
|
||||
<div style="display:flex;gap:8px;align-items:center;">
|
||||
<select id="addCategorySelect" style="background:#0f1520;color:#e7eef7;border:1px solid #253245;border-radius:8px;padding:8px 12px;font-size:13px;flex:1;">
|
||||
<select id="addCategorySelect" style="background:var(--panel-2);color:var(--text);border:1px solid var(--border-hi);border-radius:8px;padding:8px 12px;font-size:13px;flex:1;">
|
||||
<option value="">-- Agregar categoría --</option>
|
||||
${this.getAllCategories().filter(c => !categoriesArray.includes(c)).map(cat =>
|
||||
`<option value="${this.escapeHtml(cat)}">${this.escapeHtml(cat)}</option>`
|
||||
@@ -489,7 +489,7 @@ class ProductsCrud extends HTMLElement {
|
||||
<div class="field">
|
||||
<div style="display:flex;gap:8px;align-items:center;">
|
||||
<button id="saveProduct" style="flex:1;padding:10px;">Guardar cambios</button>
|
||||
<span id="saveStatus" style="font-size:12px;color:#2ecc71;"></span>
|
||||
<span id="saveStatus" style="font-size:12px;color:var(--ok);"></span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="field">
|
||||
@@ -533,7 +533,7 @@ class ProductsCrud extends HTMLElement {
|
||||
const tag = document.createElement("span");
|
||||
tag.className = "category-tag";
|
||||
tag.dataset.category = categoryName;
|
||||
tag.style = "display:inline-flex;align-items:center;gap:4px;background:#1a3a5c;color:#7eb8e7;padding:4px 8px;border-radius:6px;font-size:12px;";
|
||||
tag.style = "display:inline-flex;align-items:center;gap:4px;background:var(--accent-soft);color:var(--accent-hover);padding:4px 8px;border-radius:6px;font-size:12px;";
|
||||
tag.innerHTML = `${this.escapeHtml(categoryName)}<span class="remove-cat" style="cursor:pointer;font-weight:bold;opacity:0.7;">×</span>`;
|
||||
|
||||
// Bind remove
|
||||
@@ -545,7 +545,7 @@ class ProductsCrud extends HTMLElement {
|
||||
};
|
||||
|
||||
// Remover el mensaje "Sin categorías" si existe
|
||||
const emptyMsg = container.querySelector('span[style*="color:#8aa0b5"]');
|
||||
const emptyMsg = container.querySelector('span[style*="color:var(--text-muted)"]');
|
||||
if (emptyMsg) emptyMsg.remove();
|
||||
|
||||
container.appendChild(tag);
|
||||
@@ -652,8 +652,8 @@ class ProductsCrud extends HTMLElement {
|
||||
detail.innerHTML = `
|
||||
<div class="field">
|
||||
<label>Productos seleccionados</label>
|
||||
<div class="field-value" style="color:#2ecc71;font-weight:600;">${count} productos</div>
|
||||
<div style="font-size:11px;color:#8aa0b5;margin-top:4px;">${names}${moreText}</div>
|
||||
<div class="field-value" style="color:var(--ok);font-weight:600;">${count} productos</div>
|
||||
<div style="font-size:11px;color:var(--text-muted);margin-top:4px;">${names}${moreText}</div>
|
||||
<div style="font-size:11px;margin-top:4px;">
|
||||
<span class="badge stock" style="margin-left:0;">${inStockCount} en stock</span>
|
||||
<span class="badge nostock">${count - inStockCount} sin stock</span>
|
||||
@@ -662,26 +662,26 @@ class ProductsCrud extends HTMLElement {
|
||||
<div class="field">
|
||||
<label>Unidad de venta (para todos)</label>
|
||||
<div style="display:flex;gap:8px;align-items:center;">
|
||||
<select id="sellUnit" style="background:#0f1520;color:#e7eef7;border:1px solid #253245;border-radius:8px;padding:8px 12px;font-size:13px;flex:1;">
|
||||
<select id="sellUnit" style="background:var(--panel-2);color:var(--text);border:1px solid var(--border-hi);border-radius:8px;padding:8px 12px;font-size:13px;flex:1;">
|
||||
<option value="kg">Por peso (kg)</option>
|
||||
<option value="unit">Por unidad</option>
|
||||
</select>
|
||||
<button id="saveUnit" style="padding:8px 16px;">Aplicar</button>
|
||||
</div>
|
||||
<div style="font-size:11px;color:#8aa0b5;margin-top:4px;">
|
||||
<div style="font-size:11px;color:var(--text-muted);margin-top:4px;">
|
||||
Se aplicará a todos los productos seleccionados
|
||||
</div>
|
||||
</div>
|
||||
<div class="field">
|
||||
<label>Agregar categoría (para todos)</label>
|
||||
<div style="display:flex;gap:8px;align-items:center;">
|
||||
<select id="addCategorySelect" style="background:#0f1520;color:#e7eef7;border:1px solid #253245;border-radius:8px;padding:8px 12px;font-size:13px;flex:1;">
|
||||
<select id="addCategorySelect" style="background:var(--panel-2);color:var(--text);border:1px solid var(--border-hi);border-radius:8px;padding:8px 12px;font-size:13px;flex:1;">
|
||||
<option value="">-- Seleccionar categoría --</option>
|
||||
${this.getAllCategories().map(cat => `<option value="${this.escapeHtml(cat)}">${this.escapeHtml(cat)}</option>`).join("")}
|
||||
</select>
|
||||
<button id="addCategory" style="padding:8px 16px;">Agregar</button>
|
||||
</div>
|
||||
<div style="font-size:11px;color:#8aa0b5;margin-top:4px;">
|
||||
<div style="font-size:11px;color:var(--text-muted);margin-top:4px;">
|
||||
Se agregará esta categoría a todos los productos seleccionados
|
||||
</div>
|
||||
</div>
|
||||
|
||||
Reference in New Issue
Block a user