평도 좋고 나름 최신(?) 연도이고 무엇보다도 air에서 쓰려고 만들었다는 점이 가장 맘에 들었다.
위에 url을 가보면 친절하게도 샘플 코드도 있다.
압축을 하건 압축을 풀건 Async / Sync도 지원 하니 참으로 편하다.
암호화 기능이 있다. encrypt / decrypt
압축 할 떄 password 지정이 가능하다.
파일 타입도 체크 가능하고, 압축 파일 내용 리스트도 볼수있고 용량 사이즈도 확인가능하고 등등등...
대용량은 해보질 않았으나 내가 테스트 해 본 바로는 30mb는 잘 되었다.
첨부 파일로 확인해보면 더 좋음..
[ 압축 사용 방법 대강 정리 ]
// 압축 클래스 생성 var writer:ZipFileWriter = new ZipFileWriter();
// openAsync 쓸꺼면 리스너 달고 아님 말고 writer.addEventListener( ZipEvent.ZIP_DATA_COMPRESS, function ); writer.addEventListener( ZipEvent.ZIP_FILE_CREATED, function );
// 압축 암호도 지정가능 writer.setPassword( "password" );
// 압축 파일의 path와 name을 지정 writer.open( File.documentsDirectory.resolvePath( "sample/test.zip" ) );
// ByteArray를 이용하여 txt를 만들고 그 안에 내용 쭈루룩~~~ var data:ByteArray = new ByteArray(); data.writeUTFBytes( "SAMPLE" );
// addBytes 메소드를 이용하여 ByteArray를 이용한 저장 가능 writer.addBytes( data, "sample.txt" );
// addDirectory 메소드로 abc란 폴더 만들기 가능 writer.addDirectory( "abc" );
// File 클래스를 이용하여 압축 할 파일의 File 클래스와 압축 파일 안에 들어갈 경로를 적어주면됨 writer.addFile( File.documentsDirectory.resolvePath( "image.jpg" ), "img/image.jpg" );
// 압축하삼~~ writer.close();
[ 압축 해제 방법 대강 정리 ]
// 생성~ var reader:ZipFileReader = new ZipFileReader();
// 압축 암호 reader.setPassword("pass");
// unzipAsync 쓸꺼면 리스너 달고 아님 말고 reader.addEventListener( ZipEvent.ZIP_DATA_UNCOMPRESS, function ); reader.addEventListener( ZipEvent.ZIP_LOAD_DATA, function );
// 압축 풀 path와 name을 지정 var zipFile:File = File.documentsDirectory.resolvePath( "sample/test.zip" ); reader.open( zipFile );
// 압축 파일안의 내용 리스트를 배열로 리턴함 var list:Array = reader.getEntries();
// list를 루프 돌려봄 for each( var entry:ZipEntry in list ) { if( entry.isDirectory() ) { // 폴더일 경우 직접 폴더 생성을 해줘야 한다! var dirFile:File = rootDir.resolvePath( entry.getFilename() ); dirFile.createDirectory(); } else { // 파일이면 FileStream으로 파일을 생성
이 예제에서는 다양한 유형의 여러 파일이 포함된 간단한 .zip 파일을 읽는 방법을 보여 줍니다. 이 예제에서는 각 파일을 ByteArray에 압축 해제하고 데스크톱에 쓰며 각 파일에 대한 메타데이터에서 관련 데이터를 추출합니다. .zip 파일의 일반 구조는 http://www.pkware.com/documents/casestudies/APPNOTE.TXT에서 유지 관리되는 PKWARE Inc. 사양을 기반으로 합니다. .zip 아카이브의 첫 번째 파일에 대한 파일 헤더와 파일 데이터가 먼저 오고 그 다음에는 각 추가 파일에 대한 파일 헤더와 파일 데이터 쌍이 옵니다. 파일 헤더의 구조는 나중에 설명하겠습니다. 다음으로 .zip 파일에는 선택적으로 데이터 설명자 레코드가 포함됩니다(일반적으로 출력 zip 파일이 디스크에 저장된 것이 아니라 메모리에 만들어진 경우). 그 다음으로는 아카이브 암호 해독 헤더, 아카이브 추가 데이터 레코드, 중앙 디렉토리 구조, 중앙 디렉토리 레코드의 Zip64 끝, 중앙 디렉토리 로케이터의 Zip64 끝 및 중앙 디렉토리 레코드의 끝과 같은 여러 추가 선택적 요소가 옵니다. 이 예제의 코드는 폴더가 포함되지 않은 zip 파일을 파싱하기 위한 용도로만 작성되었으며 데이터 설명자 레코드를 필요로 하지 않습니다. 마지막 파일 데이터 다음에 오는 모든 정보는 무시됩니다. 각 파일에 대한 파일 헤더의 형식은 다음과 같습니다.
파일 헤더 서명
4바이트
필요한 버전
2바이트
일반 용도의 비트 플래그
2바이트
압축 방법
2바이트(8=DEFLATE, 0=압축 해제)
가장 최근에 파일을 수정한 시간
2바이트
가장 최근에 파일을 수정한 날짜
2바이트
crc-32
4바이트
압축된 크기
4바이트
압축 해제된 크기
4바이트
파일 이름 길이
2바이트
추가 필드 길이
2바이트
파일 이름
variable
추가 필드
variable
파일 헤더 다음에는 실제 파일 데이터가 오며 이 데이터는 압축 방법 플래그에 따라 압축하거나 압축 해제할 수 있습니다. 파일 데이터를 압축 해제하는 경우 플래그는 0이며 DEFLATE 알고리즘을 사용하여 데이터를 압축하는 경우 8, 기타 압축 알고리즘의 경우 다른 값입니다. 이 예제에 대한 사용자 인터페이스는 레이블 및 텍스트 영역( taFiles)으로 구성됩니다. 응용 프로그램은 .zip 파일에서 발견하는 각 파일에 대해 파일 이름, 압축된 크기 및 압축 해제된 크기와 같은 정보를 텍스트 영역에 씁니다. 다음 MXML 문서는 Flex 버전 응용 프로그램의 사용자 인터페이스를 정의합니다. <?xml version="1.0" encoding="utf-8"?><mx:WindowedApplicationxmlns:mx="http://www.adobe.com/2006/mxml"layout="vertical"creationComplete="init();"><mx:Script> <![CDATA[ // The application code goes here ]]> </mx:Script><mx:Form><mx:FormItemlabel="Output"><mx:TextAreaid="taFiles"width="320"height="150"/></mx:FormItem></mx:Form></mx:WindowedApplication> 프로그램의 시작 부분에서는 다음 작업을 수행합니다.
import fl.controls.*; //requires TextArea and Label components in the Library var taFiles = new TextArea(); var output = new Label(); taFiles.setSize(320, 150); taFiles.move(10, 30); output.move(10, 10); output.width = 150; output.text = "Contents of HelloAir.zip"; addChild(taFiles); addChild(output);
bytes
ByteArray 정의
var bytes:ByteArray = new ByteArray();
파일 헤더의 메타데이터를 저장할 변수 정의
// variables for reading fixed portion of file header var fileName:String = new String(); var flNameLength:uint; var xfldLength:uint; var offset:uint; var compSize:uint; var uncompSize:uint; var compMethod:int; var signature:int;
.zip 파일을 나타낼 File(
zfile
) 및 FileStream(
zStream
) 객체 정의 및 파일을 추출할 .zip 파일(데스크톱 디렉토리에서 "HelloAIR.zip"이라는 파일)의 위치 지정
// File variables for accessing .zip file var zfile:File = File.desktopDirectory.resolvePath("HelloAIR.zip"); var zStream:FileStream = new FileStream();
Flex에서 프로그램 코드는 init() 메서드에서 시작하며 이 메서드는 루트 mx:WindowedApplication 태그에 대한 creationComplete 핸들러로 호출됩니다. // for Flex privatefunctioninit():void { 이 프로그램은 READ 모드로 .zip 파일을 열어 작업을 시작합니다. zStream.open(zfile, FileMode.READ); 그런 다음 bytes의 endian 속성을 LITTLE_ENDIAN으로 설정하여 숫자 필드의 바이트 순서에서 최하위 바이트가 먼저 나옴을 나타냅니다. bytes.endian = Endian.LITTLE_ENDIAN; 다음으로 while() 문은 파일 스트림의 현재 위치가 파일 크기보다 크거나 같을 때까지 계속되는 루프를 시작합니다. while (zStream.position < zfile.size) { 루프 내의 첫 번째 문은 파일 스트림의 첫 30바이트를 ByteArray bytes로 읽어 옵니다. 첫 30바이트는 첫 번째 파일 헤더의 고정 크기 부분을 구성합니다. // read fixed metadata portion oflocal file header zStream.readBytes(bytes, 0, 30); 다음으로 이 코드는 30바이트 헤더의 첫 번째 바이트에서 정수( signature)를 읽습니다. ZIP 형식 정의는 모든 파일 헤더의 서명이 16진수 값 0x04034b50임을 지정합니다. 서명이 다른 경우 코드가 .zip 파일의 파일 부분 밖으로 이동했으며 더 이상 추출할 파일이 없음을 나타냅니다. 이 경우 코드는 바이트 배열의 끝을 기다리는 대신 while 루프를 즉시 종료합니다. bytes.position = 0; signature = bytes.readInt(); // if no longer reading data files, quit if (signature != 0x04034b50) { break; } 코드의 다음 부분은 오프셋 위치 8에서 헤더 바이트를 읽고 해당 값을 compMethod 변수에 저장합니다. 이 바이트에는 이 파일을 압축하는 데 사용된 압축 방법을 나타내는 값이 포함됩니다. 여러 가지 압축 방법이 허용되지만 실제로는 거의 모든 .zip 파일이 DEFLATE 압축 알고리즘을 사용합니다. 현재 파일이 DEFLATE 압축으로 압축된 경우 compMethod는 8이고 파일이 압축 해제된 경우 compMethod는 0입니다. bytes.position = 8; compMethod = bytes.readByte(); // store compression method (8 == Deflate) 첫 30바이트 다음에는 파일 이름과 추가 필드가 포함될 수 있는 헤더의 가변 길이 부분이 옵니다. offset 변수는 이 부분의 크기를 저장합니다. 크기는 오프셋 26 및 28의 헤더에서 읽은 파일 이름 길이 및 추가 필드 길이를 더하여 계산됩니다. offset = 0; // stores length of variable portion of metadata bytes.position = 26; // offsetto file name length flNameLength = bytes.readShort(); // store file nameoffset += flNameLength; // add length of file name bytes.position = 28; // offsetto extra field length xfldLength = bytes.readShort(); offset += xfldLength; // add length of extra field 다음으로 이 프로그램은 offset 변수에 저장된 바이트 수에 대해 파일 헤더의 가변 길이 부분을 읽습니다. // read variable length bytes between fixed-length header and compressed file data zStream.readBytes(bytes, 30, offset); 이 프로그램은 헤더의 가변 길이 부분에서 파일 이름을 읽어 파일의 압축된 크기 및 압축 해제된(원본) 크기와 함께 텍스트 영역에 표시합니다. // Flash version bytes.position = 30; fileName = bytes.readUTFBytes(flNameLength); // read file name taFiles.appendText(fileName + "\n"); // write file nametotext area bytes.position = 18; compSize = bytes.readUnsignedInt(); // store size of compressed portion taFiles.appendText("\tCompressed size is: " + compSize + '\n'); bytes.position = 22; // offsetto uncompressed size uncompSize = bytes.readUnsignedInt(); // store uncompressed size taFiles.appendText("\tUncompressed size is: " + uncompSize + '\n'); // Flex version bytes.position = 30; fileName = bytes.readUTFBytes(flNameLength); // read file name taFiles.text += fileName + "\n"; // write file nametotext area bytes.position = 18; compSize = bytes.readUnsignedInt(); // store size of compressed portion taFiles.text += "\tCompressed size is: " + compSize + '\n'; bytes.position = 22; // offsetto uncompressed size uncompSize = bytes.readUnsignedInt(); // store uncompressed size taFiles.text += "\tUncompressed size is: " + uncompSize + '\n'; 이 예제에서는 파일의 나머지 부분을 파일 스트림에서 압축된 크기로 지정된 길이에 대한 bytes로 읽어 와 첫 30바이트의 파일 헤더를 덮어씁니다. 파일이 압축되지 않은 경우에도 압축된 크기는 정확합니다. 이는 압축된 크기가 파일의 압축 해제된 크기와 같기 때문입니다. // read compressed file tooffset0of bytes; for uncompressed files // the compressed and uncompressed size is the same if (compSize == 0) continue; zStream.readBytes(bytes, 0, compSize); 다음으로 이 예제에서는 압축된 파일을 압축 해제하고 outfile() 함수를 호출하여 출력 파일 스트림에 씁니다. 이 예제에서는 outfile()에 파일 이름과 파일 데이터가 포함된 바이트 배열을 전달합니다. if (compMethod == 8) // iffile is compressed, uncompress { bytes.uncompress(CompressionAlgorithm.DEFLATE); } outFile(fileName, bytes); // call outFile() to write out the file 앞서 언급한 예제에서 bytes.uncompress(CompressionAlgorithm.DEFLATE)는 AIR 응용 프로그램에서만 작동합니다. AIR 및 Flash Player에 대해 Deflate 방식의 데이터 압축을 수행하려면 ByteArray의 inflate() 함수를 호출해야 합니다. 닫는 괄호는 outFile() 메서드를 제외하고 while 루프, init() 메서드 및 Flex 응용 프로그램 코드의 끝을 나타냅니다. while 루프의 시작 부분으로 실행 작업이 다시 돌아가며 다른 파일을 추출하거나 마지막 파일이 처리된 경우 .zip 파일 처리를 끝내 .zip 파일의 다음 바이트 처리를 계속합니다. } // endofwhileloop } // for Flex version, endof init() methodand application outfile() 함수는 데스크톱에서 WRITE 모드로 출력 파일을 열고 filename 매개 변수를 통해 제공된 이름을 해당 파일에 지정합니다. 그런 다음 이 함수는 data 매개 변수에서 출력 파일 스트림( outStream)으로 파일 데이터를 쓰고 파일을 닫습니다. // Flash version function outFile(fileName:String, data:ByteArray):void { var outFile:File = File.desktopDirectory; // destination folder is desktop outFile = outFile.resolvePath(fileName); // name of file to write var outStream:FileStream = new FileStream(); // open output file stream inWRITE mode outStream.open(outFile, FileMode.WRITE); // writeout the file outStream.writeBytes(data, 0, data.length); // close it outStream.close(); } privatefunction outFile(fileName:String, data:ByteArray):void { var outFile:File = File.desktopDirectory; // dest folder is desktop outFile = outFile.resolvePath(fileName); // name of file to write var outStream:FileStream = new FileStream(); // open output file stream inWRITE mode outStream.open(outFile, FileMode.WRITE); // writeout the file outStream.writeBytes(data, 0, data.length); // close it outStream.close(); }
I have a list of zip and rar files in a local folder. All I need to do is to extract the contents of the zip as well as rar files and to save them in a folder with the same name of the respective archive file. Since I am new to as3, I have no clue for this. Is there any Library for this???
There are a few libraries out there that deal with ZIP files in as3, but beware that this is no easy task for a beginner in ActionScript 3.
FZip seems to be the most widely used, but it requires that the ZIP archives have Adler32 checksums. Provided with the library there is a Python script that injects the checksum into ZIP files to preprocess the files before unzipping them.
As3 port of JZlib, an as3 library to use with Fzip instead of the Python script mentioned above.
AS3 Zip Library (the author states that is slower than FZip) that avoids the Addler32 checksum problem.
To decompress zip files, you can use AS3Commons Zip (formerly know as FZip). It works withoutthe Adler32 checksum requirement mentionned in a previous answer.
Here's an example of how to extract all files in a zip archive. The function below would be called when a URLLoader object has downloaded the zip file and dispatched an Event.COMPLETE event:
import org.as3commons.zip.Zip; import org.as3commons.zip.ZipFile; private function _onZipDownloaded(e:Event):void { var ba:ByteArray = ByteArray(e.target.data); var zip:Zip = new Zip(); zip.loadBytes(ba); for(var i:uint = 0; i < zip.getFileCount(); i++) { var zipFile:ZipFile = zip.getFileAt(i); var extracted:File = directory.resolvePath(zipFile.filename); var fs:FileStream = new FileStream(); fs.open(extracted, FileMode.WRITE); fs.writeBytes(zipFile.content); fs.close(); } }
Obviously, error checking should be added to the code above but you get the idea...
Creating Zip files with Adobe AIR with the nochump library
Following up on my previous post, here's how to archive a bunch of files and make a zip file.
After getting all the files you want to put in the archive, create a new ZipEntry object for each file and add it to a ZipOutput object using the putNextEntry() of the ZipOutput class. Then pass the ByteArray data of the file to the write() method of the ZipOutput class and close the entry by using the closeEntry() method. Finally, call the finish() method once you have done the same for all the files.
Here's the code snippet :
import flash.events.FileListEvent;
import flash.filesystem.*;
import nochump.util.zip.*;
private var zipInput:File = new File();
private var zipOutput:File = new File();
private var zipFile:ZipOutput;
private var files:Array = new Array();
private function loadFiles():void
{
zipInput.browseForOpenMultiple("Open ZIP file");
zipInput.addEventListener(FileListEvent.SELECT_MULTIPLE, onSelect);
}
private function onSelect(e:FileListEvent):void
{
for(var i:uint = 0;i < e.files.length;i++)
{
var stream:FileStream = new FileStream();
var f:File = e.files[i] as File;
stream.open(f,FileMode.READ);
var fileData:ByteArray = new ByteArray();
stream.readBytes(fileData);
var file:Object = new Object();
file.name = f.name;
file.data = fileData;
files.push(file);
}
}
private function createZIP():void
{
zipFile = new ZipOutput();
for(var i:uint = 0; i < files.length; i++)
{
var zipEntry:ZipEntry = new ZipEntry(files[i].name);
zipFile.putNextEntry(zipEntry);
zipFile.write(files[i].data);
zipFile.closeEntry();
}
zipFile.finish();
zipOutput.browseForSave("Select target directory");
zipOutput.addEventListener(Event.SELECT, onSave);
}
private function onSave(e:Event):void
{
var archiveFile:File = e.target as File;
if(!archiveFile.exists)
{
var stream:FileStream = new FileStream();
stream.open(archiveFile,FileMode.WRITE);
stream.writeBytes(zipFile.byteArray);
stream.close();
}
}
As usual you can download the project archive here. Happy experimenting :D