【AICon】AI 基础设施、LLM运维、大模型训练与推理,一场会议,全方位涵盖! >>> 了解详情
写点什么

从 CameraRoll 和 CameraUI 上传图像

  • 2011-09-02
  • 本文字数:3861 字

    阅读完需:约 13 分钟

目录

需求

预备知识

使用 Flash Builder、Flash Professional 或建议的其他工具构建 AIR 应用程序的一般经验。

用户水平

中级

需要的产品

示例文件

本文探讨如何通过 CameraRoll 和 CameraUI 类将获得的图像上传到服务器。

CameraRoll 类可用于打开一个图像浏览器,以便用户可从设备媒体库选择照片。CameraUI 类可用于启动设备照相机应用程序,以便用户可拍摄照片或录制视频。在两种情况下,CameraRoll 和 CameraUI 对象都通过一个包含媒体承诺的事件来返回媒体。

可用于访问媒体承诺中的数据的选项在各个平台中不尽相同。例如,在 Android 上,媒体承诺包括一个引用源图像文件 file 属性,但在 iOS 上,此属性始终为空——源文件不可访问。本文中描述的技术适用于所有平台。

获取媒体承诺

第一步是从CameraRoll 或CameraUI 对象请求图像。在两种情况下,您都会创建合适的对象,设置一些事件监听器,然后调用请求图像的函数。然后,在用户从设备媒体库选择一张照片或使用照相机拍摄一张新照片之后,运行时在一个包含媒体承诺的事件中返回图像。

CameraRoll

下面的代码示例设置必要的事件监听器来处理可由 CameraRoll 对象分配的事件,然后调用 browseForImage() 方法。当用户选择一张图像时,运行时调用一个名为 imageSelected 的事件处理函数。

复制代码
//declare cameraRoll where it won't go out of scope
var cameraRoll:CameraRoll = new CameraRoll();
if( CameraRoll.supportsBrowseForImage )
{
cameraRoll.addEventListener( MediaEvent.SELECT, imageSelected );
cameraRoll.addEventListener( Event.CANCEL, browseCanceled );
cameraRoll.addEventListener( ErrorEvent.ERROR, mediaError );
cameraRoll.browseForImage();
}
else
{
trace( "Image browsing is not supported on this device.");
}

CameraUI

从 CameraUI 对象请求图像的代码非常类似:

复制代码
private var cameraUI:CameraUI = new CameraUI();
if( CameraUI.isSupported )
{
trace( "Initializing..." );
cameraUI.addEventListener( MediaEvent.COMPLETE, imageSelected );
cameraUI.addEventListener( Event.CANCEL, browseCanceled );
cameraUI.addEventListener( ErrorEvent.ERROR, mediaError );
cameraUI.launch( MediaType.IMAGE );
}
else
{
trace( "CameraUI is not supported.");
}

要注意的一点区别是,CameraRoll 分派一个类型为 MediaEvent.SELECT 的事件,而 CameraUI 类分派一个类型为 MediaEvent.COMPLETE 的事件。

获得原始图像数据

您可以从媒体承诺以字节数组的形式获取原始图像数据。但媒体承诺对象可以以一些不同方式提供自己的数据。所以,强健的跨平台解决方案的创建需要费一定的工夫。在 Android 上,CameraRoll 和 CameraUI 分派的媒体承诺包括一个引用图像文件的 File 对象。但是,File 对象在 iOS 上不可用。如果您的代码直接打开文件,它将仅适用于 Android。

如果您只希望显示图像,可以始终使用 Loader.loadMediaPromise() 方法。此方法加载图像并将它解码为一个显示对象。但要上传图像,您需要一个包含图像数据的 File 对象或 ByteArray 对象,而不是 Flash 显示对象。您可以使用 MediaPromise 对象的 file 属性(如果可用),但当 file 属性不可用时,可以通过读取媒体承诺数据源来访问图像数据。

媒体承诺对象提供了对可能同步或异步的数据源的访问。在同步情形下,您可以打开数据源并立即读取数据。但在异步情形下,在数据可用之前必须等待事件进展或完成事件。

下面的代码示例检查媒体承诺数据是同步的还是异步的(使用 isAsync 属性)。在异步情形下,该示例添加一个事件监听器。因为您不知道实际数据源的对象类型是什么,所以必须使用“as”运算符将它转换为合适的接口。将对象转换为 IdataInput 接口,以从数据源读取数据。将他转换为 IeventDispatcher 接口,可以访问异步数据源的 addEventListener() 方法。

复制代码
private var dataSource:IDataInput;
private function imageSelected( event:MediaEvent ):void
{
trace( "Media selected..." );
var imagePromise:MediaPromise = event.data;
dataSource = imagePromise.open();
if( imagePromise.isAsync )
{
trace( "Asynchronous media promise." );
var eventSource:IEventDispatcher = dataSource as IEventDispatcher;
eventSource.addEventListener( Event.COMPLETE, onMediaLoaded );
}
else
{
trace( "Synchronous media promise." );
readMediaData();
}
}
private function onMediaLoaded( event:Event ):void
{
trace("Media load complete");
readMediaData();
}
private function readMediaData():void
{
//do something with the data
}

上传图像

上传图像的最简单方式是使用FileReference.upload()。是的,我知道我刚才告诉您不要依靠文件的存在与否。但是,如果没有从媒体承诺直接获得文件,可以自行创建一个临时文件。使用FileReference.upload() 的另一个优势是,它执行一种多部分表单样式上传。许多服务器端上传脚本要求这种类型的上传,因为允许用户上传文件的HTML 表单常常使用它。

另一个选项是使用URLLoader 类直接上传字节数组。以原始字节数组形式执行上传也很简单。但是,如果您的服务器脚本希望采用多部分表单上传,您将必须编写代码来创建对恰当格式的请求。除了FileReference.upload(),没有任何ActionScript API 可为您提供直接帮助。您可以使用一些第三方库,比如Eugene Zatepyakin 的开源 MultipartURLLoader 类。

FileReference.upload()

下面的示例演示了如何创建和上传临时文件:

复制代码
private function readMediaData():void
{
var imageBytes:ByteArray = new ByteArray();
dataSource.readBytes( imageBytes );
tempDir = File.createTempDirectory();
var now:Date = new Date();
var filename:String = "IMG" + now.fullYear + now.month + now.day + now.hours + now.minutes + now.seconds + ".jpg";
var temp:File = tempDir.resolvePath( filename );
var stream:FileStream = new FileStream();
stream.open( temp, FileMode.WRITE );
stream.writeBytes( imageBytes );
stream.close();
temp.addEventListener( Event.COMPLETE, uploadComplete );
temp.addEventListener( IOErrorEvent.IO_ERROR, ioError );
try
{
temp.upload( new URLRequest( serverURL ) );
}
catch( e:Error )
{
trace( e );
removeTempDir();
cameraUI.launch( MediaType.IMAGE )
}
}
private function removeTempDir():void
{
tempDir.deleteDirectory( true );
tempDir = null;
}

您可以使用 ActionScript 3 开发人员指南中提供的这个 PHP 脚本示例测试上传功能。(请记住增大脚本中定义的最大文件大小,以容纳您的设备照相机生成的图像。)

URLLoader

如果能够控制接受图像上传的服务器端脚本,可以直接上传图像字节,而无需创建临时文件。在 iOS 上,这可以节省一定的内存,因为内存中的图像数据副本可能比其他时候更少。但与任何优化一样,您应该评估是否值得为预期的收益付出努力。

下面的函数以字节数组形式上传图像:

复制代码
public function upload( data:ByteArray, destination:String, fileName:String = null ):void
{
if( fileName == null ) //Make a name with correct file type
{
var type:String = sniffFileType( buffer );
var now:Date = new Date();
fileName = "IMG" + now.fullYear + now.month +now.day +
now.hours + now.minutes + now.seconds + ".jpg";
}
loader = new URLLoader();
loader.dataFormat= URLLoaderDataFormat.BINARY;
var urlString:String = destination + "?file=" + fileName;
var request:URLRequest = new URLRequest( urlString );
request.data = data;
request.method = URLRequestMethod.POST;
request.contentType = "application/octet-stream";
loader.addEventListener( Event.COMPLETE, onUploadComplete );
loader.addEventListener(IOErrorEvent.IO_ERROR, onUploadError );
loader.load(request);
}

您可以在示例文件中的 Uploader 类中找到此函数的代码。您可以使用以下简单的 PHP 脚本测试该函数:

复制代码
<?php
if ( isset ( $GLOBALS["HTTP_RAW_POST_DATA"] )) {
$flux = $GLOBALS["HTTP_RAW_POST_DATA"];
$fp = fopen('.images/' . $_GET['file'], 'wb');
fwrite($fp, $flux);
fclose($fp);
}
?>

延伸阅读

本文介绍了如何从媒体承诺对象获取图像数据,以及两种将该数据上传到服务器的轻松方式。广泛来讲,您的上传函数将需要添加标头和参数,才符合特定的照片服务 API。也可以在 ActionScript 3 开发人员指南和 ActionScript 3 参考指南中找到关于 CameraRoll 和 CameraUI 的更多信息:

clip_image002 +clip_image003

此作品依据 Creative Commons Attribution-Noncommercial-Share Alike 3.0 Unported License 授权。超过本许可的与本作品中包含的代码示例相关的权限可在 Adobe 上找到。

查看原文: Uploading images from CameraRoll and CameraUI

2011-09-02 00:001906

评论

发布
暂无评论
发现更多内容

绝对干货!如何更好的利用 ChatGPT 提升编码效率?

booboosui

DevOps ChatGPT 智能编程 AI软件开发

QCA9880 QCA9882 What’s the difference?|802.11ac Qualcomm Atheros Network card Wallys

wallyslilly

QCA9880 QCA9882

SecureFX for Mac 附注册码 v9.4.1

胖墩儿不胖y

ftp Mac软件 FTP文件传输工具

解析大规模开发:提升企业级开发效率与质量,加速创新

龙智—DevSecOps解决方案

Atlassian Atlassian 云版 大规模开发

浅谈Redis - 热点key问题 | 京东云技术团队

京东科技开发者

redis Redis 热点key 企业号 8 月 PK 榜

ChatGPT:引领未来智能交互的生成式AI

百度开发者中心

#人工智能 ChatGPT 文心一言

[OpenHarmony]效果差、不便捷、无接口,TV项目共建如何避开这些坑?

LAVAL社区小助手

OpenHarmony

龙智案例:某大型零售企业如何打造高速、现代化的ITSM体系

龙智—DevSecOps解决方案

ITSM 工单管理系统 Jira Service Management

工时管理魔法课堂:如何在Jira中进行项目时间与成本管理?

龙智—DevSecOps解决方案

Jira插件 工时管理 jira工时

云堡垒机英语怎么翻译?它比硬件堡垒机便宜吗?

行云管家

云计算 堡垒机 云堡垒机

ChatGPT:生成式AI技术带来的变革

百度开发者中心

自动驾驶 #人工智能 文心一言

ChatGPT:生成式AI引领科学革命

百度开发者中心

#人工智能 ChatGPT 文心一言

汽车行业必读:ISO 26262和ASIL安全等级解读与攻略

龙智—DevSecOps解决方案

Helix QAC 安全标准 ISO 26262 静态分析工具

体育馆LED显示屏技术解决方案

Dylan

技术 系统 比赛 LED显示屏 体育

IntelliJ IDEA下基于Scala实现的Git检查工具

TiAmo

git IDEA IntelliJ IDEA 代码检查

在代码世界游走,没几把“锁”防身可不行 | 京东云技术团队

京东科技开发者

分布式锁 java锁 java‘ 企业号 8 月 PK 榜

mac前端代码编辑神器 sublime text 4中文激活码

mac大玩家j

代码编辑器 Mac软件 前端代码编辑

解析BeanDefinitionRegistry与BeanDefinition合并

华为云开发者联盟

开发 华为云 华为云开发者联盟 企业号 8 月 PK 榜

人工智能引领图文扫描新趋势

石臻臻的杂货铺

人工智能

hbuilderx打包苹果证书获取步骤

雪奈椰子

ios 开发 Windows开发 苹果证书

系统架构合理性的思考 | 京东云技术团队

京东科技开发者

架构设计 应用架构 企业号 8 月 PK 榜 架构合理性

高性能MySQL实战(一):表结构 | 京东物流技术团队

京东科技开发者

数据库表 表结构设计 企业号 8 月 PK 榜 高性能表

大数据平台需要做等保测评吗?怎么做?

行云管家

大数据 等保 大数据平台 等级保护 等保测评

Spring缓存是如何实现的?如何扩展使其支持过期删除功能? | 京东云技术团队

京东科技开发者

spring Spring Cache 企业号 8 月 PK 榜 spring缓存

[OpenHarmony]稳定性专项运作知识地图,一张图带你稳过“万重山”

LAVAL社区小助手

OpenHarmony

从CameraRoll和CameraUI上传图像_语言 & 开发_Joe Ward_InfoQ精选文章