fix: responsive layout - sidebar collapse, mobile adaptation, prevent text jumping
- Fix text jumping on right side when screen narrows (min-width:0 on flex item) - Fix sidebar toggle button not working (retry icon init, proper state sync) - Mobile adaptation: sidebar as overlay with slide animation - Auto-collapse sidebar on mobile viewport (<=768px) - Click outside sidebar to close on mobile - Responsive padding for stat cards and main content - Hide brand title on very small screens (<=480px) - Smooth transitions for sidebar collapse/expand
This commit is contained in:
parent
79a04be92b
commit
3659533102
@ -128,7 +128,9 @@ body {
|
|||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
transition: width 0.25s ease;
|
transition: width 0.3s ease, transform 0.3s ease;
|
||||||
|
position: relative;
|
||||||
|
z-index: 50;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sage-sidebar.collapsed {
|
.sage-sidebar.collapsed {
|
||||||
@ -141,12 +143,48 @@ body {
|
|||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Mobile: sidebar as overlay */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.sage-sidebar {
|
||||||
|
position: fixed;
|
||||||
|
left: 0;
|
||||||
|
top: var(--sage-topbar-height);
|
||||||
|
height: calc(100vh - var(--sage-topbar-height));
|
||||||
|
transform: translateX(0);
|
||||||
|
box-shadow: var(--sage-shadow-lg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sage-sidebar.collapsed {
|
||||||
|
transform: translateX(-100%);
|
||||||
|
width: var(--sage-sidebar-width); /* Keep full width when hidden on mobile */
|
||||||
|
}
|
||||||
|
|
||||||
|
.sage-main {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.sage-main {
|
.sage-main {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
padding: 24px;
|
padding: 24px;
|
||||||
background-color: var(--sage-bg-primary);
|
background-color: var(--sage-bg-primary);
|
||||||
|
min-width: 0; /* Prevent flex item from overflowing */
|
||||||
|
transition: margin-left 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Responsive padding for mobile */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.sage-main {
|
||||||
|
padding: 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 480px) {
|
||||||
|
.sage-main {
|
||||||
|
padding: 12px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ===== Stat Cards ===== */
|
/* ===== Stat Cards ===== */
|
||||||
@ -719,18 +757,43 @@ body {
|
|||||||
color: var(--sage-text-secondary);
|
color: var(--sage-text-secondary);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ===== Responsive ===== */
|
/* ===== Responsive - Topbar ===== */
|
||||||
@media (max-width: 768px) {
|
@media (max-width: 768px) {
|
||||||
.sage-sidebar {
|
.sage-topbar {
|
||||||
position: fixed;
|
padding: 0 8px;
|
||||||
left: 0;
|
gap: 6px;
|
||||||
top: var(--sage-topbar-height);
|
|
||||||
bottom: 0;
|
|
||||||
z-index: 200;
|
|
||||||
transform: translateX(-100%);
|
|
||||||
transition: transform 0.25s ease;
|
|
||||||
}
|
}
|
||||||
.sage-sidebar.mobile-open {
|
.sage-brand-title {
|
||||||
transform: translateX(0);
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 480px) {
|
||||||
|
.sage-brand-title {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ===== Responsive - Stat Cards ===== */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.stat-card {
|
||||||
|
padding: 14px;
|
||||||
|
}
|
||||||
|
.stat-card .stat-value {
|
||||||
|
font-size: 22px;
|
||||||
|
}
|
||||||
|
.stat-card .stat-label {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
.stat-card .stat-icon {
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
.quick-link {
|
||||||
|
padding: 14px;
|
||||||
|
}
|
||||||
|
.section-title {
|
||||||
|
font-size: 16px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -42,12 +42,61 @@
|
|||||||
function initSidebar() {
|
function initSidebar() {
|
||||||
var collapsed = false;
|
var collapsed = false;
|
||||||
try { collapsed = localStorage.getItem(SIDEBAR_KEY) === 'true'; } catch(e) {}
|
try { collapsed = localStorage.getItem(SIDEBAR_KEY) === 'true'; } catch(e) {}
|
||||||
|
|
||||||
|
// Auto-collapse on mobile
|
||||||
|
if (isMobile()) {
|
||||||
|
collapsed = true;
|
||||||
|
}
|
||||||
|
|
||||||
var sidebar = document.getElementById('sage_sidebar');
|
var sidebar = document.getElementById('sage_sidebar');
|
||||||
if (sidebar && collapsed) {
|
if (sidebar && collapsed) {
|
||||||
sidebar.classList.add('collapsed');
|
sidebar.classList.add('collapsed');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handle window resize - auto-collapse on mobile
|
||||||
|
function handleResize() {
|
||||||
|
var sidebar = document.getElementById('sage_sidebar');
|
||||||
|
if (!sidebar) return;
|
||||||
|
|
||||||
|
if (isMobile() && !sidebar.classList.contains('collapsed')) {
|
||||||
|
sidebar.classList.add('collapsed');
|
||||||
|
try { localStorage.setItem(SIDEBAR_KEY, 'true'); } catch(e) {}
|
||||||
|
updateSidebarIcon(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if we're on mobile viewport
|
||||||
|
function isMobile() {
|
||||||
|
return window.innerWidth <= 768;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close sidebar when clicking outside on mobile
|
||||||
|
function setupMobileOverlay() {
|
||||||
|
document.addEventListener('click', function(e) {
|
||||||
|
if (!isMobile()) return;
|
||||||
|
var sidebar = document.getElementById('sage_sidebar');
|
||||||
|
var toggleBtn = document.getElementById('sidebar_toggle_btn');
|
||||||
|
if (!sidebar || sidebar.classList.contains('collapsed')) return;
|
||||||
|
// If click is outside sidebar and toggle button, close sidebar
|
||||||
|
if (!sidebar.contains(e.target) && (!toggleBtn || !toggleBtn.contains(e.target))) {
|
||||||
|
sidebar.classList.add('collapsed');
|
||||||
|
try { localStorage.setItem(SIDEBAR_KEY, 'true'); } catch(ex) {}
|
||||||
|
updateSidebarIcon(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateSidebarIcon(isCollapsed) {
|
||||||
|
var btn = document.getElementById('sidebar_toggle_btn');
|
||||||
|
if (!btn) return;
|
||||||
|
if (isCollapsed) {
|
||||||
|
btn.innerHTML = '<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="13 17 18 12 13 7"/><polyline points="6 17 11 12 6 7"/></svg>';
|
||||||
|
} else {
|
||||||
|
btn.innerHTML = '<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="11 17 6 12 11 7"/><polyline points="18 17 13 12 18 7"/></svg>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Toggle sidebar collapse
|
// Toggle sidebar collapse
|
||||||
function toggleSidebar() {
|
function toggleSidebar() {
|
||||||
var sidebar = document.getElementById('sage_sidebar');
|
var sidebar = document.getElementById('sage_sidebar');
|
||||||
@ -55,16 +104,7 @@
|
|||||||
sidebar.classList.toggle('collapsed');
|
sidebar.classList.toggle('collapsed');
|
||||||
var isCollapsed = sidebar.classList.contains('collapsed');
|
var isCollapsed = sidebar.classList.contains('collapsed');
|
||||||
try { localStorage.setItem(SIDEBAR_KEY, isCollapsed); } catch(e) {}
|
try { localStorage.setItem(SIDEBAR_KEY, isCollapsed); } catch(e) {}
|
||||||
|
updateSidebarIcon(isCollapsed);
|
||||||
// Update toggle icon
|
|
||||||
var btn = document.getElementById('sidebar_toggle_btn');
|
|
||||||
if (btn) {
|
|
||||||
if (isCollapsed) {
|
|
||||||
btn.innerHTML = '<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="13 17 18 12 13 7"/><polyline points="6 17 11 12 6 7"/></svg>';
|
|
||||||
} else {
|
|
||||||
btn.innerHTML = '<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="11 17 6 12 11 7"/><polyline points="18 17 13 12 18 7"/></svg>';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialize SPA Router
|
// Initialize SPA Router
|
||||||
@ -115,11 +155,15 @@
|
|||||||
initTheme();
|
initTheme();
|
||||||
initSidebar();
|
initSidebar();
|
||||||
initRouter();
|
initRouter();
|
||||||
|
setupMobileOverlay();
|
||||||
|
window.addEventListener('resize', handleResize);
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
initTheme();
|
initTheme();
|
||||||
initSidebar();
|
initSidebar();
|
||||||
initRouter();
|
initRouter();
|
||||||
|
setupMobileOverlay();
|
||||||
|
window.addEventListener('resize', handleResize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bricks widgets render asynchronously after DOMContentLoaded.
|
// Bricks widgets render asynchronously after DOMContentLoaded.
|
||||||
@ -134,6 +178,18 @@
|
|||||||
}
|
}
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
// Retry sidebar icon init until the button element exists
|
||||||
|
(function retrySidebarIcon() {
|
||||||
|
var sidebar = document.getElementById('sage_sidebar');
|
||||||
|
var btn = document.getElementById('sidebar_toggle_btn');
|
||||||
|
if (btn && sidebar) {
|
||||||
|
var isCollapsed = sidebar.classList.contains('collapsed');
|
||||||
|
updateSidebarIcon(isCollapsed);
|
||||||
|
} else {
|
||||||
|
setTimeout(retrySidebarIcon, 200);
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
|
||||||
// Expose global functions for bricks bind access
|
// Expose global functions for bricks bind access
|
||||||
window.sageToggleTheme = toggleTheme;
|
window.sageToggleTheme = toggleTheme;
|
||||||
window.sageToggleSidebar = toggleSidebar;
|
window.sageToggleSidebar = toggleSidebar;
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user