Flex移动应用程序开发的技巧和窍门——第三部分

这是关于 Flex 移动应用程序开发的技巧和窍门系列文章的第三部分内容第一部分内容主要集中讨论了视图之间以及应用程序执行之间切换时的数据处理。第二部分则主要涵盖了应用程序动作条和标签组件风格化方面的内容在这一部分中,你将会学到如何控制动作条和标签组件的可见性,以及如何把标签组件移动到应用程序的顶端。

动作条和标签的隐藏

在使用基于 TabbedViewNavigatorApplication 的Flex移动应用程序的过程中,你可能需要隐藏 动作条组件标签组件。例如,在特定视图下,你可能想获取更大的屏幕空间,或者,你只是想根据个人喜好设置显示界面。在这些情况下,你可以使用 View 类中的两个有效的道具: actionBarVisible 以及 tabBarVisible,来达到预期的效果

为了说明 actionBarVisibletabBarVisible 究竟是如何工作的,我创建了一个简单的基于TabbedViewNavigatorApplication 的移动应用程序。如果你想查看完整的源代码,并亲自调试这个应用程序,下载这篇文章中用到的样本文件,并把项目导入到 Adobe Flash Builder 中去。

(这个应用程序的)代码中含有三个视图:View1,View2和View3。在第一个视图中含有控制ActionBar和TabBar可见性的按钮。

正如你在下面 View1 代码中看到的那样,你可以通过设置 actionBarVisibletabBarVisible来控制它们的可见性:

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <s:View xmlns:fx="http://ns.adobe.com/mxml/2009" 
  3.         xmlns:s="library://ns.adobe.com/flex/spark" title="View1">  
  4.    
  5.     <fx:Script>  
  6.         <![CDATA[  
  7.             protected function onActionBarClickHandler(event:MouseEvent):void  
  8.             {  
  9.                 if (actionBarVisible)  
  10.                 {  
  11.                     this.actionBarVisible = false;  
  12.                     b.label="Show ActionBar";  
  13.                 }  
  14.                 else {  
  15.                     this.actionBarVisible = true;  
  16.                     b.label="Hide ActionBar";  
  17.                 }  
  18.             }  
  19.    
  20.             protected function onTabBarClickHandler(event:MouseEvent):void  
  21.             {  
  22.                 if (tabBarVisible)  
  23.                 {  
  24.                     this.tabBarVisible = false;  
  25.                     b2.label="Show TabBar";  
  26.                 }  
  27.                 else {  
  28.                     this.tabBarVisible = true;  
  29.                     b2.label="Hide TabBar";  
  30.                 }  
  31.             }  
  32.         ]]> 
  33.     </fx:Script>  
  34.     <s:VGroup width="100%" height="100%" horizontalAlign="center" top="50">  
  35.         <s:TextArea text="This is View 1" editable="false"/>  
  36.         <s:HGroup>  
  37.             <s:Button id="b" label="Hide ActionBar" click="onActionBarClickHandler(event)"/>  
  38.             <s:Button id="b2" label="Hide TabBar" click="onTabBarClickHandler(event)"/>  
  39.         </s:HGroup>  
  40.     </s:VGroup>  
  41. </s:View> 

当通过(点击)标签或程序控制来实现不同视图之间的切换时,这些项目的可见性只在View1中有所改变(如图1所示)。

应用程序的开始界面(左),动作条隐藏界面(中)以及动作条和标签隐藏界面(右)

图1. 应用程序的开始界面(左),动作条隐藏界面(中)以及动作条和标签隐藏界面(右)

其他视图不受影响(如图2所示)。

点击 View1 中按钮时,View2 界面保持不变

图2. 点击 View1 中按钮时,View2 界面保持不变

这并没有包含如下的主应用程序的代码

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <s:TabbedViewNavigatorApplication xmlns:fx="http://ns.adobe.com/mxml/2009" 
  3.                                   xmlns:s="library://ns.adobe.com/flex/spark">  
  4.     <s:ViewNavigator label="View1" width="100%" height="100%" firstView="views.View1"/>  
  5.     <s:ViewNavigator label="View2" width="100%" height="100%" firstView="views.View2"/>  
  6.     <s:ViewNavigator label="View3" width="100%" height="100%" firstView="views.View3"/>  
  7. </s:TabbedViewNavigatorApplication> 

如果你亲自创建并测试这个应用程序,你会注意到,你隐藏了 ActionBar,切换到其他的视图,然后切换回 View1 时,动作条就会再次出现。之所以会有这样的情况出现,是因为当你再次把视图切换回来的时候,视图发生了重建。当你在特定的视图中,希望 ActionBar 一直处于隐藏状态,你可以通过几种不同的方法来实现。其中一种方法就是在视图中设置destructionPolicy="never";这样 ActionBar 被隐藏之后就不会再自动重现,因为视图不会再自动重建。另一种方法是在根 View 标签中设置viewActivate="actionBarVisible=false" ,这样每次该视图被激活,ActionBar 都会处于隐藏状态。但是,使用这种方法存在的一个问题是,当这个视图被激活的时候,用户都会在视图隐藏之前,看到 ActionBar 短暂的出现,这可能是我们所不希望看到的。第三种方法是,在你的根选项卡应用程序文件中加入如下的代码,来设置 tabbednavigatorIndexChangeEvent 上的 ViewNavigator 组件中的当前视图 actionBarVisible 属性

  1. <?xml version="1.0" encoding="utf-8"?> 
  2. <s:TabbedViewNavigatorApplication xmlns:fx="http://ns.adobe.com/mxml/2009"  
  3.                          xmlns:s="library://ns.adobe.com/flex/spark"      
  4.                          applicationComplete="onApplicationComplete(event)">     
  5. <fx:Script> 
  6. <![CDATA[ 
  7. import mx.events.FlexEvent; 
  8. import spark.events.IndexChangeEvent; 
  9. protected function onApplicationComplete (event:FlexEvent):void
  10.     vn1.actionBar.visible=false;                
  11.     tabbednavigator.addEventListener(IndexChangeEvent.CHANGE,onChange);        
  12.              
  13. protected function onChange(event:IndexChangeEvent):void 
  14.     if (event.newIndex==0
  15.         vn1.activeView.actionBarVisible=false
  16.                  
  17.              
  18. ]]> 
  19. </fx:Script> 
  20.      
  21.      
  22.     <s:ViewNavigator id="vn1" label="View1" width="100%" height="100%" firstView="views.View1"/> 
  23.     <s:ViewNavigator label="View2" width="100%" height="100%" firstView="views.View2"/> 
  24.     <s:ViewNavigator label="View3" width="100%" height="100%" firstView="views.View3"/> 
  25. </s:TabbedViewNavigatorApplication> 

标签定位到应用程序的顶端

正如你在本文第二个示例应用程序中看到的那样,认情况下,Flex 4.5 TabbedViewNavigatorApplication 的标签按钮是位于(应用程序的)底部的(如图3所示)。

标签的默认位置

图3. 标签认位置

但是,有的时候你可能更希望你的应用程序标签位于屏幕的顶端。

按照如下步骤,移动标签到你的应用程序的顶端:

  1. 找到并复制 TabbedViewNavigatorSkin.as 类到你的 mobile 项目目录 skins中去。在我的 Mac 上,我可以在如下目录中找到这个皮肤类(对 Flash Builder 4.5.1 而言):/Applications/Adobe Flash Builder 4.5/sdks/4.5.1/frameworks/projects/mobiletheme/src/spark/skins/mobile/TabbedViewNavigatorSkin.as
  2. 要么在你的主应用文件中要么在一个外部样式表单中使用如下的皮肤类来设置你的CSS风格:
  1. <fx:Style> 
  2.         @namespace s "library://ns.adobe.com/flex/spark"
  3.         s|TabbedViewNavigator 
  4.         { 
  5.             skinClass: ClassReference("skins.TabbedViewNavigatorSkin"); 
  6.         } 
  7. </fx:Style> 


  1. 在你新复制的 TabbedViewNavigatorSkin 类中,更改类顶部的 package 命令来反映它在你本地 skins 文件夹中的新的位置;将 package 命令的第一行由 package spark.skins.mobile 替换为package skins
  2. 滚动鼠标至皮肤类底端,找到 layoutContents() 函数。这个函数是用来设置标签条和内容布局的。这里的内容指的是你的 View 类的子元素,例如,应用程序中的 UI 组件以及结果 List 等。
  1. override protected function layoutContents(unscaledWidth:Number,unscaledHeight:Number):void 
  2.     super.layoutContents(unscaledWidth,unscaledHeight); 
  3.   
  4.     var tabBarHeight:Number = 0;  
  5.   
  6.     if (tabBar.includeInLayout) 
  7.     { 
  8.         tabBarHeight = Math.min(tabBar.getPreferredBoundsHeight(),unscaledHeight); 
  9.         tabBar.setLayoutBoundsSize(unscaledWidth,tabBarHeight); 
  10.         tabBar.setLayoutBoundsPosition(0,unscaledHeight - tabBarHeight); 
  11.         tabBarHeight = tabBar.getLayoutBoundsHeight();  
  12.   
  13.         // backgroundAlpha is not a declared style on buttonbar 
  14.         // TabbedViewNavigatorbuttonbarSkin implements for overlay support 
  15.         var backgroundAlpha:Number = (_isOverlay) ? 0.75 : 1
  16.         tabBar.setStyle("backgroundAlpha",backgroundAlpha); 
  17.     } 
  18.   
  19.     if (contentGroup.includeInLayout) 
  20.     { 
  21.         var contentGroupHeight:Number = (_isOverlay) ? unscaledHeight : Math.max(unscaledHeight - tabBarHeight,0); 
  22.         contentGroup.setLayoutBoundsSize(unscaledWidth,contentGroupHeight); 
  23.         contentGroup.setLayoutBoundsPosition(0,0); 
  24.     } 


在这里,关键是 tabBar.setLayoutBoundsPosition(x,y) 函数。如果你修改(这个函数代码中的 y 值为 0标签就会被置于应用程序的顶端:

  1. if (tabBar.includeInLayout) 
  2.     tabBarHeight = Math.min(tabBar.getPreferredBoundsHeight(),unscaledHeight); 
  3.     tabBar.setLayoutBoundsSize(unscaledWidth,tabBarHeight); 
  4.     tabBar.setLayoutBoundsPosition(0,0); 
  5.     tabBarHeight = tabBar.getLayoutBoundsHeight();  
  6.   
  7.     // backgroundAlpha is not a declared style on buttonbar 
  8.     // TabbedViewNavigatorbuttonbarSkin implements for overlay support 
  9.     var backgroundAlpha:Number = (_isOverlay) ? 0.75 : 1
  10.     tabBar.setStyle("backgroundAlpha",backgroundAlpha); 
  11. if (contentGroup.includeInLayout) 
  12.     var contentGroupHeight:Number = (_isOverlay) ? unscaledHeight : Math.max(unscaledHeight - tabBarHeight,0); 
  13.     contentGroup.setLayoutBoundsSize(unscaledWidth,contentGroupHeight); 
  14.     contentGroup.setLayoutBoundsPosition(0,0); 


标签置于顶端的应用程序

图4. 标签置于顶端的应用程序

如果你想去掉动作条来制造更大的屏幕空间(如图5所示),你只需要进入 View 类并设置 actionBarVisible;例如:

  1. <s:View xmlns:fx="http://ns.adobe.com/mxml/2009" 
  2.         xmlns:s="library://ns.adobe.com/flex/spark" title="Search Location..." actionBarVisible="false" xmlns:json="com.adobe.serialization.json.*" > 

标签位于顶端且没有动作条的应用程序

图5. 标签位于顶端且没有动作条的应用程序

相关文章

一:display:flex布局display:flex是一种布局方式。它即可以...
1. flex设置元素垂直居中对齐在之前的一篇文章中记载过如何...
移动端开发知识点pc端软件和移动端apppc端软件是什么,有哪些...
最近挺忙的,准备考试,还有其他的事,没时间研究东西,快周...
display:flex;把容器设置为弹性盒模型(设置为弹性盒模型之后...
我在网页上运行了一个Flex应用程序,我想使用Command←组合键...