开关菜单

/*
    将需要的信息添加到类的静态数组里备用
    设置一个值openBool,默认为false;菜单默认display为none
    点击时,如果目标元素是子元素,则不做开关菜单操作,直接return。
    点击时,读取openbool。
    bool为false,打开菜单;bool为true时,关闭菜单。每次执行菜单操作之后,bool取反。循环往复。
*/
import Utils from "./Utils.js";
export default class ProClassify {

    static proList = [
        { id: 1, class: "diamondClass", name: "钻石", item: ["GIA钻石", "30分钻石", "50分钻石", "北极光钻石"] },
        { id: 2, class: "ringClass", name: "钻戒", item: ["铂金钻戒", "30分钻戒", "50分钻戒", "70分钻戒", "克拉钻戒"] },
        { id: 3, class: "doubleRingClass", name: "对戒", item: ["铂金对戒", "18K对戒"] },
        { id: 4, class: "ornamentClass", name: "配饰", item: ["吊坠", "耳钉", "套链", "项链", "手链"] },
    ];
    constructor() {
        this.elem = this.createElem();
    }

    createElem() {
        let con = document.createDocumentFragment();
        ProClassify.proList.map(proItem => {
            let div = document.createElement("div");
            div.className = "classItem";
            div.innerHTML = `
            <li id="${proItem.class}" class="${proItem.class}">${proItem.name}
                <ul>
                ${
                proItem.item.reduce((total, cur) => {
                    return total + `<a href="###">${cur}</a>`;
                }, "")
                }
                </ul>
            </li>`;
            //将id下的所有子元素添加到this
            Utils.getIdElem(div, this);
            div.openBool = false;
            this[proItem.class].addEventListener("click", e => this.liClickHandler(e));
            con.appendChild(div);
        });
        return con;
    }

    //点击开关菜单
    liClickHandler(e) {
        if (e.target.constructor === HTMLAnchorElement) return;
        e.target.children[0].style.display = e.target.parentNode.openBool ? "none" : "block";
        e.target.parentNode.openBool = !e.target.parentNode.openBool;
    }

    //插入父容器最前位置,bool为true则添加到末尾
    unShiftIn(parent,bool) {
        parent = HTMLElement.isPrototypeOf(parent.constructor) ? parent : document.querySelector(parent);
        if (parent) bool ? parent.appendChild(this.elem) : parent.insertBefore(this.elem, parent.firstChild);
    }
} 

Utils

export default class Utils{
    //将id下的所有子元素添加到this
    static getIdElem(elem,o){
        if(elem.id) o[elem.id]=elem;
        if(elem.children.length===0) return o;
        for(var i=0;i<elem.children.length;i++){
            Utils.getIdElem(elem.children[i],o);
        }
    }
}

效果:

开:

关:

后记

display:none不占位置,不在文档流中,导致transition失效。考虑动画或者visibility+定位实现

多级菜单:考虑用递归,关闭父菜单关闭时维护其所有子菜单的开关状态、openBool值。

js开关菜单的更多相关文章

  1. JS树形菜单

    超全的JS树形菜单源代码共享(有实例图) 树形菜单是很常用的效果,常用在管理软件当中,但是一套树形菜单已经不能满足需求,所以如果能有一套比较全面的树形菜单JS特效代码,将会非常方便,下面懒人萱将超全的 ...

  2. js矩阵菜单或3D立体预览图片效果

    js矩阵菜单或3D立体预览图片效果 下载地址: http://files.cnblogs.com/elves/js%E7%9F%A9%E9%98%B5%E8%8F%9C%E5%8D%95%E6%88% ...

  3. 顶 兼容各种浏览器js折叠菜单

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  4. 实例介绍Cocos2d-x开关菜单

    开关菜单是MenuItemToggle类实现的,它是一种可以进行两种状态切换的菜单.它可以通过下面的函数创建: static MenuItemToggle*createWithCallback  ( ...

  5. js开关插件使用

    一.简介 本篇文章介绍一个比较好使用的js开关插件Switchery,该插件的样式是ios7的滑动按钮插件,并且将很多功能加入到配置项,简单.灵活,支持的绝大部分浏览器(Chrome, Firefox ...

  6. 基于Bootstrap Ace模板+bootstrap.addtabs.js的菜单

    这几天研究了基于bootstrap Ace模板+bootstra.addtabs.js实现菜单的效果 参考了这个人的博客 https://www.cnblogs.com/landeanfen/p/76 ...

  7. 适合移动手机使用的js环形菜单特效插件

    blooming-menu是一款适合在移动手机上使用的js环形菜单插件.该环形菜单提供了众多的參数,通过结合CSS3动画制作出效果很炫酷的圆形菜单展开和隐藏动画效果. 以下是这个圆形菜单菜价的可用參数 ...

  8. jquery.smint.js 页面菜单浮动之谷歌浏览器异常

    jquery.smint.js 做公司项目时,谷歌测试,页面向下拉,刷新后,导航栏菜单与顶部距离发生变动,并不在设置的relative元素top下固定像素 我的relative元素的高为80,然后在j ...

  9. js侧边菜单

    目标 实现一个侧边栏菜单,最多二级,可以收起展开.用于系统左侧的主菜单. 大多数系统都会有这样的菜单,用于导航功能,切换到不同的操作页面.在单页应用系统中,菜单一般是固定在左侧,分组节点上配图标,高亮 ...

随机推荐

  1. Outlook365(Oulook2016 或2013) 写邮件输入收件人时的推荐联系人如何清理?

    · 在Outlook365(Oulook2016 或2013) 中写邮件,输入收件人邮箱地址时,会出现“最近联系人”  “其他建议”等推荐的联系人,可以方便选择.如果里面有很多邮箱地址的已经无效的话, ...

  2. Codeforces Round #603 (Div. 2) A.Sweet Problem

    #include <cstdio> #include <algorithm> using namespace std; int main() { int t; scanf(&q ...

  3. EF的预先加载--Eager Loading

    预先加载 在对一种类型的实体进行查询时,将相关的实体作为查询的一部分一起加载.预先加载可以使用Include()方法实现. 在此需要说明的是:EF中有两种表关联的方法,一种是Join()方法,一种是I ...

  4. [CCPC2019秦皇岛] F. Forest Program

    [CCPC2019秦皇岛 F] Link https://codeforces.com/gym/102361/problem/F Description 给定一个仙人掌,删去一些边可以让它变成一个森林 ...

  5. 基于Java+HttpClient+TestNG的接口自动化测试框架(八)------ 针对文件的处理

    在实际的接口测试中,有时需要根据情况进行文件的上传和下载.在文件数量比较小的时候,我们当然可以直接处理(比如若干个接口都用一个文件).但是,如果我们上传的文件需要使用不同文件夹里不同的文件,而且数量又 ...

  6. sql查询 —— 连接查询

    -- 执行sql文件 test.sql -- 打开终端 -- cd sql文件所在路径 -- 进入mysql -- use 数据库 -- 执行 source test.sql; -- 自关联 -- 一 ...

  7. linux复制,网络报错

    我拷贝了过来的Linux虚拟机无法上网,我用ifconfig命令查询了一下发现只有eth1和lo设备,没有eth0.于是我在Google上搜索了一下, <VMWare克隆或复制Linux虚拟机后 ...

  8. Resharper快捷键使用

    1:   Alt+F7将你光标所在位置的变量的所有使用以列表的方式显示出来,显示结果的窗体可以像其他窗体那样停靠. 它的优点包括: 可以从所有使用中挑选只显示read usage或者write usa ...

  9. C++-POJ1018-Communication System

    贪心算法:先排序,再枚举最小带宽(B),每次更新当前最小花费(P)和以及答案(ans) #include <cstdio> #include <algorithm> using ...

  10. Apache Kafka(五)- Safe Kafka Producer

    Kafka Safe Producer 在应用Kafka的场景中,需要考虑到在异常发生时(如网络异常),被发送的消息有可能会出现丢失.乱序.以及重复消息. 对于这些情况,我们可以创建一个“safe p ...