ExternalInterface.call的问题
请留意以下情景,要利用外部应用程序弹出文件选择窗体,并返回所选的文件路径。
使用ExternalInterface.call来实现,当用户60秒(Flex已经设置最长的等待时间)内未能够完成文件选择会弹出"1502"的错误,严重影响用户体现。
所以,一些比较耗时的操作(超过60s),或无法确定何时调用完成的方法(如上述),应该使用异步调用的方式,当方法执行完成后进行回调。
<![CDATA[
实现异步调用机制
Flex端
定义一个Dictionary,用来缓存回调的方法
private var m_AsyncDict:Dictionary;
定义OpenFileDialogAsync方法,用fscommand来实现,fscommand是单向的调用,不存在60秒超时的问题。
//弹出打开文件窗口,选择单个文件
public function OpenFileDialogAsync(title:String = "请选择1个文件", initFolder:String = "", filter:String = "", callbackString:Function = null):void
{
//创建异步ID
var id:String = mx.utils.UIDUtil.createUID();
//创建自定义异步请求
var request:String = "<args>";
request += "<arg>" + id + "</arg>";
request += "<arg>" + title + "</arg>";
request += "<arg>" + initFolder + "</arg>";
request += "<arg>" + filter + "</arg>";
request += "</args>";
//缓存回调方法
if (callbackString != null)
m_AsyncDict[id] = callbackString;
flash.system.fscommand(FLASH_TO_APP_OPEN_FILE_DIALOG_ASYNC, request);
}
定义并注册AsyncCallback和CancelAsyncCall方法,让外部应用程序能够响应异步回调或取消异步回调。
public function AsyncCallback(id:String, value:Object):void
{
if (m_AsyncDict[id])
{
//执行回调方法
m_AsyncDict[id](value);
delete m_AsyncDict[id];
}
}
public function CancelAsyncCall(id:String):void
{
if (m_AsyncDict[id])
{
delete m_AsyncDict[id];
}
}
…
if (ExternalInterface.available)
{
ExternalInterface.addCallback(APP_TO_FLASH_ASYNC_CALLBACK, AsyncCallback);
ExternalInterface.addCallback(APP_TO_FLASH_CANCEL_ASYNC_CALL, CancelAsyncCall);
}
EXE端
为ShockwaveFlashObject注册FSCommand事件,并添加事件处理的方法。
axShockwaveFlash1.FlashCall += new AxShockwaveFlashObjects._IShockwaveFlashEvents_FlashCallEventHandler(axShockwaveFlash1_FlashCall);
…
void axShockwaveFlash1_FSCommand(object sender, AxShockwaveFlashObjects._IShockwaveFlashEvents_FSCommandEvent e)
{
//请求e.args是自定义Xml格式
XmlDocument request = new XmlDocument();
request.LoadXml(e.args);
if (e.command == FLASH_TO_APP_OPEN_FILE_DIALOG_ASYNC)
{
//获取回调的键值
string asyncId = request.DocumentElement.ChildNodes[0].InnerText;
//获取窗体参数
string title = request.DocumentElement.ChildNodes[1].InnerText;
string initFolder = request.DocumentElement.ChildNodes[2].InnerText;
string filter = request.DocumentElement.ChildNodes[3].InnerText;
using (OpenFileDialog dlg = new OpenFileDialog())
{
dlg.Title = title;
dlg.InitialDirectory = initFolder;
dlg.Filter = filter;
if (dlg.ShowDialog() == DialogResult.OK)
{
AsyncCallback(asyncId, dlg.FileName);
}
else
{
CancelAsyncCall(asyncId);
}
}
}
}
定义2个方法调用Flex里注册的AsyncCallback和CancelAsyncCall方法。
//异步回调
public void AsyncCallback(string asyncId, string value)
{
StringBuilder sb = new StringBuilder();
sb.Append("<invoke name=/"").Append(APP_TO_FLASH_ASYNC_CALLBACK).Append("/" returntype=/"xml/">");
sb.Append("<arguments>");
sb.Append("<string>").Append(asyncId).Append("</string>");
sb.Append("<string>").Append(value).Append("</string>");
sb.Append("</arguments>");
sb.Append("</invoke>");
axShockwaveFlash1.CallFunction(sb.ToString());
}
//撤销异步回调
private void CancelAsyncCall(string asyncId)
{
StringBuilder sb = new StringBuilder();
sb.Append("<invoke name=/"").Append(APP_TO_FLASH_CANCEL_ASYNC_CALL).Append("/" returntype=/"xml/">");
sb.Append("<arguments>");
sb.Append("<string>").Append(asyncId).Append("</string>");
sb.Append("</arguments>");
sb.Append("</invoke>");
axShockwaveFlash1.CallFunction(sb.ToString());
}
完整代码
Flex端
LocalAPI.as
package
{
import flash.external.ExternalInterface;
import flash.system.fscommand;
import flash.utils.Dictionary;
import mx.utils.UIDUtil;
public class LocalAPI
{
//通讯命令定义
private static const FLASH_TO_APP_OPEN_FILE_DIALOG:String = "OpenFileDialog";
private static const FLASH_TO_APP_OPEN_FILE_DIALOG_ASYNC:String = "OpenFileDialogAsync";
private static const APP_TO_FLASH_ASYNC_CALLBACK:String = "AsyncCallback";
private static const APP_TO_FLASH_CANCEL_ASYNC_CALL:String = "CancelAsyncCall";
//Async dict
private var m_AsyncDict:Dictionary;
//Singleton static obj
private static var g_Instance:LocalAPI = null;
//获取LocalAPI单件实例
public static function get Instance():LocalAPI
{
if (g_Instance == null)
{
g_Instance = new LocalAPI();
}
return g_Instance;
}
public function LocalAPI()
{
if (g_Instance != null)
throw new Error("Singleton class. Please use Instance static filed.");
m_AsyncDict = new Dictionary();
if (ExternalInterface.available)
{
ExternalInterface.addCallback(APP_TO_FLASH_ASYNC_CALLBACK, AsyncCallback);
ExternalInterface.addCallback(APP_TO_FLASH_CANCEL_ASYNC_CALL, CancelAsyncCall);
}
}
//弹出打开文件窗口,选择单个文件(60s局限)
public function OpenFileDialog(title:String = "请选择1个文件", initFolder:String = "", filter:String = ""):String
{
if(!ExternalInterface.available) return null;
return ExternalInterface.call(FLASH_TO_APP_OPEN_FILE_DIALOG, title, initFolder, filter);
}
//异步通信
//弹出打开文件窗口,选择单个文件
public function OpenFileDialogAsync(title:String = "请选择1个文件", initFolder:String = "", filter:String = "", callbackString:Function = null):void
{
//创建异步ID
var id:String = mx.utils.UIDUtil.createUID();
//创建自定义异步请求
var request:String = "<args>";
request += "<arg>" + id + "</arg>";
request += "<arg>" + title + "</arg>";
request += "<arg>" + initFolder + "</arg>";
request += "<arg>" + filter + "</arg>";
request += "</args>";
//缓存回调方法
if (callbackString != null)
m_AsyncDict[id] = callbackString;
flash.system.fscommand(FLASH_TO_APP_OPEN_FILE_DIALOG_ASYNC, request);
}
public function AsyncCallback(id:String, value:Object):void
{
if (m_AsyncDict[id])
{
//执行回调方法
m_AsyncDict[id](value);
delete m_AsyncDict[id];
}
}
public function CancelAsyncCall(id:String):void
{
if (m_AsyncDict[id])
{
delete m_AsyncDict[id];
}
}
}
}
Demo.mxml
<?xml version="1.0" encoding="utf-8"?>
<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml" layout="absolute" fontSize="12">
<mx:Script>
private function GetFilePath():void
{
var path:String = LocalAPI.Instance.OpenFileDialog();
if (path != null)
_Path1.text = path;
}
//异步方式
private function GetFilePathAsync():void
{
LocalAPI.Instance.OpenFileDialogAsync("请选择1个文件", "", "", function(result:String):void
{
_Path2.text = result;
});
}
</mx:Script>
<mx:Button x="445.75" y="100" label="浏览1" click="GetFilePath()"/>
<mx:Button x="445.75" y="147" label="浏览2" click="GetFilePathAsync()"/>
<mx:Label x="71" y="102" text="文件路径"/>
<mx:TextInput x="188.25" y="100" width="249.5" id="_Path1"/>
<mx:TextInput x="188.25" y="147" width="249.5" id="_Path2"/>
<mx:Label x="71" y="149" text="文件路径(异步方式)"/>
<mx:Text x="71" y="196" width="459.75" height="213" id="_Trace"/>
</mx:Application>
EXE端
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Xml;
namespace WinFormDemo
{
public partial class Form1 : Form
{
//定义通信命令
private string FLASH_TO_APP_OPEN_FILE_DIALOG = "OpenFileDialog";
private string FLASH_TO_APP_OPEN_FILE_DIALOG_ASYNC = "OpenFileDialogAsync";
private string APP_TO_FLASH_ASYNC_CALLBACK = "AsyncCallback";
private string APP_TO_FLASH_CANCEL_ASYNC_CALL = "CancelAsyncCall";
private bool m_Init = false;
public Form1()
{
InitializeComponent();
}
private void Form1_Load(object sender, EventArgs e)
{
if (m_Init) return;
string swf = System.IO.Path.Combine(Application.StartupPath, "Demo.swf");
axShockwaveFlash1.FSCommand += new AxShockwaveFlashObjects._IShockwaveFlashEvents_FSCommandEventHandler(axShockwaveFlash1_FSCommand);
axShockwaveFlash1.FlashCall += new AxShockwaveFlashObjects._IShockwaveFlashEvents_FlashCallEventHandler(axShockwaveFlash1_FlashCall);
axShockwaveFlash1.Movie = swf;
m_Init = true;
}
void axShockwaveFlash1_FlashCall(object sender, AxShockwaveFlashObjects._IShockwaveFlashEvents_FlashCallEvent e)
{
//请求e.request是Xml格式,详细请查阅文档
XmlDocument request = new XmlDocument();
request.LoadXml(e.request);
string command = request.DocumentElement.Attributes["name"].Value;
if (command == FLASH_TO_APP_OPEN_FILE_DIALOG)
{
//获取窗体参数
string title = request.DocumentElement.ChildNodes[0].ChildNodes[0].InnerText;
string initFolder = request.DocumentElement.ChildNodes[0].ChildNodes[1].InnerText;
string filter = request.DocumentElement.ChildNodes[0].ChildNodes[2].InnerText;
using (OpenFileDialog dlg = new OpenFileDialog())
{
dlg.Title = title;
dlg.InitialDirectory = initFolder;
dlg.Filter = filter;
if (dlg.ShowDialog() == DialogResult.OK)
{
axShockwaveFlash1.SetReturnValue("<string>" + dlg.FileName + "</string>");
}
else
{
axShockwaveFlash1.SetReturnValue("<null/>");
}
}
}
}
void axShockwaveFlash1_FSCommand(object sender, AxShockwaveFlashObjects._IShockwaveFlashEvents_FSCommandEvent e)
{
//请求e.args是自定义Xml格式
XmlDocument request = new XmlDocument();
request.LoadXml(e.args);
if (e.command == FLASH_TO_APP_OPEN_FILE_DIALOG_ASYNC)
{
//获取回调的键值
string asyncId = request.DocumentElement.ChildNodes[0].InnerText;
//获取窗体参数
string title = request.DocumentElement.ChildNodes[1].InnerText;
string initFolder = request.DocumentElement.ChildNodes[2].InnerText;
string filter = request.DocumentElement.ChildNodes[3].InnerText;
using (OpenFileDialog dlg = new OpenFileDialog())
{
dlg.Title = title;
dlg.InitialDirectory = initFolder;
dlg.Filter = filter;
if (dlg.ShowDialog() == DialogResult.OK)
{
AsyncCallback(asyncId, dlg.FileName);
}
else
{
CancelAsyncCall(asyncId);
}
}
}
}
//异步回调
public void AsyncCallback(string asyncId, string value)
{
StringBuilder sb = new StringBuilder();
sb.Append("<invoke name=/"").Append(APP_TO_FLASH_ASYNC_CALLBACK).Append("/" returntype=/"xml/">");
sb.Append("<arguments>");
sb.Append("<string>").Append(asyncId).Append("</string>");
sb.Append("<string>").Append(value).Append("</string>");
sb.Append("</arguments>");
sb.Append("</invoke>");
axShockwaveFlash1.CallFunction(sb.ToString());
}
//撤销异步回调
private void CancelAsyncCall(string asyncId)
{
StringBuilder sb = new StringBuilder();
sb.Append("<invoke name=/"").Append(APP_TO_FLASH_CANCEL_ASYNC_CALL).Append("/" returntype=/"xml/">");
sb.Append("<arguments>");
sb.Append("<string>").Append(asyncId).Append("</string>");
sb.Append("</arguments>");
sb.Append("</invoke>");
axShockwaveFlash1.CallFunction(sb.ToString());
}
}
}
系列索引
Flex通信篇——Flex和外部应用程序进行通信
Flex通信篇——Flex和外部进行异步通信
Flex通信篇——Flex键盘组合键
Flex通信篇——构建企业级HTTP通信层
]]>
分享到:
相关推荐
本文将围绕“FLEX入门篇——日期的格式化”这一主题,深入探讨如何在Flex中处理和格式化日期。日期格式化在任何应用程序中都是常见的需求,尤其是在展示或存储用户友好的时间信息时。 首先,我们要了解Flex中处理...
总之,通过正确的ActionScript编程和项目配置,我们可以将外部SWF资源作为具体对象在Flex应用中使用,从而极大地扩展了我们的功能和灵活性。这在处理第三方组件或者复用已有代码库时尤其有用。为了获取更多具体的...
FlexViewer 使用 ArcGIS Server 提供的地图服务,通过 SOAP 或 REST API 与服务器进行通信。学习如何在 Flex 中添加和使用地图服务是掌握 ArcGIS Flex 的关键步骤。 6. **调试与开发** 利用 Flex Builder 或其他 ...
创建第一个Flex应用——编码模式创建第一个Flex应用——编码模式创建第一个Flex应用——编码模式
创建第一个Flex应用——设计模式创建第一个Flex应用——设计模式创建第一个Flex应用——设计模式
这些类将通过BlazeDS与Flex前端进行通信。 #### 四、使用BlazeDS进行Flex与Java通信 完成上述步骤后,就可以开始使用BlazeDS进行Flex与Java之间的通信了。具体的通信流程包括: 1. **定义服务接口**:在Java中...
《精通Flex 3.0——基于ActionScript 3.0实现》一书源代码。 Flex 3.0 ActionScript 3.0源代码 Flex 3.0源代码。 --------------------------- 第1篇 Flex技术概述 第1章 Flex概述 3 1.1 Flex简介 3 1.2 Flex...
携程在线首页。适合零基础入门的人使用的demo,对布局这块有不错的...采用的是flex布局使用了原生js实现的tab栏切换等。自己可以通过swiper插件或者TouchSlide插件以及bootstrap框架来替换!可以把这个demo进行更改。
Flex通信-Java服务端通信实例主要探讨的是在Web开发中,如何使用Adobe Flex与Java后端进行交互。Flex是一款强大的富互联网应用程序(RIA)开发工具,它可以创建动态、交互式的用户界面,而Java则通常作为服务器端的...
数据绑定是Flex中一个核心的概念,它简化了用户界面与应用程序数据之间的通信,使得数据的更新能够自动反映在UI上,反之亦然。 在Flex中,数据绑定是一种声明式编程方式,开发者无需编写复杂的事件处理代码,即可...
5. **事件驱动编程**:Flex和Java之间的通信通常是异步的,基于事件的。当服务器响应到达时,会触发特定的事件,客户端可以通过监听这些事件来处理结果。 6. **数据管理**:在Java后端,可能会使用Spring、...
flex界面布局和导航,flex各个控件的使用,布局的容器和布局的空间,详细的代码演示
在Flex和Java之间进行通信时,通常会用到两种主要技术:AMF(Action Message Format)和SOAP(Simple Object Access Protocol)。AMF是一种二进制协议,用于提高数据传输效率,而SOAP则是基于XML的标准协议,更适用...
Flex游戏——水果乐园(含源码) 游戏介绍: * 通过键盘方向键控制小熊(游戏主人翁)在迷宫内行走,在指定时间内拾取所有水果就通过游戏否不通过。 * 每次拾取水果必须正确回答问题才能够真正获取水果。 * 游戏中有...
根据给定的信息,我们可以深入探讨Flex中的异步加载Tree组件的相关知识点。 ### 一、Flex与AsyncTree组件 #### 1. Flex简介 Flex是一种开源的框架,用于构建跨平台的桌面应用程序和移动应用程序。它提供了强大的...
这篇资料提供了一套AS(ActionScript)源码,用于实现Flex客户端与服务端的交互,支持同步和异步两种通信方式,且采用HTTPService进行封装,避免了对BlazeDS的依赖。主要知识点包括: 1. **Flex与服务端通信**:...