`

从零开始编写一个flex组件。FLEX自定义控件。

    博客分类:
  • Flex
阅读更多

来自:http://www.5uflash.com/Html/flex/223818539.html
文原文:Component Class
原文地址:http://weblogs.macromedia.com/pent/archives/2007/10/component_class.cfm
原文作者:Peter Ent
                                  组件类
我经常看到这个话题——编写组件。通过这个例子,我将告诉你如何从零开始编写一个组件。我将要写几篇文章来探讨这个主题,在最后你就可以创建自己的组件了。
我把自己选择的这个组件叫做循环选择按钮。这个组件的显示的是dataProvider 中的某条纪录和两个箭头组成的一个圆环。当你选择按钮的时候箭头就会旋转一点同时显示dataProvider 中的下一个值。你可以把它想象成一个没有下拉列表的ComboBox组件。

你可以试着点击按钮,然后你就可以看到它是如何在各个值之间循环的。

Flex框架
想要创建一个Flex组件你必须理解Flex框架。一个组件的创建需要几个阶段,而在这些阶段Flex框架将多次遍历组件的继承层次来决定布局。
举个例子,假设一个Application中有两个VBox。一个VBox中有一些Button控件而另一个中有一些Label控件。要创建这些组件,Flex框架需要首先创建Application,然后创建VBox,然后是VBox的子控件。这是一次遍历。
如果没有对任何一个VBox指定高度或者宽度,那么你可能希望VBox足够大以便能够包含所有的子控件,对吗?为了计算VBox的大小,Flex框架需要决定它的每个子控件的大小。由于Button和Label并没有明确指定大小,所以Flex框架还需要决定它们的大小。这是另一次遍历。
一旦所有的计算都完毕的时候,Flex框架需要获得组件的大小属性和它们的位置。这又是一次遍历。
如同你看到的那样,创建组件并不是一件简单的事情,但是一旦你掌握了其中的窍门,它也并不是那么地困难。
为了方便地创建组件,Flex框架调用各个组件中的特定方法。通过实现这些方法,可以简单地使你的组件很好地适应框架。
有两种创建组件的方法:扩展一个已经实现了你想要的大部分功能的既有组件,或者由所有类的父类也就是最基础的类“从零开始”创建组件。
扩展一个既有的组件是最经常用的方法,而且他是你编写一个Flex程序的时候一直在使用的方法。当你使用一个根标签<mx:Application>创建主程序文件的时候,你就创建了一个组件(扩展自Application)。而当你创建了一个根标签为其它任意组件的MXML文件的时候,也相当于创建了一个组件;不论你是通过MXML还是ActionScript进行扩展组件的。
程序初稿
我们将会以修改HBox组件开始,但是从根本上来说我们想要扩展UIComponent。使用HBox可以很好地阐述概念。
下载文件
上面的文件是程序源代码,并且包含了一个用来绘图的图标。
让我们来看一下这个组件的设计,很显然有两部分:一个圆环按钮和一个标签(Label)。但是由于我们想要通过点击标签(Label)在各个选项之间循环,而Button当鼠标放上去或者进行点击的时候都会提供一些反馈,所以使用一个Button看起来更实用。但是一个Button看起来一点也不像上面的那个控件,所以可能使用LinkButton看起来会好一点。
使用HBox作为根标签创建一个新的MXML组件并将其命名为CycleSelectButtonV1.mxml(V1代表version 1)。向其中添加两个子控件:一个Image和一个LinkButton(将它的id设为”linkButton”)。将Image的大小设成20×20并将HBox的verticalAlign属性设为”middle”。如果你是使用FlexBuilder来创建这个组件的话,请将HBox预设的width和height去掉。
对Image标签做以下更改:
<mx:Image source="@Embed('../assets/cycle_component.gif')" 
    width="20" height="20" />
你可以像下面这样使用这个新组件:
<CycleSelectButtonV1 dataProvider="{choices}"
    change="handleCycleChange(event)" />
这个HBox组件并没有dataProvider属性,也没有change事件。这些东西是需要你进行自定义。
事件和属性
这个组件将要分派一个change事件,所以你需要将这个告诉Flex编译器组件。将下面的代码添加到HBox根标签:
<mx:Metadata>
    Event(name="change",type="flash.events.Event")]
</mx:Metadata>
Event 元数据标签告诉Flex编译器在MXML标签中可以包含 change=”…”。用来作为事件的数据类型的类是缺省的,不过我想要明确写出来,这样就对哪个事件类会出现在事件处理函数中就没有疑问了。
要添加dataProvider这个属性,你需要编写一点ActionScript代码。将下面的Script块添加到Metadata标签下面:
<mx:Script>
<![CDATA[
]]>
</mx:Script>
<![CDATA 和 ]]>语法是为了告诉XML其中的任何东西都不需要做XML解析。在Script标签中添加CDATA块并不是必须的,但是加入你使用像 < 的符号,XML解析器会认为你正在写一个新的标签!
将下面的代码添加到 CDATA 块中:
import mx.collection.ArrayCollection;
private var _dataProvider:ArrayCollection;
public function set dataProvider( value:ArrayCollection ) : void
{
    _dataProvider = value;
}
public function get dataProvider() : ArrayCollection
{
    return _dataProvider;
}
这是编写一个属性的标准方式——使用set和get函数,并且变量值和函数名是一样的,只是前面加了一个下划线。有时候你可能会看到这种方式被称作后台变量(backing variable)。
这时候你可以测试一下组件了,而且可以将dataProvider和change添加到组件标签中;尽管它们不会起到任何作用。
当我思考这个组件的时候,我想到要像为ComboBox提供数据一样为这个组件提供数据。下面是一个示例:
[ {label:"Apples", value:1}, {label:"Oranges", value:2}, etc. ]
你可以添加所有你想要让标签(Label)显示的数据,接下来我将告诉你如何实现这个功能。
commitProperties
到现在为止,组件并不知道如何在LinkButton中显示数据。所以我们需要一些ActionScript代码。将下面的代码添加到Script块中:
override protected function commitProperties() : void
{
    super.commitProperties();
    // we'll fill this in below
}
部分Flex框架周期包括一个对commitProperties()函数的调用。在组件的所有属性都设定完毕之后就会调用这个函数,这一点很重要,因为你不知道在一个属性的set函数中是否调用了另外一个属性的set函数。例如,如果第三个属性的设定需要用到其它两个的值,那么设定第三个属性的唯一合理的地方就是在commitProperties()中。
我们将使用commitProperties()将LinkButton的label属性设定为dataProvider中的一个值。这个例子中它将会被设定为第一个条目。将下面的代码添加到super.commitProperties()下面:
linkButton.label = dataProvider[0].label;
这里可能会发生很多错误:dataProvider属性可能从来没有被设定或者它的第一个条目可能并没有label属性。不过现在我们假设所有的东西都设定好了。
再次运行这个程序你将会看到LinkButton的label变成了”Apples”。
如果在主程序文件中添加一些代码对CycleSelectButtonV1的dataProvider做一些更改,你觉得会发生什么事情?如果你更改了一个ComboBox 或者 DataGrid的dataProvider将会发生什么事情?ComboBox 或者 DataGrid将会显示新的值对不对?你肯定希望CycleSelectButton有同样的效果。
如果你这样写:cycleButton.dataProvider = newValue,这就是对dataProvider调用了组件的set函数:
public function set dataProvider( value:ArrayCollection ) : void
{
    _dataProvider = value;
}
重点提示:其中的_dataProvider发生了改变,但是LinkButton的label并没有改变。你可能会尝试将linkButton.label=value[0].label添加到这个set函数中,但是我希望你看到这样做有很多问题。首先,它“看上去”就不对。而且真正的问题是,当组件正在创建属性还未设定完毕的时候,LinkButton的label属性可能并不能被赋值。当然,在那之后它将可以被赋值,但是在属性设定阶段很多组件都不可能很容易地被改变。
可以被赋值的地方就是commitProperties()函数。调用commitProperties()函数最合理的地方就是set函数。尽管这“看上去”也不对,而且,它所做的事情不但和上面所做的是一样的,还具有相同的问题。
接下来我们就想要通知Flex框架commitProperties()函数需要被调用,而这只需要在set函数中调用invalidateProperties()函数就可以了。在Flex框架中invalidateProperties()函数的作用是设定一个标记(flag),这个函数的好处是你可以设定100个属性并调用invalidateProperties()函数100次,而commitProperties()只会被调用一次,非常有效率。对set函数作以下更改:
public function set dataProvider( value:ArrayCollection ) : void
{
    _dataProvider = value;
    invalidateProperties();
}
循环显示数据
上面的代码只是为了下一件事情做准备:将LinkButton的label显示dataProvider的下一条目。
第一个条目已经被直接写到了commitProperties()函数中,很显然这里我们需要一个变量来保存条目的索引。嗯,selectedIndex 这个变量听起来是个不错的选择,而且它和ComboBox以及很多其他的Flex控件也保持一致。为selectedIndex也建立一个set函数和一个get函数:
private var _selectedIndex:int = 0;
public function set selectedIndex( value:int ) : void
{
    _selectedIndex = value;
    invalidateProperties();
}
public function get selectedIndex() : int
{
    return _selectedIndex;
}
你还需要更改commitProperties函数中的一行代码来使用selectedIndex:
linkButton.label = dataProvider[selectedIndex].label;
然后让LinkButton的click事件调用一个函数来在索引之间切换:
private function handleClick() : void
{
    selectedIndex = selectedIndex + 1;
    if( selectedIndex >= dataProvider.length ) selectedIndex = 0;
    linkButton.label = dataProvider[selectedIndex].label;
}
当然也要为LinkButton添加click事件:
<mx:LinkButton label="linkButton" click=handleClick()" />
现在运行程序并点击组件中的LinkButton,它应该循环显示dataProvider中所有条目的label。当点击LinkButton的时候就会调用handleClick()函数,这会使selectedIndex的值加1,所以当commitProperties被调用的时候就会显示与新的selectedIndex对应的dataProvider中条目。很酷吧?哈!
分派事件
最后,我们还要在selectedIndex发生改变的时候分派一个change事件。你可以在selectedIndex 的set函数中实现这个功能,但是这也意味着任何通过ActionScript对selectedIndex所做的更改都会分派这个事件,这不符合标准的Flex工作流程。所以,我们将在LinkButton的click事件处理函数中分派这个事件:
private function handleClick() : void
{
    selectedIndex = selectedIndex + 1;
    if( selectedIndex >= dataProvider.length ) selectedIndex = 0;
    linkButton.label = dataProvider[selectedIndex].label;
    dispatchEvent( new Event(Event.CHANGE) );
}
在主程序中你可以处理这个change事件并使用组件的selectedIndex来查看当前选中的是哪个条目。
总结
你可以使用任何Flex容器(例如HBox, VBox, Canvas)为基础来编写自己的组件,你只需要用MXML标签将向其中添加一些子组件,创建一个Script块并添加一些属性(使用set和get函数),然后覆写commitProperties()函数来应用这些属性。
在下一篇文章中我将会用ActionScript写一个相同的组件并且使其在每次被点击的时候旋转那个圆环图像。

分享到:
评论

相关推荐

    Flex 分页控件组件

    使用这样的组件,开发者可以快速集成分页功能,而无需从零开始编写代码。这将节省大量的时间和精力,让开发者可以专注于应用程序的核心业务逻辑。 从文件名“WAIPaginatorTest”来看,这可能是一个测试示例或演示...

    FLEX学习资料,从零基础开始学FLEX

    通过学习这些资料,你可以从零开始逐步掌握FLEX开发,从搭建开发环境、编写基础代码,到设计复杂的用户界面和实现数据交互,最终成为一个熟练的Flex开发者。在学习过程中,不断实践和理解Flex的组件模型、数据绑定...

    手把手教你学flex教程

    Flex教程详解:从零开始掌握Flex开发 Flex是一种基于Adobe Flash Platform的开源框架,主要用于构建富互联网应用程序(RIA)。在本教程中,我们将通过一系列实际示例,逐步探索Flex的基础知识,帮助初学者快速入门...

    Flex教程案例

    首先,从零开始的部分将涵盖Flex SDK的安装和配置,以及集成开发环境(IDE)的选择,如Flash Builder或IntelliJ IDEA。你将学习如何设置新的Flex项目,并了解项目结构。 接下来,教程会详细介绍Flex中的控件。Flex...

    基于WEB的Flash 统计图控件(Flex---FusionCharts_Evaluation v3)及其中文开发指南

    - **实例演示**:通过实际案例展示如何从零开始创建一个完整的图表应用,包括代码示例和步骤解析。 5. **应用实践**: 开发者可以利用这些知识点来创建实时更新的业务仪表板,可视化销售数据,或者在教育领域制作...

    google flex图文混排 空间

    总的来说,这个压缩包提供了一个关于如何在Flex中实现高效图文混排的解决方案,通过预封装的`RichTextField` 控件,开发者可以快速创建出美观且功能丰富的用户界面,而无需从零开始编写自定义组件。通过学习和使用`...

    《Flex完全自学手册》源码+电子教案(PPT)

    《Flex完全自学手册》是一本深入探讨Adobe Flex技术的专业书籍,该书的源码和电子教案(PPT)提供了一套全面的学习资源,旨在帮助读者从零基础开始掌握Flex编程。Flex是Adobe公司开发的一个开源框架,主要用于构建富...

    FLEX教程-FLEX教程

    - **自定义组件**: 扩展 Flex 的内置组件,或者从零开始创建新的 UI 控件。 - **实现**: - 定义新的事件类。 - 创建自定义组件类,并实现相应的逻辑。 #### 13. Demo7:页面跳转 - **页面管理**: Flex 应用中...

    flex viewer

    Flex Viewer提供了大量的预置组件,如地图、图层控制、查询工具等,开发者可以通过配置XML文件来定制自己的GIS应用,而无需从零开始编写代码。 3. **GIS与WebGIS**:GIS(Geographic Information System)是用于...

    flex-input,支持emoji、媒体附件和拍照的android文本输入小部件.zip

    通过这个组件,开发者可以快速地在自己的应用中添加这些功能,而无需从零开始编写大量的自定义代码。 1. **支持Emoji表情**: "flex-input"允许用户在输入框中直接插入emoji表情,这对于社交媒体、聊天应用等需要...

    arcgisviewer flex下地图入门实例

    ArcGIS Viewer for Flex是Esri公司提供的一个强大的工具,它允许开发者构建交互式的Web地图应用程序,无需从零开始编写复杂的GIS代码。 首先,我们需要了解Flex的基础。Flex是一种基于ActionScript 3的开源框架,...

    yingyan.rar_actionscript_flex_yingyan

    Flex的组件模型使得开发者可以快速构建用户界面,而无需从零开始编写所有视觉元素。 在"yingyan.swf"文件中,我们找到了编译后的Flash应用程序,它是ActionScript代码和Flex组件的最终运行形式。用户通过这个SWF...

    Flex课件模板皮肤样式

    这些开发者可以避免从零开始构建每一张幻灯片或每一个导航链接,而是通过修改和扩展模板来达到所需的功能和外观。此外,这些模板和组件是完全可定制的,因此开发者可以根据教学需求进行更深层次的调整。 除了提供...

    ArcGIS_API_for_Flex_教程

    以上内容是针对初学者的入门教程,目的是让学习者能够从零开始,逐步掌握使用ArcGIS API for Flex 构建简单GIS应用的能力。本教程旨在为初学者提供一套完整的学习路径,帮助他们在学习过程中逐渐深入GIS开发,并最终...

    flex 移动开发

    这一部分对于初学者来说至关重要,因为它提供了从零开始构建移动应用所需的基本指导。 **1.2 移动、桌面和浏览器应用程序开发的区别** 这一小节详细对比了不同平台上的应用开发过程。对于开发者而言,理解这些差异...

    Developing Mobile Applications with ADOBE FLEX and ADOBE FLASH BUILDER

    ### 开发移动应用使用ADOBE FLEX与ADOBE FLASH ...无论是从零开始创建新应用还是对现有项目进行升级,这些工具都能提供强大而灵活的支持。随着移动互联网的普及和技术的不断进步,掌握这些技能将为开发者带来更多机会。

    FlexViewer2.5

    FlexViewer作为ArcGIS for Flex的核心组件,提供了一套预定义的用户界面元素和控件,帮助开发者快速搭建地图应用而无需从零开始编写大量代码。 在FlexViewer2.5中,以下几个关键知识点值得深入探讨: 1. **地图...

    MATLAB GUI设计学习手记(第2版)_matlab_GUI_

    “MATLAB GUI设计学习手记(第2版).pdf”这份资料将系统地引导初学者从零开始掌握MATLAB GUI设计,通过理论结合实践,使读者能够独立开发出具有专业水准的图形用户界面。无论是进行科学计算、数据分析还是工程应用,...

Global site tag (gtag.js) - Google Analytics