記事ソース: Xiaoqiang オリジナル
公開時間: 2016-05-19
キーワード: mui、html5+、webview
転載する場合は、この記事の元のアドレスを示す必要があります: http://zhaomenghuan.github.io/#!/blog/20160519
このシリーズ この記事では、mui を使用して NetEase Cloud Music API に基づいた音楽プレーヤー APP を実装し、Ring または Fusion Cloud に基づいてチャット機能を実装します。この記事では、シリーズの最初の記事として、HTML5+ でアプリケーション ウィンドウ インターフェイスを管理する Webview モジュールの使用方法について詳しく説明します。初心者向けのチュートリアルであるため、初心者ユーザーが必要とする原理についてはあまり説明しません。基本的な使い方を知り、mui のコンポーネントを使用してページの効果を表示します。
Webview モジュールは、アプリケーション ウィンドウ インターフェイスを管理し、マルチウィンドウの論理制御と管理操作を実現します。アプリケーション インターフェイス管理オブジェクトは、plus.webview を通じて取得できます。
ここで最初に例を示します。誰もがブラウザを使用したことがありますが、一般的に使用されているブラウザでは複数の Web ページを開くことができます。コンピュータ画面は単なるウィンドウです。Web ビューの切り替えを制御することで、ページごとに異なる Web ビューを使用できます。
私たちにとって、HTML ページはウィンドウであり、HTML ページは複数の Web ビューを作成できます。この Web ビューは、Android と iOS の両方のネイティブ APP で Web ページを閲覧するためのコンポーネントです。html5plus の Web ビューはネイティブ Web ビューのカプセル化であり、js で呼び出すことができるため、その実行環境は APP 環境であり、これはサポートされていません。通常のブラウザ。
まず、html5plus 公式 Web サイトの Webview API ドキュメント を見てみましょう。 ここでは、次のメソッドに焦点を当てます。
WebviewObject plus.webview.create( url, id, styles, extras );
新しい HTML ページをロードするための Webview ウィンドウを作成します。作成後、表示するには show メソッドを呼び出す必要があります。 Webビューウィンドウ。
void plus.webview.show( id_wvobj, aniShow, duration, showedCB, extras );
作成または非表示にした Webview ウィンドウを表示するには、最初にウィンドウ オブジェクトまたはウィンドウ ID を取得する必要があります。表示ウィンドウのアニメーションとアニメーションの継続時間を指定できます。
void plus.webview.hide( id_wvobj, aniHide, duration, extras );
指定された WebviewObject オブジェクトまたは ID に基づいて Webview ウィンドウを非表示にし、ウィンドウを非表示にします。
WebviewObject plus.webview.currentWebview();
現在のページが属する Webview ウィンドウ オブジェクトを取得します。
WebviewObject plus.webview.getWebviewById( id );
作成されたウィンドウ リストから指定された ID を持つ Webview ウィンドウを検索して返します。 指定された ID のウィンドウが見つからない場合は、null が返されます。同じ ID の Webview ウィンドウが複数ある場合は、最初に作成された Webview ウィンドウが返されます。 アプリケーションのエントリ ページが属する Webview ウィンドウを取得する場合、その識別子はアプリケーションの %APPID% であり、plus.runtime.appid を通じて取得できます。
WebviewObject plus.webview.open( url, id, styles, aniShow, duration, showedCB );
新しい HTML ページをロードするための Webview ウィンドウを作成して表示します。 Webview ウィンドウのスタイルは次のとおりです。スタイルを通じて設定すると、作成後に Webview ウィンドウが自動的に表示されます。
上記は html5plus ドキュメントからのものです。これはドキュメントではないため、明確ではありません。詳細については、こちらの html5plus Webview API をご覧ください。
まず、白い画面のフォーム切り替えと領域スクロールの問題を解決するために、mui が提案するデュアル Web ビュー モードを理解する必要があります。
mui フレームワークでは、多くの機能設定が mui.init メソッドに集中しています。特定の機能を使用するには、mui で対応するパラメータ設定を完了するだけです。現在、mui.init メソッドでサポートされている機能には、サブページの作成、ページの終了、ジェスチャ イベントの設定、プリロード、プルダウンの更新、プルアップのロード、およびシステム ステータスの背景色の設定が含まれます。バー。 mui は、ページのロード時にリターン キーの監視など、多くの基本的なコントロールを初期化する必要があるため、すべてのページで呼び出す必要があります。
mui.init({ //子页面 subpages: [{ //... }], //预加载 preloadPages:[ //... ], //下拉刷新、上拉加载 pullRefresh : { //... }, //手势配置 gestureConfig:{ //... }, //侧滑关闭 swipeBack:true, //Boolean(默认false)启用右滑关闭功能 //监听Android手机的back、menu按键 keyEventBind: { backbutton: false, //Boolean(默认truee)关闭back按键监听 menubutton: false //Boolean(默认true)关闭menu按键监听 }, //处理窗口关闭前的业务 beforeback: function() { //... //窗口关闭前处理其他业务详情点击 ↑ "关闭页面"链接查看 }, //设置状态栏颜色 statusBarBackground: '#9defbcg', //设置状态栏颜色,仅iOS可用 preloadLimit:5//预加载窗口数量限制(一旦超出,先进先出)默认不限制})
以下は、現在のページの URL を出力する例です:
mui.plusReady(function(){ console.log("当前页面URL:"+plus.webview.currentWebview().getURL());});
在mobile app开发过程中,经常遇到卡头卡尾的页面,此时若使用局部滚动,在android手机上会出现滚动不流畅的问题; mui的解决思路是:将需要滚动的区域通过单独的webview实现,完全使用原生滚动。具体做法则是:将目标页面分解为主页面和内容页面,主页面显示卡头卡尾区域,比如顶部导航、底部选项卡等;内容页面显示具体需要滚动的内容,然后在主页面中调用mui.init方法初始化内容页面。
mui.init({ subpages:[{ url:your-subpage-url,//子页面HTML地址,支持本地地址和网络地址 id:your-subpage-id,//子页面标志 styles:{ top:subpage-top-position,//子页面顶部位置 bottom:subpage-bottom-position,//子页面底部位置 width:subpage-width,//子页面宽度,默认为100% height:subpage-height,//子页面高度,默认为100% ...... }, extras:{}//额外扩展参数 }] });
styles:表示窗口属性,参考5+规范中的WebviewStyle;特别注意,height和width两个属性,即使不设置,也默认按100%计算;因此若设置了top值为非"0px"的情况,建议同时设置bottom值,否则5+ runtime根据高度100%计算,可能会造成页面真实底部位置超出屏幕范围的情况;left、right同理。
index.html的作用就是显示固定导航,list.html显示具体列表内容,列表项的滚动是在list.html所在webview中使用原生滚动,既保证了滚动条不会穿透顶部导航,符合app的体验,也保证了列表流畅滚动,解决了区域滚动卡顿的问题。 list.html就是index.html的子页面,创建代码比较简单,如下:
mui.init({ subpages:[{ url:'list.html', id:'list.html', styles:{ top:'45px',//mui标题栏默认高度为45px; bottom:'0px'//默认为0px,可不定义; } }]});
做 web app ,一个无法避开的问题就是转场动画; web 是基于链接构建的,从一个页面点击链接跳转到另一个页面,如果通过有刷新的打开方式,用户要面对一个空白的页面等待;如果通过无刷新的方式,用 Javascript 移入DOM节点(常见的 SPA 解决方案),会碰到很高的性能挑战:DOM节点繁多,页面太大,转场动画不流畅甚至导致浏览器崩溃; mui的解决思路是:单 webview 只承载单个页面的dom,减少dom层级及页面大小;页面切换使用原生动画,将最耗性能的部分交给原生实现。
mui.openWindow({ url:new-page-url, id:new-page-id, styles:{ top:newpage-top-position,//新页面顶部位置 bottom:newage-bottom-position,//新页面底部位置 width:newpage-width,//新页面宽度,默认为100% height:newpage-height,//新页面高度,默认为100% ...... }, extras:{ .....//自定义扩展参数,可以用来处理页面间传值 }, createNew:false,//是否重复创建同样id的webview,默认为false:不重复创建,直接显示 show:{ autoShow:true,//页面loaded事件发生后自动显示,默认为true aniShow:animationType,//页面显示动画,默认为”slide-in-right“; duration:animationTime//页面动画持续时间,Android平台默认100毫秒,iOS平台默认200毫秒; }, waiting:{ autoShow:true,//自动显示等待框,默认为true title:'正在加载...',//等待对话框上显示的提示内容 options:{ width:waiting-dialog-widht,//等待框背景区域宽度,默认根据内容自动计算合适宽度 height:waiting-dialog-height,//等待框背景区域高度,默认根据内容自动计算合适高度 ...... } }})
styles:表示窗口参数, 参考5+规范中的WebviewStyle ;特别注意, height 和 width 两个属性,即使不设置,也默认按 100% 计算;因此若设置了 top 值为非 "0px" 的情况,建议同时设置 bottom 值,否则 5+ runtime 根据高度 100% 计算,可能会造成页面真实底部位置超出屏幕范围的情况, left 、 right 同理。
extras:新窗口的额外扩展参数,可用来处理页面间传值;例如:
var webview = mui.openWindow({ url:'info.html', extras:{ name:'mui' }});console.log(webview.name);
控制台会输出"mui"字符串;注意:扩展参数仅在打开新窗口时有效,若目标窗口为预加载页面,则通过mui.openWindow方法打开时传递的extras参数无效。
createNew:是否重复创建相同id的 webview ;为优化性能、避免app中重复创建webview,mui v1.7.0开始增加 createNew 参数,默认为false;判断逻辑如下:若createNew为true,则不判断重复,每次都新建 webview ;若为fasle,则先计算当前App中是否已存在同样id的webview,若存在则直接显示;否则新创建并根据show参数执行显示逻辑;该参数可能导致的影响:若业务写在 plusReady 事件中,而 plusReady 事件仅首次创建时会触发,则下次再次通过 mui.openWindow 方法打开同样webview时,是不会再次触发 plusReady 事件的,此时可通过自定义事件触发;案例参考: http://ask.dcloud.net.cn/question/6514 ;
show:表示窗口显示控制。autoShow:目标窗口loaded事件发生后,是否自动显示;若目标页面为预加载页面,则该参数无效;aniShow表示页面显示动画,比如从右侧划入、从下侧划入等,具体可参考 5+规范中的AnimationTypeShow 。
waiting:表示系统等待框;mui框架在打开新页面时等待框的处理逻辑为:显示等待框-->创建目标页面webview-->目标页面loaded事件发生-->关闭等待框;因此,只有当新页面为新创建页面(webview)时,会显示等待框,否则若为预加载好的页面,则直接显示目标页面,不会显示等待框。waiting中的参数:autoShow表示自动显示等待框,默认为true,若为false,则不显示等待框;注意:若显示了等待框,但目标页面不自动显示,则需在目标页面中通过如下代码关闭等待框 plus.nativeUI.closeWaiting() ;。title表示等待框上的提示文字,options表示等待框显示参数,比如宽高、背景色、提示文字颜色等,具体可参考 5+规范中的WaitingOption 。
//tap为mui封装的单击事件,可参考手势事件章节document.getElementById('info').addEventListener('tap', function() { //打开关于页面 mui.openWindow({ url: 'examples/info.html', id:'info' });});
因没有传入styles参数,故默认全屏显示;也没有传入show参数,故使用slide-in-right动画,新页面从右侧滑入。
第一步,B页面loaded事件发生后,不自动显示
//A页面中打开B页面,设置show的autoShow为false,则B页面在其loaded事件发生后,不会自动显示;mui.openWindow({ url: 'B.html', show:{ autoShow:false }});
第二步,在B页面获取列表数据后,再关闭等待框、显示B页面
//B页面onload从服务器获取列表数据;window.onload = function(){ //从服务器获取数据 .... //业务数据获取完毕,并已插入当前页面DOM; //注意:若为ajax请求,则需将如下代码放在处理完ajax响应数据之后; mui.plusReady(function(){ //关闭等待框 plus.nativeUI.closeWaiting(); //显示当前页面 mui.currentWebview.show(); });}
mui框架将窗口关闭功能封装在mui.back方法中,具体执行逻辑是:若当前webview为预加载页面,则hide当前webview;否则,close当前webview。
在mui框架中,有三种操作会触发页面关闭(执行mui.back方法)。
点击包含.mui-action-back类的控件
在页面上,向右快速滑动
Android手机按下back按键
hbuilder中敲mheader生成的代码块,会自动生成带有返回导航箭头的标题栏,点击返回箭头可关闭当前页面,原因就是因为该返回箭头包含.mui-action-back类,代码如下:
<header class="mui-bar mui-bar-nav"> <a class="mui-action-back mui-icon mui-icon-left-nav mui-pull-left"></a> <h1 class="mui-title">标题</h1></header>
若希望在顶部导航栏之外的其它区域添加关闭页面的控件,只需要在对应控件上添加.mui-action-back类即可,如下为一个关闭按钮示例:
<button type="button" class='mui-btn mui-btn-danger mui-action-back'>关闭</button>
mui框架封装的页面右滑关闭功能,默认未启用,若要使用右滑关闭功能,需要在 mui.init() ;方法中设置 swipeBack 参数,如下:
mui.init({ swipeBack:true //启用右滑关闭功能});
mui框架默认会监听Android手机的back按键,然后执行页面关闭逻辑; 若不希望mui自动处理back按键,可通过如下方式关闭mui的back按键监听;
mui.init({ keyEventBind: { backbutton: false //关闭back按键监听 }});
除了如上三种操作外,也可以直接调用 mui.back() 方法,执行窗口关闭逻辑; mui.back() 仅处理窗口逻辑,若希望在窗口关闭之前再处理一些其它业务逻辑,则可将业务逻辑抽象成一个具体函数,然后注册为 mui.init 方法的 beforeback 参数; beforeback 的执行逻辑为:
执行 beforeback 参数对应的函数若返回false,则不再执行mui.back()方法;否则(返回true或无返回值),继续执行mui.back()方法;
示例:从列表打开详情页面,从详情页面再返回后希望刷新列表界面,此时可注册beforeback参数,然后通过自定义事件通知列表页面刷新数据,示例代码如下:
mui.init({ beforeback: function(){ //获得列表界面的webview var list = plus.webview.getWebviewById('list'); //触发列表界面的自定义事件(refresh),从而进行数据刷新 mui.fire(list,'refresh'); //返回true,继续页面关闭逻辑 return true; }});
注意: beforeback 的执行返回必须是同步的(阻塞模式),若使用 nativeUI 这种异步js(非阻塞模式),则可能会出现意想不到的结果;比如:通过 plus.nativeUI.confirm() 弹出确认框,可能用户尚未选择,页面已经返回了( beforeback 同步执行完毕,无返回值,继续执行 mui.back() 方法, nativeUI 不会阻塞js进程):在这种情况下,若要自定义业务逻辑,就需要复写 mui.back 方法了;如下为一个自定义示例,每次都需要用户确认后,才会关闭当前页面。
//备份mui.back,mui.back已将窗口关闭逻辑封装的比较完善(预加载及父子窗口),因此最好复用mui.backvar old_back = mui.back;mui.back = function(){ var btn = ["确定","取消"]; mui.confirm('确认关闭当前窗口?','Hello MUI',btn,function(e){ if(e.index==0){ //执行mui封装好的窗口关闭逻辑; old_back(); } });}
注意:自定义关闭逻辑时,一定要重写 mui.back ,不能简单通过 addEventListener 增加back按键监听, 因为 addEventListener 只会增加新的执行逻辑,老的监听逻辑依然会执行;
这个系列的教程我准备带大家一起实现音乐播放器和即时通讯的功能,先上图不多说:
开始的页面效果很简单,就是一个tab bar页面切换组件,我们重点讲解实现方法,至于美化是后面的事。在开始项目之前我先抄了文档的内容,不是为了凑内容,只是想让新手在开始项目之前还是多看看基本概念,俗话说磨刀不误砍柴工,我们对mui的设计思路有一定了解之后写起来才能得心应手。
相信大家对于 mui 的双 webview 模式有初步认识,我们可以分析一下我们接下来要做的这个的实际例子,首先我们的入口文件 index.html 是一个包括头部和底部的导航栏的 webview ,中间是一个动态的 webview ,我们通过点击底部导航栏进行页面切换,并且动态的改变顶部导航栏的内容。
下面我们新建一个mui项目,这里我命名为M-BOX:
点击【文件】=》【新建】=》【移动APP】
设置应用名称、文件存储路径、选择模板
js,css,fonts文件的详细介绍请戳这里 手把手教你开发HelloWord
HBuilder对前端代码的调试方式 这个是很重要的一个部分,限于篇幅,这里不能讲解,请自己看教程。
相信很多人看了前面那么多文档介绍内心肯定是崩溃的,其实我也是,毕竟写了那么多还没有开始写代码我也是拒绝的,只是考虑到很多新手对于找文档这事不一定有经验,那还是先贴一下,大不了回过头再去看咯。
好,那我们开始写布局文件:
在 MUI开发注意事项 这篇文章中提到了几个重要的注意事项,我们在一个就注意一下会比较好,这里不再一一详细列举了,读者自己去看。
文章中DOM结构提到:
固定栏靠前
所谓的固定栏,也就是带有.mui-bar属性的节点,都是基于fixed定位的元素;常见组件包括:顶部导航栏(.mui-bar-nav)、底部工具条(.mui-bar-footer)、底部选项卡(.mui-bar-tab);这些元素使用时需遵循一个规则:放在.mui-content元素之前,即使是底部工具条和底部选项卡,也要放在.mui-content之前,否则固定栏会遮住部分主内容;
一切内容都要包裹在mui-content中
除了固定栏之外,其它内容都要包裹在.mui-content中,否则就有可能被固定栏遮罩,原因:固定栏基于Fixed定位,不受流式布局限制,普通内容依然会从top:0的位置开始布局,这样就会被固定栏遮罩,mui为了解决这个问题,定义了如下css代码:
.mui-bar-nav ~ .mui-content { padding-top: 44px;}.mui-bar-footer ~ .mui-content { padding-bottom: 44px;}.mui-bar-tab ~ .mui-content { padding-bottom: 50px;}
我们这里重点看这两条规则,因为这个对于我们正确布局是至关重要的。
下面我们体验一下 hbuilder 的代码块功能,在 index.html 文件的 body 之间输入 mheader ,回车试试。
哈哈,页面头部出来了,不错,这里我们然后删除下面的:
<a class="mui-action-back mui-icon mui-icon-left-nav mui-pull-left"></a>
先去掉返回箭头。
然后继续输入 mtab ,回车,底部导航栏也出来了,我们修改一下导航栏的内容,把代码稍微调整一下。
整体代码如下:
<!DOCTYPE html><html><head> <meta charset="utf-8"> <meta name="viewport" content="width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no" /> <title>M-BOX</title> <link href="css/mui.min.css" rel="stylesheet"/></head><body> <header class="mui-bar mui-bar-nav"> <h1 class="mui-title">标题</h1> </header> <nav class="mui-bar mui-bar-tab"> <a class="mui-tab-item mui-active"> <span class="mui-icon mui-icon-home"></span> <span class="mui-tab-label">首页</span> </a> <a class="mui-tab-item"> <span class="mui-icon mui-icon-chatboxes"></span> <span class="mui-tab-label">消息</span> </a> <a class="mui-tab-item"> <span class="mui-icon mui-icon-gear"></span> <span class="mui-tab-label">设置</span> </a> </nav> <script src="js/mui.min.js"></script>; <script type="text/javascript" charset="utf-8"> // 初始化mui.init()写在这里 </script></body></html>
首页的静态布局我们写完了,我们接下来新建三个含mui的html文件:
选择工程名,邮件就可以看到【新建】,然后就是选择【目录】新建文件夹和【html文件】新建含mui的html文件。我们新建一个文件夹 html ,并且在html文件夹下新建, home.html , message.html , setting.html 。
在三个页面 body 之间分别输入 mbody ,就可以开始分别写页面了,比如可以先在页面上写上文件名,我们先来完善首页的子页切换逻辑。
主要方法就是用 plus.webview.create 和 plus.webview.hide();
//设置默认打开首页显示的子页序号;var Index=0;//把子页的路径写在数组里面var subpages = ['html/home.html','html/message.html','html/setting.html'];//所有的plus-*方法写在mui.plusReady中或者后面。mui.plusReady(function() { //获取当前页面所属的Webview窗口对象 var self = plus.webview.currentWebview(); for (var i = 0; i < 3; i++) { //创建webview子页 var sub = plus.webview.create( subpages[i], //子页url subpages[i], //子页id { top: '45px',//设置距离顶部的距离 bottom: '50px'//设置距离底部的距离 } ); //如不是我们设置的默认的子页则隐藏,否则添加到窗口中 if (i != Index) { sub.hide(); } //将webview对象填充到窗口 self.append(sub); }});
注:如果Index不是0,需要将nav下的a标签中的 .mui-active 属性写到对应的a标签下。
执行完我们会发现home.html的内容显示出来了,但是底部切换还不能,因为这里我们还没有监听底部的点击事件。在进行下一步之前,我们可以先做一个小实验,将上面的代码中的top或者bottom改为0,我们会发现,底部栏或者底部栏会被覆盖,这是因为mui一个重要的 潜规则 , 父子结构的页面子页面会比父页面层级高 ,说白了就是子页面可以盖住父页面。这会导致开发者常犯的一个错误:将弹出层或者弹出菜单写在父页面被子页面盖住的bug。
这里的apend()方法在html5plus文档中没有提到,这里的这个方法是将webview对象填充到窗口的方法。
mui 内部封装了一些常用的方法,其中DOM选择器、事件绑定等事件管理。具体可以参考文档: 选择器 、 事件管理 。
mui()
mui使用css选择器获取HTML元素,返回mui对象数组。
mui("p") :选取所有 p 元素
mui("p.title") :选取所有包含 .title 类的
元素
若要将mui对象转化成dom对象,可使用如下方法(类似jquery对象转成dom对象):
//obj1是mui对象var obj1 = mui("#title");//obj2是dom对象var obj2 = obj1[0];
.on( event , selector , handler )
event
Type: String
需监听的事件名称,例如:'tap'
selector
Type: String-选择器
handler
Type: Function( Event event )
事件触发时的回调函数,通过回调中的event参数可以获得事件详情
除了可以使用addEventListener()方法监听某个特定元素上的事件外, 也可以使用.on()方法实现批量元素的事件绑定。
这里我们将为底部导航按钮添加事件:
//选项卡点击事件mui('.mui-bar-tab').on('tap', 'a', function(e) { alert(true);});
当我们点击底部选项卡的时候会弹出 true ,这不够,我们要能够分辨当前对象具体是哪一个,有两种思路:
第一我们能够知道当前点击的a标签所在序号就好了,就是找到 index ,然后根据上面那个 subpages 数组,利用 plus.webview.show(subpages[index]) 方法显示。
我们给当前点击的a标签添加一个可以识别的属性,然后根据那个属性获取当前a的特征,然后就可以显示点击的子页,隐藏当前子页。
第一种方法需要遍历此案获取 index ,第二种方法添加一个 href 很容易拿到子页 id ,我们采用第二种方案。
getAttribute()
getAttribute() 方法返回指定属性名的属性值。提示:如果您希望以 Attr 对象返回属性,请使用 getAttributeNode。
于是我们可以这样写:
//当前激活选项var activeTab = subpages[Index],title=document.querySelector(".mui-title");//选项卡点击事件mui('.mui-bar-tab').on('tap', 'a', function(e) { //获取目标子页的id var targetTab = this.getAttribute('href'); if (targetTab == activeTab) { return; } //更换标题 title.innerHTML = this.querySelector('.mui-tab-label').innerHTML; //显示目标选项卡 plus.webview.show(targetTab); //隐藏当前选项卡 plus.webview.hide(activeTab); //更改当前活跃的选项卡 activeTab = targetTab;});
虽然最后实现的效果很简单,好像直接看demo就可以写出来,但是新手甚至写了一段时间的同学也不见得对 webview 掌握得很好,这篇文章花了很长的篇幅去讲解 webview 的用法,旨在为新手建立一种学习mui这边的思路,那就是先看html5plus里面的模块,然后看mui对应的文档,最后看 hello mui 的 demo ,把握这种学习路线个人觉得是一种最佳的方案。本文作为系列文章第一篇讲代码的,所以做了很多铺垫,所以有一定基础的同学可能会觉得写得并没有什么看点,后面的肯定会有所不一样的。下一篇讲解的是网络请求 XMLHttpRequest 模块,下一讲会结合 mui.ajax 和网易云音乐 API 一起讲解。
文章原始地址是我博客地址:
http://zhaomenghuan.github.io
最近在开始写MUI小白文档,地址:
https://zhaomenghuan.gitbooks.io/mui/content/
MUI小白文档git仓库地址:
https://git.gitbook.com/zhaomenghuan/mui.git