开发者

how do I make non-document-class classes 'aware' of stage components in Flash AS3?

I am working on a text adventure game which will have at least a few components (a text area for narrative and text input for user input) on the stage at all times. Therefore, I have created those components statically through Flash's WYSIWYG design environment. I gave them instance names "myTA" and "myTI" respectively. I was able to get my main class (the document class for the stage) to interact with them (dynamically adding text one character at a time like a typewriter at runtime), but other classes in the same package don开发者_StackOverflow社区't seem able to recognize the stage components. Below is the relevant code:

Case A, in which everything happens within the Main class:

package { 
    public class Main extends MovieClip { 
      public var myTA:TextArea; 
      var displayedChar:String = new String(); 
      var textToWrite:String = new String(); 
      var i:int = 0; var intervalId:uint; 
      var done:int = 0; 

      public function Main { 
       setUpTA(); 
      } 

      public function setUpTA(){ 
       myTA.text = "" + playAtInterval("Hello Player!"); 
      }         

      public function writeCharsSlowly(){     
        textToWrite = arguments[0];     
        displayedChar=textToWrite.substring(i,i+1); 
        myTA.appendText(displayedChar);                             
        i++;     
        if (i == textToWrite.length) {         
          done = 1;         
          clearInterval(intervalId);     
        }             
      }                 

      public function playAtInterval(theText:String) {                
        i = 0;              
        intervalId = setInterval(writeCharsSlowly, 100, theText);           
      }  
    } 
}

Case B, where Main calls on a second class 'TypeWriter' to handle the typewriter-printing:

Main:

package { 
    public class Main extends MovieClip { 
      public var myTA:TextArea; 
      public var myTI:TextInput; 
      var str:String = new String(); 

      public function Main{ 
        testTypeWriter(); 
      } 

      public function testTypeWriter(){ 
        typeW.playAtInterval("Hello Player");
        typeW.addEventListener(MouseEvent.CLICK,testTypeWriter2);
        typeW.addEventListener(KeyboardEvent.KEY_DOWN,inputEngine2) 
        addChild(typeW); 
      } 

      public function testTypeWriter2(event:MouseEvent){ 
        if (myTI.text == "a") {   
          typeW.playAtInterval("yo");  
        } else {   
          typeW.playAtInterval("Greetings, I am a test...");  
        }             
        addChild(typeW); 
      } 

      public function inputEngine2(event:KeyboardEvent){ 
        str = String.fromCharCode(event.charCode); 
        myTI.appendText(str); 
      }

TypeWriter:

package { 
    public class TypeWriter extends MovieClip { 
      public var myTI:TextInput; 
      public var myTA:TextArea; 
      var i:int = 0; 
      var done:int = 0; 
      var intervalId:uint; 
      var displayedChar:String = new String(); 
      var textToWrite:String = new String(); 

      public function TypeWriter(){ 
        ///nothing here 
      } 

      public function writeCharsSlowly(){     
        textToWrite = arguments[0];     
        displayedChar = textToWrite.substring(i,i+1);
        myTA.appendText(displayedChar);                             
        i++;     
        if (i == textToWrite.length) {         
          done = 1;         
          clearInterval(intervalId);     
        }             
      }                 

      public function playAtInterval(theText:String) {                
        i = 0;              
        intervalId = setInterval(writeCharsSlowly, 100, theText);           
      }  
    } 
}

Case A works, but in case B Flash is giving me the error "Error #1009: Cannot access a property or method of a null object reference" and notes the first line in TypeWriter where I try to operate on myTA as the problem.

how can I make other classes besides the document class 'aware' of existing stage components?

Thanks,

CCJ


I would recommend the Service Locator Pattern for this. The most naive approach would be to create a resource class which contains public static variables. Then in your document class you assign the stage instances to the corresponding static variable in the resource class. Then you can simply access these stage components anywhere.

var someTextArea = Resource.TA; //probably should rename to something more meaningful

For something a little more ingenious you should read the article I linked to.

I think this is better than the dependency injection as constructor injection could lead to huge parameter list as you might add more items to the stage, and I am not so fond on setter injection as it is easy to forget to set them.

EDIT:

Just to make it a bit more clear I thought I would add some code :)

Resource class

package
{
    //TODO imports
    public class Resource
    {
        public static var TA:TextArea;
        public static var TI:TextInput;
    }
}

Document class

//....setup function
Resource.TA = myTA; //myTA is the name of the instance on stage
Resource.TI = myTI;

Foo class

Resource.TA.x = 100;
//or
_myClassMemberVariable = Resource.TA;
_myClassMemberVariable.x = 100;


I think some dependency injection will solve this problem. When you instantiate your Typewriter class, pass references to myTA and myTI. ex:

public function Main{ 
    testTypeWriter(this.myTA, this.myTI); 
} 

Then your Typewriter constructor should look like this:

public function TypeWriter(ta:TextArea, ti:TextArea){ 
    this.myTA = ta;
    this.myTI = ti;
  } 

This also has the benefit of making your application less tightly coupled, so for example you can reuse your Typewriter class with a different text area and text input.

Edit

Some extra info that may help you in the future: you can access stage elements through the root object. But this only works with objects that have been added to the display list. Let's say that Typewriter represents an object in the display list, you could then access myTA like this:

MovieClip(root).myTA

(Change MovieClip to Sprite if that's what your document class extends).

However, since it seems that Typewriter does not get added to the display list, I recommend using my first suggestion of dependancy injection.

Also check out this page, it dicusses using CasaLib to access the stage from any object. I personally haven't tried it, so that's why it's at the end here ;-)


Are myTA and myTI actually on the stage at author time or are the added dynamically?

In the first case, just add an instance name to each. Then add a variable to each class

var main_mc : Main = root as Main;

You can then access the instances via main_mc.myTA and main_mc.myTI (assuming those are the instances names you chose) and everything should be type-safe.

In the dynamic case, just make sure you save off references in the main class to each as you add them.

Another option is to have classes for myTA and myTI send an event in their constructor to announce their existence. The main class can then listen for these and register the references.

To be honest, though, you are mixing up your display with your logic. Read up on MVC and PureMVC in particular. With a good design, everything can be handled by messages, and the instances don't need direct references to each other. IIRC, Moock's AS3 book has a chapter on MVC (but it could be his AS2 book).


The fact that they will be on stage at all times shouldn't stop you from creating specific classes for them.

Depending on your game structure, you could either create a MovieClip with your TextFields and link them to the Typewriter class, or simply create a class for the TextFields and use Events to modify the text content.

You're using external classes so there are no reason to keep any kind of logic inside Flash CS.


Just to be clear. When you are writing stage you really mean the document class for your flash document. Stage is a class that every instance that has been added to the displaylist Have access to.

I would pass the textfields into the classes that needs to update them.

var tw : Typewriter = new Typewriter();
    tw.inputField = myTI;
    tw.textArea = myTA;

Or

var tw : Typewriter = new Typewriter(myTI, myTA);
0

上一篇:

下一篇:

精彩评论

暂无评论...
验证码 换一张
取 消

最新问答

问答排行榜