写点什么

从 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:002020

评论

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

搭建一站式OpenHarmony设备开发Windows开发环境。

坚果

OpenHarmony 三周年连更

第二届中国国际软件发展大会|华为:构筑坚实软件根基,赋能数字经济高质量发展

Geek_2d6073

MySQL的varchar字段最大长度真的是65535吗?

Java你猿哥

MySQL 后端 SSM框架 varchar

中移链合约常用开发介绍(三)工程化开发智能合约

BSN研习社

从原理聊JVM(一):染色标记和垃圾回收算法

小小怪下士

Java 程序员 后端 JVM

右键助手:MouseBoost PRO mac激活版

真大的脸盆

Mac Mac 软件 鼠标增强工具 鼠标辅助软件

大厂面试难?对标大厂“Java系统性能优化实战”二面阿里轻松搞定

Java你猿哥

面试 性能优化 SSM框架 Java性能优化

解决90%的面试!GitHub新兴“java面试手册 2023” 一网打尽BAT大厂

Java你猿哥

Java MySQL Spring Boot JVM MySQL面试

震撼!阿里架构师全新产出Java面试突击宝典。Github标星疯涨!

Java你猿哥

spring Spring Boot JVM mybatis java面试

破防了!阿里用17个真实企业级项目阐述Java系统分析与架构设计

Java你猿哥

微服务架构 架构设计 Java系统性能 Redis开发与运维 MySQl部署

数智融合,生态链接丨 亚信科技“信伙伴”交流会(成都站)成功举办

亚信AntDB数据库

AntDB AntDB数据库 企业号 4 月 PK 榜

海通证券与易观千帆达成合作,构建优质客户生态圈

易观分析

金融 证券 经济

从原理聊JVM(一):染色标记和垃圾回收算法

京东科技开发者

Java JVM 三色标记 垃圾回收器 企业号 4 月 PK 榜

微服务 Spring Boot 整合Redis分布式锁 实现优惠卷秒杀 一人一单

Bug终结者

redis 底层原理 三周年连更

DataEase 对接明道云展示表格应用数据

搞大屏的小北

数据可视化 明道云 对接api 展示明道云

行业分析| 新的学习方式——在线自习室

anyRTC开发者

音视频 在线教育 视频直播 直播连麦 在线自习室

某程序员:被裁了要求公司足额补缴全部公积金,一次补二十多万!

Java你猿哥

Java 程序员 SSM框架

校企共建|阿里云与重庆大学人才培养交流会顺利举行

云布道师

阿里云

大数据Hadoop之——HDFS小文件问题与处理实战操作

Openlab_cosmoplat

hdfs 开源社区 大数据Hadoop

BSN-DDC基础网络详解(九):跨链机制

BSN研习社

深度学习基础入门篇[六(1)]:模型调优:注意力机制[多头注意力、自注意力],正则化【L1、L2,Dropout,Drop Connect】等

汀丶人工智能

人工智能 机器学习 深度学习

Spring Boot如何使用Undertow容器?超级详细,建议收藏

bug菌

Spring Boot 三周年连更 Undertow

和 if else说再见,SpringBoot 这样做参数校验才足够优雅!

Java你猿哥

Java spring Spring Boot ssm if-else

薪资结构重铸: Zebec将业务范围扩大到Web2薪资管理领域

威廉META

css实现瀑布流效果

格斗家不爱在外太空沉思

CSS 三周年连更

【4.14-4.21】写作社区优秀技术博文一览

InfoQ写作社区官方

热门活动 优质创作周报

薪资结构重铸: Zebec将业务范围扩大到Web2薪资管理领域

鳄鱼视界

求你看完再去面试!涨薪必备分布式事务小抄,狂怼面试官

程序知音

Java 分布式 后端 java架构 Java进阶

企业微信接入系列-上传附件资源

六月的雨在InfoQ

企业微信 三周年连更 企业微信接入 企微上传附件

Spring Boot过滤器实现项目内接口过滤

Java你猿哥

Java Spring Boot SSM框架

进击的 Java !

OpenAnolis小助手

Java 开源 云原生 GOTC 龙蜥技术

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