Проблема внедрения зависимости Петрушки в Flex 4 с использованием шаблона Presentation Model - PullRequest
1 голос
/ 08 сентября 2011

У меня есть класс View EmployeeList следующим образом:

<?xml version="1.0" encoding="utf-8"?>
<s:NavigatorContent xmlns:fx="http://ns.adobe.com/mxml/2009" 
                    xmlns:s="library://ns.adobe.com/flex/spark"
                    xmlns:parsley="http://www.spicefactory.org/parsley"
                    xmlns:mx="library://ns.adobe.com/flex/mx" width="400" height="300">


    <fx:Script>
        <![CDATA[
            import cafeparsley.model.EmployeeListPM;

            [Inject]
            [Bindable]
            public var model : EmployeeListPM;

            [Init]
            public function init () : void {
                model.init();
            }


        ]]>
    </fx:Script>    

    <s:Panel title="Employee List" horizontalCenter="0">
        <s:HGroup paddingTop="50">
            <s:Button label="Add New Employee" click="model.addNewEmployee()" />
            <mx:Spacer width="100%" />
            <s:Button label="Logout" click="model.logout()" />
            <mx:Spacer width="100%" height="20" />  
        </s:HGroup>
        <s:List id="empList" dataProvider="{ model.employees }" labelFunction="model.properName" 
                 change="model.initUpdateEmployee(empList.selectedItem);empList.selectedIndex = -1;" width="100%" />
        <s:Label id="error" color="0xFF0000" />
    </s:Panel>  

</s:NavigatorContent>

PM выглядит следующим образом:

package cafeparsley.model
{
    import cafeparsley.events.EmployeeEvent;
    import cafeparsley.events.NavigationEvent;
    import cafeparsley.services.impl.EmployeeServiceImpl;
    import cafeparsley.vo.Employee;

    import flash.events.EventDispatcher;

    import mx.collections.ArrayCollection;
    import mx.rpc.IResponder;

    [Bindable]
    [Event(name="navigationEvent", type="cafeparsley.events.NavigationEvent")]
    [ManagedEvents("navigationEvent")]
    public class EmployeeListPM extends EventDispatcher implements IResponder
    {

        public var employeeService : EmployeeServiceImpl = new EmployeeServiceImpl();

        public var employees : ArrayCollection;

        public function init() : void
        {
            loadEmployees();
        }


        public function EmployeeListPM() 
        {
        }

        public function loadEmployees():void
        {
            employeeService.loadEmployees( this );
        }  

Независимо от того, использую ли я или автоматическую разводку для выполнения впрыска, когда я запускаюэто я получаю следующее сообщение об ошибке:

TypeError: Ошибка # 1009: Не удается получить доступ к свойству или методу ссылки на пустой объект.в cafeparsley.view :: EmployeeList / _EmployeeList_List1_i () [C: \ dev \ code \ workspace \ examples \ CafeParsley \ src \ cafeparsley \ view \ EmployeeList.mxml: 29] в cafeparsley.view :: EmployeeList / _EmployeeList_Array2xc (at).core :: DeferredInstanceFromFunction / getInstance () [E: \ dev \ 4.x \ frameworks \ projects \ framework \ src \ mx \ core \ DeferredInstanceFromFunction.as: 105] в spark.components :: SkinnableContainer / createDeferredContent () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ SkinnableContainer.as: 985] на spark.components :: SkinnableContainer / createContentIfNeeded () [E: \ dev \ 4.x \ frameworks \ projects\ spark \ src \ spark \ components \ SkinnableContainer.as: 1014] по адресу spark.components :: SkinnableContainer / createChildren () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ SkinnableContainer.как: 827] в mx.core :: UIComponent / initialize () [E: \ dev \ 4.x \ frameworks \ projects \ framework \ src \ mx \ core \ UIComponent.as: 7349] в mx.core :: UIComponent/ http://www.adobe.com/2006/flex/mx/internal::childAdded()[E:\dev\4.x\frameworks\projects\framework\src\mx\core\UIComponent.as:7241] в mx.core :: UIComponent / addChildAt () [E: \ dev \ 4.x \ frameworks \ projects \ framework \ src \ mx \ core \ UIComponent.as: 6947] по адресу spark.components :: Group / addDisplayObjectToDisplayList () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ Group.as: 1825] at spark.components :: Group / http://www.adobe.com/2006/flex/mx/internal::elementAdded()[E:\dev\4.x\frameworks\projects\spark\src\spark\components\Group.as:1416] at spark.components :: Group / setMXMLContent () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ Group.as: 512] в spark.components :: Group / set mxmlContent () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ Group.as: 452] в spark.components :: SkinnableContainer/ set mxmlContent () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ SkinnableContainer.as: 604] на spark.components :: SkinnableContainer / createDeferredContent () [E: \ dev \4.x \ frameworks \ projects \ spark \ src \ spark \ components \ SkinnableContainer.as: 986] на spark.components :: SkinnableContainer / createContentIfNeeded () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src\ spark \ components \ SkinnableContainer.as: 1014] на сайте spark.components :: SkinnableContainer / createChildren () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ SkinnableContainer.as:827] по адресу spark.components :: NavigatorContent / createChildren () [E: \ dev \ 4.x \ frameworks \ projects \ spark \ src \ spark \ components \ NavigatorContent.as: 225] по адресу mx.core:: UIComponent / initialize () [E: \ dev \ 4.x \ frameworks \ projects \ framework \ src \ mx \ core \ UIComponent.as: 7349] в cafeparsley.view :: EmployeeList / initialize ()

Таким образом, employeeListPM имеет значение null при возникновении ошибки.Однако, если я закомментирую компонент <s:List>, перезапущу и установлю точку останова в методе init, будет вызвана init ().Так что дело не в том, что моя контекстная конфигурация неверна, просто в том, что ПМ не был введен вовремя и выдается ошибка.Но согласно руководству Parsley, если я использую автопроводку или <parsley:configure/>, PM должен быть введен в то время, когда это необходимо.

Я не вижу, что я делаю неправильно, в том, что я считал относительно простым сценарием внедрения зависимости.Вы можете помочь?

Ответы [ 2 ]

1 голос
/ 09 сентября 2011

Пара моментов здесь:

Ваш пример не включает теги <Configure /> или <FastInject />, однако, так как вы упомянули их в своем посте, я предполагаю, что они просто отсутствуют впример кода.(Если нет, то вам нужно добавить один из них, чтобы это работало).

Однако, более вероятно, у вас есть условия гонки в вашем коде.

В частности, эти строки:

labelFunction="model.properName" 
change="model.initUpdateEmployee(empList.selectedItem);empList.selectedIndex = -1;"

Model является внедренным свойством, однако не гарантируется, что оно было внедрено во время первого запуска кода.

Вместо этого переместите код в скрипт в классе, который выполняет проверку на ноль, а затем отложите логику до PM.

, то есть:

labelFunction="nameFunction"


private function nameFunction(item:Object):String
{
    return (model) ? model.properName(item) : "";
}
0 голосов
/ 14 сентября 2011

Еще одна вещь, которую стоит учесть, это то, что вы вызываете model.init () из функции View, помеченной метатегом Parsley [Init].Я бы посоветовал применить тот же метатег [Init] к методу init () модели.

[Init]    
public function loadEmployees():void {
     employeeService.loadEmployees( this );
}

Делайте это вместо вызова model.init () из представления.В документации по жизненному циклу говорится:

Методы, помеченные [Init], вызываются после создания экземпляра объекта и обработки всех инъекций.

У меня былогораздо более согласованные результаты, не вызывающие какие-либо методы init () между внедренными объектами напрямую, а скорее использующие подход метаданных.

...