Some checks failed
continuous-integration/drone/push Build is failing
Implement a new sidebar component based on DaisyUI Drawer pattern without custom CSS variants. The sidebar supports desktop (expanded/collapsed states) and mobile (overlay drawer) with full accessibility compliance. Sidebar Implementation: - Refactor sidebar component with sidebar_header, menu_item, menu_group, sidebar_footer sub-components - Add logo (mila.svg) with size-8 (32px) always visible - Implement toggle button with icon swap (chevron-left/right) for desktop - Add nested menu support with details/summary (expanded) and dropdown (collapsed) patterns - Implement footer with language selector (expanded-only), theme toggle, and user menu with avatar - Update layouts.ex to use drawer pattern with data-sidebar-expanded attribute for state management CSS & JavaScript: - Add CSS styles for sidebar state management via data-attribute selectors - Implement SidebarState JavaScript hook for localStorage persistence - Add smooth width transitions (w-64 ↔ w-16) for desktop collapsed state - Add CSS classes for expanded-only, menu-label, and icon visibility Documentation: - Add sidebar-analysis-current-state.md: Analysis of current implementation - Add sidebar-requirements-v2.md: Complete specification for new sidebar - Add daisyui-drawer-pattern.md: DaisyUI pattern documentation - Add umsetzung-sidebar.md: Step-by-step implementation guide Testing: - Add comprehensive component tests for all sidebar sub-components - Add integration tests for sidebar state management and mobile drawer - Extend accessibility tests (ARIA labels, roles, keyboard navigation) - Add regression tests for duplicate IDs, hover effects, and tooltips - Ensure full test coverage per specification requirements
236 lines
7.3 KiB
CSS
236 lines
7.3 KiB
CSS
/* See the Tailwind configuration guide for advanced usage
|
|
https://tailwindcss.com/docs/configuration */
|
|
|
|
@import "tailwindcss";
|
|
@source "../css";
|
|
@source "../js";
|
|
@source "../../lib/mv_web";
|
|
|
|
/* A Tailwind plugin that makes "hero-#{ICON}" classes available.
|
|
The heroicons installation itself is managed by your mix.exs */
|
|
@plugin "../vendor/heroicons";
|
|
|
|
/* daisyUI Tailwind Plugin. You can update this file by fetching the latest version with:
|
|
curl -sLO https://github.com/saadeghi/daisyui/releases/latest/download/daisyui.js
|
|
Make sure to look at the daisyUI changelog: https://daisyui.com/docs/changelog/ */
|
|
@plugin "../vendor/daisyui" {
|
|
themes: false;
|
|
}
|
|
|
|
/* daisyUI theme plugin. You can update this file by fetching the latest version with:
|
|
curl -sLO https://github.com/saadeghi/daisyui/releases/latest/download/daisyui-theme.js
|
|
We ship with two themes, a light one inspired on Phoenix colors and a dark one inspired
|
|
on Elixir colors. Build your own at: https://daisyui.com/theme-generator/ */
|
|
@plugin "../vendor/daisyui-theme" {
|
|
name: "dark";
|
|
default: false;
|
|
prefersdark: true;
|
|
color-scheme: "dark";
|
|
--color-base-100: oklch(30.33% 0.016 252.42);
|
|
--color-base-200: oklch(25.26% 0.014 253.1);
|
|
--color-base-300: oklch(20.15% 0.012 254.09);
|
|
--color-base-content: oklch(97.807% 0.029 256.847);
|
|
--color-primary: oklch(58% 0.233 277.117);
|
|
--color-primary-content: oklch(96% 0.018 272.314);
|
|
--color-secondary: oklch(58% 0.233 277.117);
|
|
--color-secondary-content: oklch(96% 0.018 272.314);
|
|
--color-accent: oklch(60% 0.25 292.717);
|
|
--color-accent-content: oklch(96% 0.016 293.756);
|
|
--color-neutral: oklch(37% 0.044 257.287);
|
|
--color-neutral-content: oklch(98% 0.003 247.858);
|
|
--color-info: oklch(58% 0.158 241.966);
|
|
--color-info-content: oklch(97% 0.013 236.62);
|
|
--color-success: oklch(60% 0.118 184.704);
|
|
--color-success-content: oklch(98% 0.014 180.72);
|
|
--color-warning: oklch(66% 0.179 58.318);
|
|
--color-warning-content: oklch(98% 0.022 95.277);
|
|
--color-error: oklch(58% 0.253 17.585);
|
|
--color-error-content: oklch(96% 0.015 12.422);
|
|
--radius-selector: 0.25rem;
|
|
--radius-field: 0.25rem;
|
|
--radius-box: 0.5rem;
|
|
--size-selector: 0.21875rem;
|
|
--size-field: 0.21875rem;
|
|
--border: 1.5px;
|
|
--depth: 1;
|
|
--noise: 0;
|
|
}
|
|
|
|
@plugin "../vendor/daisyui-theme" {
|
|
name: "light";
|
|
default: true;
|
|
prefersdark: false;
|
|
color-scheme: "light";
|
|
--color-base-100: oklch(98% 0 0);
|
|
--color-base-200: oklch(96% 0.001 286.375);
|
|
--color-base-300: oklch(92% 0.004 286.32);
|
|
--color-base-content: oklch(21% 0.006 285.885);
|
|
--color-primary: oklch(70% 0.213 47.604);
|
|
--color-primary-content: oklch(98% 0.016 73.684);
|
|
--color-secondary: oklch(55% 0.027 264.364);
|
|
--color-secondary-content: oklch(98% 0.002 247.839);
|
|
--color-accent: oklch(0% 0 0);
|
|
--color-accent-content: oklch(100% 0 0);
|
|
--color-neutral: oklch(44% 0.017 285.786);
|
|
--color-neutral-content: oklch(98% 0 0);
|
|
--color-info: oklch(62% 0.214 259.815);
|
|
--color-info-content: oklch(97% 0.014 254.604);
|
|
--color-success: oklch(70% 0.14 182.503);
|
|
--color-success-content: oklch(98% 0.014 180.72);
|
|
--color-warning: oklch(66% 0.179 58.318);
|
|
--color-warning-content: oklch(98% 0.022 95.277);
|
|
--color-error: oklch(58% 0.253 17.585);
|
|
--color-error-content: oklch(96% 0.015 12.422);
|
|
--radius-selector: 0.25rem;
|
|
--radius-field: 0.25rem;
|
|
--radius-box: 0.5rem;
|
|
--size-selector: 0.21875rem;
|
|
--size-field: 0.21875rem;
|
|
--border: 1.5px;
|
|
--depth: 1;
|
|
--noise: 0;
|
|
}
|
|
|
|
/* Add variants based on LiveView classes */
|
|
@custom-variant phx-click-loading (.phx-click-loading&, .phx-click-loading &);
|
|
@custom-variant phx-submit-loading (.phx-submit-loading&, .phx-submit-loading &);
|
|
@custom-variant phx-change-loading (.phx-change-loading&, .phx-change-loading &);
|
|
|
|
/* Make LiveView wrapper divs transparent for layout */
|
|
[data-phx-session] { display: contents }
|
|
|
|
/* ============================================
|
|
Sidebar Base Styles
|
|
============================================ */
|
|
|
|
/* Desktop Sidebar Base */
|
|
.sidebar {
|
|
@apply flex flex-col bg-base-200 min-h-screen;
|
|
@apply transition-[width] duration-300 ease-in-out;
|
|
@apply relative;
|
|
width: 16rem; /* Expanded: w-64 */
|
|
z-index: 40;
|
|
}
|
|
|
|
/* Collapsed State */
|
|
[data-sidebar-expanded="false"] .sidebar {
|
|
width: 4rem; /* Collapsed: w-16 */
|
|
}
|
|
|
|
/* ============================================
|
|
Text Labels - Hide in Collapsed State
|
|
============================================ */
|
|
|
|
.menu-label {
|
|
@apply transition-all duration-200 whitespace-nowrap;
|
|
}
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .menu-label {
|
|
@apply opacity-0 w-0 overflow-hidden pointer-events-none;
|
|
}
|
|
|
|
/* ============================================
|
|
Toggle Button Icon Swap
|
|
============================================ */
|
|
|
|
.sidebar-collapsed-icon {
|
|
@apply hidden;
|
|
}
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .sidebar-expanded-icon {
|
|
@apply hidden;
|
|
}
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .sidebar-collapsed-icon {
|
|
@apply block;
|
|
}
|
|
|
|
/* ============================================
|
|
Menu Groups - Show/Hide Based on State
|
|
============================================ */
|
|
|
|
.expanded-menu-group {
|
|
@apply block;
|
|
}
|
|
|
|
.collapsed-menu-group {
|
|
@apply hidden;
|
|
}
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .expanded-menu-group {
|
|
@apply hidden;
|
|
}
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .collapsed-menu-group {
|
|
@apply block;
|
|
}
|
|
|
|
/* ============================================
|
|
Elements Only Visible in Expanded State
|
|
============================================ */
|
|
|
|
.expanded-only {
|
|
@apply block transition-opacity duration-200;
|
|
}
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .expanded-only {
|
|
@apply hidden;
|
|
}
|
|
|
|
/* ============================================
|
|
Tooltip - Only Show in Collapsed State
|
|
============================================ */
|
|
|
|
.sidebar .tooltip::before,
|
|
.sidebar .tooltip::after {
|
|
@apply opacity-0 pointer-events-none;
|
|
}
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .tooltip:hover::before,
|
|
[data-sidebar-expanded="false"] .sidebar .tooltip:hover::after {
|
|
@apply opacity-100;
|
|
}
|
|
|
|
/* ============================================
|
|
Menu Item Alignment - Center in Collapsed State
|
|
============================================ */
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .menu > li > a,
|
|
[data-sidebar-expanded="false"] .sidebar .menu > li > button {
|
|
@apply justify-center px-0;
|
|
}
|
|
|
|
/* ============================================
|
|
Footer Button Alignment - Center in Collapsed State
|
|
============================================ */
|
|
|
|
[data-sidebar-expanded="false"] .sidebar .dropdown > button {
|
|
@apply justify-center px-0;
|
|
}
|
|
|
|
/* ============================================
|
|
Mobile Drawer Width
|
|
============================================ */
|
|
|
|
/* Auf Mobile (< 1024px) ist die Sidebar immer w-64 (16rem) wenn geöffnet */
|
|
@media (max-width: 1023px) {
|
|
.drawer-side .sidebar {
|
|
width: 16rem; /* w-64 auch auf Mobile */
|
|
}
|
|
}
|
|
|
|
/* ============================================
|
|
Drawer Side Overflow Fix für Desktop
|
|
============================================ */
|
|
|
|
/* Im Desktop-Modus (lg:drawer-open) overflow auf visible setzen
|
|
damit Dropdowns und Tooltips über Main Content erscheinen können */
|
|
@media (min-width: 1024px) {
|
|
.drawer.lg\:drawer-open .drawer-side {
|
|
overflow: visible !important;
|
|
overflow-x: visible !important;
|
|
overflow-y: visible !important;
|
|
}
|
|
}
|
|
|
|
/* This file is for your main application CSS */
|