замена экранного объекта новым экранным объектом во Flash - PullRequest
0 голосов
/ 15 декабря 2011

Я создаю Flash-фильм, который загружает XML-файл, содержащий сведения о тексте и URL-адрес изображения, который должен заменить существующий MovieClip во Flash-фильме.

Есть ли во Flash способ заменить существующий MovieClip или обновить его таким образом, чтобы обновленный / новый экранный объект сохранял исходный вид исходного экранного объекта, а любой анимационный ролик для исходного экранного объекта по-прежнему работает, мы новый или обновленный экранный объект.

Спасибо

Стивен

Обновление: вот код, который я использую для загрузки этого:


import flash.events.IOErrorEvent;
import flash.net.URLLoader;
import flash.net.URLRequest;
import flash.events.Event;
import flash.display.DisplayObject;
import flash.display.MovieClip;
import flash.display.Sprite;
import flash.display.Loader;
import flash.text.TextField;
import flash.events.ProgressEvent;


// get the number of children in display list
var count:uint = numChildren;

// get the url for the Ceres server
//var configUrl:String = loaderInfo.parameters.configUrl;

// load the config file from the server using the configUrl
var configUrl:String = "C:\development\projects\ReadXMLInFlash\example_xml_ceres_response.xml";   //loaderInfo.parameters.configFileUrl;
var defaultImgUrl:String = loaderInfo.parameters.defaultImgUrl;
var urlRequest:URLRequest = new URLRequest( configUrl );

// create the XML loader
var urlLoader:URLLoader = new URLLoader();
urlLoader.load( urlRequest );
urlLoader.addEventListener(Event.COMPLETE, onDataLoaded);
//Error handling    ;
urlLoader.addEventListener(IOErrorEvent.IO_ERROR, onIOError);
urlLoader.addEventListener(SecurityErrorEvent.SECURITY_ERROR, onSecurityError);
//Could be an error or just a message;
urlLoader.addEventListener(HTTPStatusEvent.HTTP_STATUS, onHTTPStatus);

function onDataLoaded(evt:Event):void
{
    var myXML:XML = new XML(evt.target.data);
    var node:XML;
    // loop through the XML
    for each (node in myXML.dynamicGroups.DynamicGroup.assets.Asset.elements.Element)
    {
        // check the assetValue of the XML, if contains http then a link use 'createNewImageLoader'
        if(String(node.assetTagName) == 'Image' )
        {
            createNewImageLoader( node.assetValue, node.elementName );
        }
        else
        {
            // else it is text (e.g. a product descrition) use createNewTextLoader
            createNewTextLoader( node.assetValue, node.elementName );
        }

    }

}

//error callbacks
function onIOError(evt:IOErrorEvent)
{
    trace("IOError: "+evt.text);
    // if an error load the default image
    var defaultImgLoader:Loader = new Loader();
    var defaultImgUrl:URLRequest = new URLRequest( defaultImgUrl );
    defaultImgLoader.load( defaultImgUrl );
    addChild( defaultImgLoader );

}
function onHTTPStatus(evt:HTTPStatusEvent)
{
    //trace("HTTPStatus: "+evt.status);
}
function onSecurityError(evt:SecurityErrorEvent)
{
    trace("SecurityError: "+evt.text);
}

// loads a image asset
function createNewImageLoader(loadURL:String, elementname:String):void
{
    for(var i:uint=0;i<numChildren;i++)
    {
        var display:DisplayObject = getChildAt(i);
        if (elementname == display.name)
        {
            var loader:Loader = new Loader();
            var urlRequest:URLRequest = new URLRequest(loadURL);
            loader.addEventListener(Event.COMPLETE, function(){
                                    loader.x = display.x;
                                    loader.y = display.y;
                                    addChild( loader );
                                    removeChild(display);
                                    });
            loader.load( urlRequest );

        }
    }
}
// adds new text to a display object
function createNewTextLoader( textToLoad:String, elementname:String):void
{
    trace( "elementname:" + elementname );

    for(var i:uint=0;i<count;i++)
    {
        var display:DisplayObject = getChildAt(i);

        trace( "displayName:" + display.name );
        if (elementname == display.name)
        {
            var textToDisplay:TextField = new TextField();
            textToDisplay.text = textToLoad;
            textToDisplay.x = display.x;
            textToDisplay.y = display.y;
            addChild(textToDisplay);
            removeChild(display);
            break;
        }
    }
}

Ответы [ 2 ]

0 голосов
/ 16 декабря 2011

Я не совсем понимаю, почему у вас уже есть MC на сцене, и вы должны обновить их после загрузки XML ... разве вы не можете просто создать их после загрузки XML?

в обоих случаях было бы лучше создать два собственных класса (подкласса MovieClip): TextLoader и ImageLoader с методом обновления.

package   
{
    import flash.display.MovieClip;

    public class TextLoader extends MovieClip 
    {
        private var _txt:TextField;

        public function TextLoader() 
        {
            _txt = new TextField();
            // do TextFormat stuff, set size and position of _txt
            addChild(_txt);

            // you could also set some initial text like:
            _txt.text = "loading ...";
            // this will be overwritten later
        }

        public function update(label:String):void 
        {
            _txt.text = label;
        }
    }
}

ImageLoader также выглядел бы так, но вместо TextField получил загрузчик, который обновляется с помощью метода update(url:String).

создайте MC перед загрузкой XML следующим образом:

var allTxts:Array = [];
for (var i:int=0;i<10;++i)
{
    var txt:TextLoader = new TextLoader();
    txt.x = 50;
    txt.y = (i+1) * 50;
    txt.name = "txt_" + i;  // the names you got in your xml
    addChild(txt);

    allTxts.push(txt);
}

при загрузке XML обновите все MC:

function getTxtWithName(name:String):TextLoader
{
    for (var i:int=0;i<allTxts.length;++i)
    {
        var txt:TextLoader = allTxts[i] as TextLoader;
        if (txt != null && txt.name == name)
        {
            return txt;
        }
    }
    return null;
}

function onDataLoaded(evt:Event):void
{
    var myXML:XML = new XML(evt.target.data);
    var node:XML;
    // loop through the XML
    for each (node in myXML.dynamicGroups.DynamicGroup.assets.Asset.elements.Element)
    {
        // check the assetValue of the XML, if contains http then a link use 'createNewImageLoader'
        if(String(node.assetTagName) == 'Image' )
        {
            var txt:TextLoader = getTxtWithName(node.elementName);
            if (txt != null)
            {
                txt.update(node.assetValue);
            }
        }
        else
        {
            // else it is text (e.g. a product descrition) use createNewTextLoader
            var img:ImageLoader = getImageWithName(node.elementName);
            if (img != null)
            {
                img.update(node.assetValue);
            }
        }
    }
}
0 голосов
/ 16 декабря 2011

Я не совсем уверен, в чем ваша проблема или чего вы пытаетесь достичь, но я постараюсь ответить на ваш вопрос в меру своих возможностей.

Из того, что я прочитал в комментариях и в вашем коде, вы пытаетесь изменить текст и изображение внутри фрагмента ролика. Для этого просто измените этот код:

function createNewTextLoader( textToLoad:String, elementname:String):void
{
    trace( "elementname:" + elementname );

    for(var i:uint=0;i<count;i++)
    {
        var display:DisplayObject = getChildAt(i);

        trace( "displayName:" + display.name );
        if (elementname == display.name)
        {
            var textToDisplay:TextField = new TextField();
            textToDisplay.text = textToLoad;
            textToDisplay.x = display.x;
            textToDisplay.y = display.y;
            addChild(textToDisplay);
            removeChild(display);
            break;
        }
    }
}

до:

function createNewTextLoader( textToLoad:String, elementname:String):void
{
// If the display object by the name of the value of "elementname" exists and it is a text field
    if (getChildByName(elementname) != undefined &&
        getChildByName(elementname) is TextField)
    {
        var textToDisplay:TextField = (TextField)getChildByName(elementname);// I dont know if you need a cast here but just in case
        textToDisplay.text = textToLoad;// change the text in the text field
    }
}

и замените это:

function createNewImageLoader(loadURL:String, elementname:String):void
{
    for(var i:uint=0;i<numChildren;i++)
    {
        var display:DisplayObject = getChildAt(i);
        if (elementname == display.name)
        {
            var loader:Loader = new Loader();
            var urlRequest:URLRequest = new URLRequest(loadURL);
            loader.addEventListener(Event.COMPLETE, function(){
                                    loader.x = display.x;
                                    loader.y = display.y;
                                    addChild( loader );
                                    removeChild(display);
                                    });
            loader.load( urlRequest );
        }
    }
}

с этим:

function createNewImageLoader(loadURL:String, elementname:String):void
{
    if (getChildByName(elementname) != undefined &&
        getChildByName(elementname) is Loader)
    {
        var loader:Loader = (Loader)(getChildByName(elementname);
        var urlRequest:URLRequest = new URLRequest(loadURL);
        loader.load( urlRequest );
    }
}

На самом деле я еще не тестировал этот код, но сложно сказать, отвечаю ли я на правильный вопрос.

...