feat(admin): improve uploads and dashboard UX

This commit is contained in:
2026-03-23 16:07:11 +01:00
parent 45bca018b5
commit ef9763692d
22 changed files with 2531 additions and 212 deletions

View File

@@ -0,0 +1,788 @@
.mx-dashboard {
display: grid;
gap: 1.25rem;
padding-bottom: 1.5rem;
}
.mx-tabs {
display: grid;
gap: 1rem;
}
.mx-tab-toggle {
position: absolute;
opacity: 0;
pointer-events: none;
}
.mx-tab-nav {
display: inline-flex;
gap: 0.55rem;
width: fit-content;
max-width: 100%;
padding: 0.45rem;
overflow-x: auto;
border: 1px solid rgba(73, 208, 200, 0.12);
border-radius: 999px;
background: rgba(6, 21, 28, 0.72);
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.03);
}
.mx-tab-label {
display: inline-flex;
align-items: center;
justify-content: center;
min-height: 2.55rem;
padding: 0.55rem 1rem;
border-radius: 999px;
color: var(--mx-text-muted);
font-size: 0.88rem;
font-weight: 700;
letter-spacing: 0.03em;
white-space: nowrap;
cursor: pointer;
transition:
color 0.18s ease,
background-color 0.18s ease,
transform 0.18s ease,
box-shadow 0.18s ease;
}
.mx-tab-label:hover {
color: #f8ffff;
transform: translateY(-1px);
}
#mx-tab-overview:checked ~ .mx-tab-nav label[for="mx-tab-overview"],
#mx-tab-analytics:checked ~ .mx-tab-nav label[for="mx-tab-analytics"],
#mx-tab-admin:checked ~ .mx-tab-nav label[for="mx-tab-admin"] {
color: #f9ffff;
background: linear-gradient(135deg, rgba(73, 208, 200, 0.22), rgba(126, 166, 255, 0.18));
box-shadow:
inset 0 1px 0 rgba(255, 255, 255, 0.04),
0 10px 24px rgba(1, 8, 11, 0.22);
}
.mx-tab-panels {
display: grid;
}
.mx-tab-panel {
display: none;
gap: 1rem;
animation: mx-dashboard-rise 0.35s ease both;
}
#mx-tab-overview:checked ~ .mx-tab-panels .mx-tab-panel--overview,
#mx-tab-analytics:checked ~ .mx-tab-panels .mx-tab-panel--analytics,
#mx-tab-admin:checked ~ .mx-tab-panels .mx-tab-panel--admin {
display: grid;
}
.mx-hero,
.mx-panel,
.mx-stat-card {
position: relative;
overflow: hidden;
border: 1px solid rgba(73, 208, 200, 0.14);
border-radius: 24px;
background:
linear-gradient(180deg, rgba(6, 21, 28, 0.94), rgba(5, 16, 22, 0.98)),
radial-gradient(circle at top right, rgba(126, 166, 255, 0.16), transparent 32%);
box-shadow:
inset 0 1px 0 rgba(255, 255, 255, 0.03),
0 24px 60px rgba(1, 8, 11, 0.38);
}
.mx-hero::before,
.mx-panel::before,
.mx-stat-card::before {
content: "";
position: absolute;
inset: 0;
background:
radial-gradient(circle at top left, rgba(73, 208, 200, 0.08), transparent 32%),
linear-gradient(135deg, rgba(255, 255, 255, 0.02), transparent 36%);
pointer-events: none;
}
.mx-hero,
.mx-main-grid,
.mx-split-grid,
.mx-stat-card,
.mx-panel,
.mx-source-card,
.mx-app-card {
animation: mx-dashboard-rise 0.6s ease both;
}
.mx-dashboard > *:nth-child(2) {
animation-delay: 0.06s;
}
.mx-dashboard > *:nth-child(3) {
animation-delay: 0.12s;
}
.mx-dashboard > *:nth-child(4) {
animation-delay: 0.18s;
}
.mx-dashboard > *:nth-child(5) {
animation-delay: 0.24s;
}
.mx-hero {
display: grid;
grid-template-columns: minmax(0, 1.2fr) minmax(320px, 0.8fr);
gap: 1.5rem;
padding: 1.7rem;
}
.mx-hero__copy,
.mx-hero__visual,
.mx-panel > *,
.mx-stat-card > * {
position: relative;
z-index: 1;
}
.mx-eyebrow {
margin: 0 0 0.45rem;
color: var(--mx-text-dim);
font-size: 0.76rem;
font-weight: 700;
letter-spacing: 0.18em;
text-transform: uppercase;
}
.mx-hero__title {
margin: 0;
color: #f7fffe;
font-size: clamp(1.8rem, 3vw, 2.8rem);
font-weight: 700;
line-height: 1.05;
}
.mx-hero__lead {
max-width: 52rem;
margin: 0.95rem 0 0;
color: var(--mx-text-muted);
font-size: 1rem;
line-height: 1.65;
}
.mx-hero__facts {
display: grid;
grid-template-columns: repeat(3, minmax(0, 1fr));
gap: 0.85rem;
margin-top: 1.4rem;
}
.mx-hero__fact {
padding: 0.95rem 1rem;
border: 1px solid rgba(73, 208, 200, 0.1);
border-radius: 18px;
background: rgba(6, 23, 28, 0.72);
backdrop-filter: blur(10px);
}
.mx-hero__fact-label,
.mx-panel__note,
.mx-source-card__metrics span,
.mx-bar-item__head span,
.mx-feed-item__meta time,
.mx-feed-item__note,
.mx-ring-card__legend-item span,
.mx-action-card span,
.mx-stat-card__caption,
.mx-app-card__head span {
color: var(--mx-text-dim);
}
.mx-hero__fact-label,
.mx-stat-card__label {
display: block;
font-size: 0.78rem;
font-weight: 600;
letter-spacing: 0.06em;
text-transform: uppercase;
}
.mx-hero__fact-value {
display: block;
margin-top: 0.4rem;
color: #f6fffe;
font-size: 1.05rem;
font-weight: 700;
line-height: 1.3;
}
.mx-hero__visual {
display: flex;
align-items: stretch;
}
.mx-ring-card {
display: grid;
gap: 1rem;
width: 100%;
min-height: 100%;
padding: 1.2rem;
border: 1px solid rgba(73, 208, 200, 0.1);
border-radius: 22px;
background:
linear-gradient(180deg, rgba(8, 27, 33, 0.84), rgba(5, 18, 24, 0.92)),
radial-gradient(circle at top, rgba(126, 166, 255, 0.18), transparent 38%);
}
.mx-ring-chart {
position: relative;
width: min(290px, 100%);
aspect-ratio: 1;
margin: 0 auto;
border-radius: 50%;
box-shadow:
inset 0 0 0 1px rgba(255, 255, 255, 0.04),
0 16px 50px rgba(0, 0, 0, 0.25);
}
.mx-ring-chart::before {
content: "";
position: absolute;
inset: 20%;
border-radius: 50%;
background:
radial-gradient(circle at 30% 30%, rgba(73, 208, 200, 0.16), transparent 34%),
linear-gradient(180deg, rgba(5, 18, 23, 0.96), rgba(4, 14, 18, 0.98));
border: 1px solid rgba(73, 208, 200, 0.08);
}
.mx-ring-chart__center {
position: absolute;
inset: 0;
display: grid;
align-content: center;
justify-items: center;
gap: 0.15rem;
text-align: center;
}
.mx-ring-chart__center span {
color: var(--mx-text-dim);
font-size: 0.72rem;
font-weight: 700;
letter-spacing: 0.14em;
text-transform: uppercase;
}
.mx-ring-chart__center strong {
color: #f8ffff;
font-size: clamp(1.5rem, 2vw, 2.2rem);
line-height: 1;
}
.mx-ring-card__legend {
display: grid;
gap: 0.75rem;
}
.mx-ring-card__legend-item {
display: grid;
grid-template-columns: 12px minmax(0, 1fr);
gap: 0.75rem;
align-items: start;
}
.mx-ring-card__legend-item strong,
.mx-bar-item__head strong,
.mx-feed-item strong,
.mx-action-card strong,
.mx-app-card__head h3 {
color: #f5fffe;
}
.mx-dot {
display: inline-block;
width: 12px;
height: 12px;
margin-top: 0.28rem;
border-radius: 999px;
box-shadow: 0 0 0 4px rgba(255, 255, 255, 0.03);
}
.mx-overview-grid {
display: grid;
grid-template-columns: repeat(4, minmax(0, 1fr));
gap: 1rem;
}
.mx-stat-card {
padding: 1.15rem 1.2rem;
}
.mx-stat-card__value {
display: block;
margin-top: 0.45rem;
color: #fbffff;
font-size: clamp(1.5rem, 2vw, 2rem);
line-height: 1;
}
.mx-stat-card__caption {
margin: 0.8rem 0 0;
line-height: 1.55;
}
.mx-stat-card--cyan {
background:
linear-gradient(180deg, rgba(7, 25, 32, 0.95), rgba(5, 16, 22, 0.98)),
radial-gradient(circle at top right, rgba(73, 208, 200, 0.18), transparent 40%);
}
.mx-stat-card--blue {
background:
linear-gradient(180deg, rgba(7, 25, 32, 0.95), rgba(5, 16, 22, 0.98)),
radial-gradient(circle at top right, rgba(126, 166, 255, 0.18), transparent 40%);
}
.mx-stat-card--amber {
background:
linear-gradient(180deg, rgba(7, 25, 32, 0.95), rgba(5, 16, 22, 0.98)),
radial-gradient(circle at top right, rgba(255, 200, 87, 0.16), transparent 40%);
}
.mx-stat-card--rose {
background:
linear-gradient(180deg, rgba(7, 25, 32, 0.95), rgba(5, 16, 22, 0.98)),
radial-gradient(circle at top right, rgba(255, 122, 162, 0.16), transparent 40%);
}
.mx-main-grid,
.mx-split-grid {
display: grid;
gap: 1rem;
}
.mx-main-grid {
grid-template-columns: minmax(0, 1.25fr) minmax(300px, 0.75fr);
}
.mx-split-grid {
grid-template-columns: repeat(2, minmax(0, 1fr));
}
.mx-side-stack {
display: grid;
gap: 1rem;
}
.mx-panel {
padding: 1.35rem;
}
.mx-panel__header {
display: flex;
gap: 1rem;
align-items: start;
justify-content: space-between;
margin-bottom: 1.05rem;
}
.mx-panel__header h2 {
margin: 0;
color: #f8ffff;
font-size: 1.2rem;
line-height: 1.2;
}
.mx-panel__note {
max-width: 24rem;
margin: 0;
font-size: 0.92rem;
line-height: 1.55;
text-align: right;
}
.mx-source-grid {
display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr));
gap: 1rem;
}
.mx-source-card,
.mx-action-card,
.mx-app-card,
.mx-feed-item,
.mx-bar-item {
position: relative;
overflow: hidden;
border: 1px solid rgba(73, 208, 200, 0.1);
border-radius: 20px;
background: rgba(6, 23, 29, 0.78);
}
.mx-source-card {
padding: 1.05rem;
}
.mx-source-card::before,
.mx-action-card::before,
.mx-app-card::before,
.mx-feed-item::before,
.mx-bar-item::before {
content: "";
position: absolute;
inset: 0 auto 0 0;
width: 4px;
background: var(--mx-accent, rgba(73, 208, 200, 0.65));
opacity: 0.9;
}
.mx-source-card__top,
.mx-bar-item__head,
.mx-feed-item__meta,
.mx-app-card__head {
display: flex;
gap: 0.75rem;
align-items: start;
justify-content: space-between;
}
.mx-status-pill,
.mx-feed-item__kind {
display: inline-flex;
align-items: center;
min-height: 1.7rem;
padding: 0.25rem 0.65rem;
border-radius: 999px;
font-size: 0.74rem;
font-weight: 700;
letter-spacing: 0.08em;
text-transform: uppercase;
}
.mx-tone--success .mx-status-pill,
.mx-tone--success .mx-feed-item__kind {
background: rgba(45, 212, 191, 0.16);
color: #95fff0;
}
.mx-tone--warning .mx-status-pill,
.mx-tone--warning .mx-feed-item__kind {
background: rgba(255, 200, 87, 0.16);
color: #ffe09a;
}
.mx-tone--danger .mx-status-pill,
.mx-tone--danger .mx-feed-item__kind {
background: rgba(255, 107, 159, 0.18);
color: #ffc2d7;
}
.mx-tone--muted .mx-status-pill,
.mx-tone--muted .mx-feed-item__kind,
.mx-tone--cyan .mx-feed-item__kind {
background: rgba(126, 166, 255, 0.14);
color: #d8e4ff;
}
.mx-source-card__share {
color: #f8ffff;
font-size: 1.1rem;
font-weight: 700;
}
.mx-source-card h3,
.mx-app-card__head h3 {
margin: 0.8rem 0 0;
font-size: 1.02rem;
line-height: 1.3;
}
.mx-source-card p,
.mx-feed-item p,
.mx-action-card span {
margin: 0.55rem 0 0;
line-height: 1.55;
}
.mx-source-card__metrics {
display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr));
gap: 0.85rem;
margin-top: 1rem;
}
.mx-source-card__metrics strong {
display: block;
margin-top: 0.18rem;
color: #f8ffff;
font-size: 1.05rem;
}
.mx-meter {
height: 10px;
margin-top: 0.9rem;
overflow: hidden;
border-radius: 999px;
background: rgba(255, 255, 255, 0.04);
}
.mx-meter span {
display: block;
width: var(--mx-bar-value, 0%);
height: 100%;
border-radius: inherit;
background: linear-gradient(90deg, var(--mx-accent), rgba(255, 255, 255, 0.68));
}
.mx-meter--soft {
margin-top: 0.45rem;
}
.mx-meter--soft span {
width: var(--mx-org-bar-value, 0%);
opacity: 0.72;
}
.mx-source-card__footer {
display: grid;
gap: 0.35rem;
margin-top: 0.95rem;
color: var(--mx-text-dim);
font-size: 0.9rem;
}
.mx-source-card__error {
color: #ffc2d7;
}
.mx-action-list,
.mx-feed-list,
.mx-bar-list,
.mx-app-grid {
display: grid;
gap: 0.85rem;
}
.mx-action-card,
.mx-feed-item,
.mx-bar-item {
padding: 1rem;
text-decoration: none !important;
transition:
transform 0.2s ease,
border-color 0.2s ease,
background-color 0.2s ease;
}
.mx-action-card:hover,
.mx-app-card:hover {
transform: translateY(-2px);
border-color: rgba(73, 208, 200, 0.22);
}
.mx-action-card strong {
display: block;
font-size: 1rem;
}
.mx-feed-item__meta {
margin-bottom: 0.55rem;
}
.mx-feed-item__note {
display: block;
margin-top: 0.45rem;
line-height: 1.45;
}
.mx-bar-item__head {
margin-bottom: 0.7rem;
}
.mx-bar-track {
height: 12px;
overflow: hidden;
border-radius: 999px;
background: rgba(255, 255, 255, 0.05);
}
.mx-bar-track span {
display: block;
height: 100%;
border-radius: inherit;
background: linear-gradient(90deg, rgba(73, 208, 200, 0.95), rgba(126, 166, 255, 0.88));
}
.mx-bar-track--violet span {
background: linear-gradient(90deg, rgba(126, 166, 255, 0.95), rgba(255, 122, 162, 0.88));
}
.mx-app-grid {
grid-template-columns: repeat(2, minmax(0, 1fr));
}
.mx-app-card {
padding: 1rem;
}
.mx-app-card__head {
margin-bottom: 0.8rem;
}
.mx-app-card__head h3 {
margin: 0;
}
.mx-app-card__body {
display: grid;
gap: 0.75rem;
}
.mx-app-model {
display: grid;
gap: 0.7rem;
padding-top: 0.7rem;
border-top: 1px solid rgba(255, 255, 255, 0.05);
}
.mx-app-model:first-child {
padding-top: 0;
border-top: 0;
}
.mx-app-model__name a {
color: #f7fffe;
font-weight: 600;
}
.mx-app-model__actions {
display: flex;
flex-wrap: wrap;
gap: 0.55rem;
}
.mx-mini-button {
display: inline-flex;
align-items: center;
justify-content: center;
min-height: 2rem;
padding: 0.42rem 0.75rem;
border: 1px solid rgba(73, 208, 200, 0.14);
border-radius: 999px;
background: linear-gradient(135deg, rgba(51, 179, 168, 0.2), rgba(65, 112, 196, 0.16));
color: #f6fffe !important;
font-size: 0.83rem;
font-weight: 700;
text-decoration: none !important;
transition:
transform 0.2s ease,
border-color 0.2s ease,
filter 0.2s ease;
}
.mx-mini-button--ghost {
background: rgba(255, 255, 255, 0.03);
}
.mx-mini-button:hover {
filter: brightness(1.05);
transform: translateY(-1px);
}
.mx-empty-state {
margin: 0;
padding: 1rem;
border: 1px dashed rgba(73, 208, 200, 0.18);
border-radius: 18px;
color: var(--mx-text-dim);
background: rgba(5, 18, 24, 0.64);
}
@keyframes mx-dashboard-rise {
from {
opacity: 0;
transform: translateY(10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
@media (max-width: 1440px) {
.mx-hero {
grid-template-columns: minmax(0, 1fr);
}
.mx-hero__visual {
justify-content: center;
}
}
@media (max-width: 1200px) {
.mx-main-grid,
.mx-split-grid,
.mx-overview-grid,
.mx-source-grid,
.mx-app-grid {
grid-template-columns: minmax(0, 1fr);
}
.mx-panel__header {
flex-direction: column;
}
.mx-panel__note {
max-width: none;
text-align: left;
}
}
@media (max-width: 900px) {
.mx-hero,
.mx-panel {
padding: 1rem;
}
.mx-hero__facts {
grid-template-columns: minmax(0, 1fr);
}
}
@media (max-width: 640px) {
.content-wrapper > .content {
padding-left: 0.55rem;
padding-right: 0.55rem;
}
.content-wrapper > .content > .container-fluid {
padding-left: 0;
padding-right: 0;
}
.mx-source-card__metrics,
.mx-bar-item__head,
.mx-feed-item__meta,
.mx-app-card__head,
.mx-source-card__top {
grid-template-columns: minmax(0, 1fr);
display: grid;
}
.mx-source-card__share {
justify-self: start;
}
.mx-ring-chart {
width: 100%;
max-width: 250px;
}
.mx-tab-nav {
width: 100%;
}
}

View File

@@ -405,6 +405,15 @@ a.button:hover,
transform: translateY(-1px);
}
.object-tools .mx-object-tool-form {
display: inline-flex;
margin: 0;
}
.object-tools .mx-object-tool-button {
cursor: pointer;
}
.deletelink,
.btn-danger {
background: linear-gradient(135deg, rgba(255, 107, 159, 0.82), rgba(167, 53, 118, 0.88));

View File

@@ -0,0 +1,151 @@
.mx-upload-page {
max-width: 980px;
}
.mx-upload-shell {
display: grid;
gap: 1rem;
}
.mx-upload-intro {
padding: 1.15rem 1.25rem;
border: 1px solid var(--mx-border);
border-radius: 16px;
background:
linear-gradient(135deg, rgba(8, 24, 30, 0.96), rgba(5, 18, 24, 0.9)),
radial-gradient(circle at top right, rgba(73, 208, 200, 0.12), transparent 32%);
box-shadow: var(--mx-shadow);
}
.mx-upload-title {
margin: 0;
color: #f5fbff;
font-size: 1.35rem;
font-weight: 700;
letter-spacing: 0.01em;
}
.mx-upload-description {
margin: 0.55rem 0 0;
max-width: 64ch;
color: var(--mx-text-muted);
line-height: 1.55;
}
.mx-upload-form {
display: grid;
gap: 0.9rem;
}
.mx-upload-card {
margin: 0;
padding: 1.25rem;
border-radius: 16px;
}
.mx-upload-grid {
display: grid;
gap: 1rem;
}
.mx-upload-field {
display: grid;
gap: 0.45rem;
}
.mx-upload-label {
margin: 0;
color: #d8e7ff;
font-size: 0.95rem;
font-weight: 600;
}
.mx-upload-input,
.mx-upload-select {
width: min(100%, 520px);
padding: 0.7rem 0.9rem;
}
.mx-upload-help {
margin: 0;
max-width: 72ch;
color: var(--mx-text-dim) !important;
line-height: 1.45;
}
.mx-upload-file {
width: min(100%, 720px);
padding: 0.35rem;
cursor: pointer;
}
.mx-upload-file::file-selector-button {
margin-right: 0.8rem;
padding: 0.82rem 1rem;
border: 1px solid rgba(98, 119, 211, 0.48);
border-radius: 10px;
background: linear-gradient(135deg, rgba(51, 179, 168, 0.9), rgba(65, 112, 196, 0.82));
color: #ffffff;
font-weight: 700;
cursor: pointer;
}
.mx-upload-file::-webkit-file-upload-button {
margin-right: 0.8rem;
padding: 0.82rem 1rem;
border: 1px solid rgba(98, 119, 211, 0.48);
border-radius: 10px;
background: linear-gradient(135deg, rgba(51, 179, 168, 0.9), rgba(65, 112, 196, 0.82));
color: #ffffff;
font-weight: 700;
cursor: pointer;
}
.mx-upload-actions {
display: flex;
align-items: center;
gap: 0.8rem;
margin: 0;
padding: 1rem 1.15rem;
border: 1px solid var(--mx-border);
border-radius: 14px;
background: rgba(5, 18, 24, 0.84);
box-shadow: var(--mx-shadow);
}
.mx-upload-cancel {
color: var(--mx-azure);
font-weight: 600;
}
.mx-upload-cancel:hover {
color: #88d9ff;
}
@media (max-width: 900px) {
.mx-upload-file::file-selector-button,
.mx-upload-file::-webkit-file-upload-button {
width: 100%;
margin-right: 0;
margin-bottom: 0.65rem;
}
}
@media (max-width: 640px) {
.mx-upload-card,
.mx-upload-intro,
.mx-upload-actions {
padding: 1rem;
}
.mx-upload-input,
.mx-upload-select,
.mx-upload-file {
width: 100%;
}
.mx-upload-actions {
flex-direction: column;
align-items: stretch;
}
}

View File

@@ -0,0 +1,33 @@
document.addEventListener("DOMContentLoaded", () => {
const pickers = document.querySelectorAll("[data-file-picker]");
for (const picker of pickers) {
const input = picker.querySelector("[data-file-input]");
const summary = picker.querySelector("[data-file-summary]");
if (!input || !summary) {
continue;
}
const emptyText = summary.dataset.emptyText || "Файл не выбран";
const renderSummary = () => {
const files = Array.from(input.files || []);
if (!files.length) {
summary.textContent = emptyText;
return;
}
if (files.length === 1) {
summary.textContent = files[0].name;
return;
}
summary.textContent = `Выбрано файлов: ${files.length}`;
};
input.addEventListener("change", renderSummary);
renderSummary();
}
});