javascript技巧

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript技巧 > JavaScript响应式横向多级菜单

JavaScript实现响应式横向多级菜单的示例代码

作者:百锦再@新空间

这篇文章主要为大家详细介绍了如何使用JavaScript实现响应式横向多级菜单功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下

1. 项目概述与需求分析

1.1 核心需求

横向菜单结构:

交互状态样式:

自适应功能:

布局行为:

1.2 技术指标

2. 技术选型与设计原理

2.1 HTML结构设计

采用语义化HTML5结构:

2.2 CSS方案

2.3 JavaScript功能

3. HTML结构设计

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>高级响应式多级菜单系统</title>
    <link rel="stylesheet" href="styles.css" rel="external nofollow" >
</head>
<body>
    <header class="page-header">
        <nav class="main-navigation" aria-label="主导航">
            <button class="mobile-menu-toggle" aria-expanded="false" aria-controls="primary-menu">
                <span class="toggle-icon"></span>
                <span class="sr-only">菜单</span>
            </button>
            
            <ul id="primary-menu" class="menu">
                <li class="menu-item has-submenu">
                    <a href="#home" rel="external nofollow"  class="menu-link">首页</a>
                    <ul class="submenu">
                        <li class="submenu-item"><a href="#home-overview" rel="external nofollow"  class="submenu-link">首页概览</a></li>
                        <li class="submenu-item"><a href="#home-features" rel="external nofollow"  class="submenu-link">首页特性</a></li>
                    </ul>
                </li>
                <li class="menu-item has-submenu">
                    <a href="#products" rel="external nofollow"  class="menu-link">产品</a>
                    <ul class="submenu">
                        <li class="submenu-item"><a href="#product-1" rel="external nofollow"  class="submenu-link">旗舰产品</a></li>
                        <li class="submenu-item"><a href="#product-2" rel="external nofollow"  class="submenu-link">新产品</a></li>
                    </ul>
                </li>
                <!-- 重复8个菜单项,结构相同 -->
                <li class="menu-item more-menu">
                    <a href="javascript:void(0)" rel="external nofollow"  class="menu-link more-link">更多 <span class="more-icon">›</span></a>
                    <ul class="more-submenu">
                        <!-- 动态填充隐藏的菜单项 -->
                    </ul>
                </li>
            </ul>
        </nav>
    </header>
    
    <main class="page-content">
        <!-- 页面内容 -->
        <div style="height: 2000px;"></div>
    </main>
    
    <script src="menu.js"></script>
</body>
</html>

4. CSS样式系统

:root {
    /* 颜色系统 */
    --primary-color: #2c3e50;
    --secondary-color: #3498db;
    --text-color: #333;
    --text-inverse: #fff;
    --hover-color: #2980b9;
    --active-color: #1abc9c;
    --border-color: #e0e0e0;
    --shadow-color: rgba(0, 0, 0, 0.1);
    
    /* 间距系统 */
    --space-xs: 0.25rem;
    --space-sm: 0.5rem;
    --space-md: 1rem;
    --space-lg: 1.5rem;
    --space-xl: 2rem;
    
    /* 过渡效果 */
    --transition-fast: 0.15s;
    --transition-normal: 0.3s;
    
    /* 断点 */
    --breakpoint-sm: 576px;
    --breakpoint-md: 768px;
    --breakpoint-lg: 992px;
    --breakpoint-xl: 1200px;
}

/* 基础重置 */
* {
    margin: 0;
    padding: 0;
    box-sizing: border-box;
}

body {
    font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
    line-height: 1.6;
    color: var(--text-color);
}

/* 辅助类 */
.sr-only {
    position: absolute;
    width: 1px;
    height: 1px;
    padding: 0;
    margin: -1px;
    overflow: hidden;
    clip: rect(0, 0, 0, 0);
    white-space: nowrap;
    border-width: 0;
}

/* 页面布局 */
.page-header {
    position: fixed;
    top: 0;
    left: 0;
    right: 0;
    background-color: var(--primary-color);
    z-index: 1000;
    box-shadow: 0 2px 10px var(--shadow-color);
}

.main-navigation {
    display: flex;
    justify-content: space-between;
    align-items: center;
    max-width: 1400px;
    margin: 0 auto;
    padding: 0 var(--space-md);
}

/* 移动端菜单按钮 */
.mobile-menu-toggle {
    display: none;
    background: none;
    border: none;
    color: var(--text-inverse);
    font-size: 1.5rem;
    cursor: pointer;
    padding: var(--space-sm);
}

/* 主菜单样式 */
.menu {
    display: flex;
    list-style: none;
    margin: 0;
    padding: 0;
}

.menu-item {
    position: relative;
}

.menu-link {
    display: block;
    padding: var(--space-md) var(--space-lg);
    color: var(--text-inverse);
    text-decoration: none;
    font-weight: 500;
    transition: background-color var(--transition-fast);
    white-space: nowrap;
}

/* 菜单项交互状态 */
.menu-link:hover,
.menu-link:focus {
    background-color: var(--hover-color);
    outline: none;
}

.menu-item.active > .menu-link {
    background-color: var(--active-color);
}

/* 子菜单样式 */
.submenu {
    position: absolute;
    top: 100%;
    left: 0;
    min-width: 200px;
    background-color: white;
    list-style: none;
    padding: var(--space-sm) 0;
    margin: 0;
    box-shadow: 0 3px 10px var(--shadow-color);
    opacity: 0;
    visibility: hidden;
    transform: translateY(10px);
    transition: all var(--transition-normal);
    z-index: 100;
}

.has-submenu:hover .submenu,
.has-submenu:focus-within .submenu {
    opacity: 1;
    visibility: visible;
    transform: translateY(0);
}

.submenu-item {
    position: relative;
}

.submenu-link {
    display: block;
    padding: var(--space-sm) var(--space-lg);
    color: var(--text-color);
    text-decoration: none;
    transition: all var(--transition-fast);
}

.submenu-link:hover,
.submenu-link:focus {
    background-color: #f5f5f5;
    color: var(--hover-color);
    padding-left: var(--space-xl);
}

/* 更多菜单样式 */
.more-menu {
    display: none;
}

.more-submenu {
    right: 0;
    left: auto;
}

/* 响应式设计 */
@media screen and (max-width: 1200px) {
    .menu-item:nth-last-child(-n+2) {
        display: none;
    }
    .more-menu {
        display: block;
    }
}

@media screen and (max-width: 1000px) {
    .menu-item:nth-last-child(-n+4) {
        display: none;
    }
}

@media screen and (max-width: 800px) {
    .menu-item:nth-last-child(-n+6) {
        display: none;
    }
}

@media screen and (max-width: 600px) {
    .mobile-menu-toggle {
        display: block;
    }
    
    .menu {
        position: fixed;
        top: 60px;
        left: 0;
        right: 0;
        flex-direction: column;
        background-color: var(--primary-color);
        max-height: 0;
        overflow: hidden;
        transition: max-height var(--transition-normal);
    }
    
    .menu.show {
        max-height: 100vh;
    }
    
    .menu-item {
        width: 100%;
    }
    
    .submenu {
        position: static;
        box-shadow: none;
        transform: none;
        max-height: 0;
        overflow: hidden;
        transition: max-height var(--transition-normal);
    }
    
    .has-submenu:hover .submenu,
    .has-submenu:focus-within .submenu {
        max-height: 500px;
    }
}

5. JavaScript交互逻辑

document.addEventListener('DOMContentLoaded', function() {
    const nav = document.querySelector('.main-navigation');
    const menu = document.getElementById('primary-menu');
    const menuItems = document.querySelectorAll('.menu-item');
    const toggleBtn = document.querySelector('.mobile-menu-toggle');
    const moreMenu = document.querySelector('.more-menu');
    const moreSubmenu = document.querySelector('.more-submenu');
    
    // 初始化菜单
    initMenu();
    
    // 移动端菜单切换
    toggleBtn.addEventListener('click', function() {
        const isExpanded = this.getAttribute('aria-expanded') === 'true';
        this.setAttribute('aria-expanded', !isExpanded);
        menu.classList.toggle('show');
    });
    
    // 窗口大小变化处理
    const resizeObserver = new ResizeObserver(handleResize);
    resizeObserver.observe(document.body);
    
    // 动态调整子菜单宽度
    adjustSubmenuWidths();
    
    // 点击外部关闭菜单
    document.addEventListener('click', function(e) {
        if (!nav.contains(e.target) && !toggleBtn.contains(e.target)) {
            menu.classList.remove('show');
            toggleBtn.setAttribute('aria-expanded', 'false');
        }
    });
    
    // 初始化菜单
    function initMenu() {
        // 设置ARIA属性
        menuItems.forEach(item => {
            if (item.classList.contains('has-submenu')) {
                const submenu = item.querySelector('.submenu');
                item.setAttribute('aria-haspopup', 'true');
                item.setAttribute('aria-expanded', 'false');
                
                // 添加键盘导航支持
                item.addEventListener('keydown', function(e) {
                    if (e.key === 'Enter' || e.key === ' ') {
                        e.preventDefault();
                        toggleSubmenu(this);
                    }
                });
            }
        });
        
        // 处理更多菜单
        updateMoreMenu();
    }
    
    // 调整子菜单宽度
    function adjustSubmenuWidths() {
        document.querySelectorAll('.submenu').forEach(submenu => {
            let maxWidth = 0;
            
            submenu.querySelectorAll('.submenu-link').forEach(link => {
                const width = link.scrollWidth;
                if (width > maxWidth) maxWidth = width;
            });
            
            submenu.style.minWidth = `${maxWidth + 32}px`;
        });
    }
    
    // 处理窗口大小变化
    function handleResize(entries) {
        updateMoreMenu();
        adjustSubmenuWidths();
    }
    
    // 更新更多菜单内容
    function updateMoreMenu() {
        if (!moreMenu) return;
        
        // 清空现有内容
        moreSubmenu.innerHTML = '';
        
        // 获取隐藏的菜单项
        const hiddenItems = Array.from(menuItems).filter(item => {
            return item.style.display === 'none' && !item.classList.contains('more-menu');
        });
        
        if (hiddenItems.length > 0) {
            moreMenu.style.display = 'block';
            
            hiddenItems.forEach(item => {
                const clone = item.cloneNode(true);
                clone.style.display = 'block';
                moreSubmenu.appendChild(clone);
            });
        } else {
            moreMenu.style.display = 'none';
        }
    }
    
    // 切换子菜单显示/隐藏
    function toggleSubmenu(menuItem) {
        const isExpanded = menuItem.getAttribute('aria-expanded') === 'true';
        menuItem.setAttribute('aria-expanded', !isExpanded);
    }
});

6. 响应式布局实现

6.1 断点设计

我们设置了多个断点来适应不同视口宽度:

6.2 实现原理

使用CSS媒体查询结合:nth-last-child选择器动态隐藏菜单项:

@media screen and (max-width: 1200px) {
    .menu-item:nth-last-child(-n+2) {
        display: none;
    }
    .more-menu {
        display: block;
    }
}

JavaScript动态检测隐藏的菜单项并将其填充到"更多"下拉菜单中。

7. 固定定位与滚动处理

7.1 固定定位实现

.page-header {
    position: fixed;
    top: 0;
    left: 0;
    right: 0;
    z-index: 1000;
}

7.2 内容偏移处理

由于菜单固定在顶部,页面内容需要向下偏移菜单高度:

.page-content {
    margin-top: 60px; /* 等于菜单高度 */
}

8. 完整代码实现

由于篇幅限制,完整代码已在前文分章节展示。以下是关键点总结:

HTML结构:语义化导航结构,包含主菜单和子菜单

9. 测试与验证方案

9.1 测试用例

桌面端测试:

移动端测试:

无障碍测试:

9.2 验证指标

10. 性能优化与兼容性

10.1 性能优化

事件委托:减少事件监听器数量

防抖处理:优化resize事件

硬件加速:使用transform进行动画

内存管理:及时清理不用的引用

10.2 兼容性处理

浏览器前缀:添加必要的vendor前缀

特性检测:检查ResizeObserver等新API支持

降级方案:为旧浏览器提供基本功能

触摸支持:同时处理鼠标和触摸事件

11. 扩展可能性

多级子菜单:支持三级甚至更深的菜单结构

动画效果:添加更丰富的过渡动画

主题系统:通过CSS变量实现多主题支持

配置化:通过JSON配置动态生成菜单

权限控制:根据用户角色显示不同菜单项

12. 总结

本文详细实现了一个功能完善的响应式多级菜单系统,具有以下特点:

到此这篇关于JavaScript实现响应式横向多级菜单的示例代码的文章就介绍到这了,更多相关JavaScript响应式横向多级菜单内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文