326 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Haxe
		
	
	
	
	
	
			
		
		
	
	
			326 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Haxe
		
	
	
	
	
	
package kha;
 | 
						|
 | 
						|
import haxe.io.Bytes;
 | 
						|
import haxe.Unserializer;
 | 
						|
 | 
						|
using StringTools;
 | 
						|
 | 
						|
private typedef AssetDataObject = {
 | 
						|
	/** File name, given by khamake, used as identifier in `Assets.someList.get()` function **/
 | 
						|
	var name: String;
 | 
						|
 | 
						|
	/** List of file paths, unified by khamake to single file with `name`. **/
 | 
						|
	var files: Array<String>;
 | 
						|
 | 
						|
	/** File sizes in bytes **/
 | 
						|
	var file_sizes: Array<Int>;
 | 
						|
 | 
						|
	/** Can be `image`, `sound`, `blob`, `font` and `video` **/
 | 
						|
	var type: String;
 | 
						|
 | 
						|
	/** Original file width (only for images) **/
 | 
						|
	var ?original_width: Int;
 | 
						|
 | 
						|
	/** Original file height (only for images) **/
 | 
						|
	var ?original_height: Int;
 | 
						|
}
 | 
						|
 | 
						|
@:forward(name, files, file_sizes, type, original_width, original_height)
 | 
						|
private abstract AssetData(AssetDataObject) from AssetDataObject {
 | 
						|
	@:op(a.b) function _get(key: String): Dynamic {
 | 
						|
		return Reflect.getProperty(this, key);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
@:build(kha.internal.AssetsBuilder.build("image"))
 | 
						|
private class ImageList {
 | 
						|
	public function new() {}
 | 
						|
 | 
						|
	public function get(name: String): Image {
 | 
						|
		return Reflect.field(this, name);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
@:build(kha.internal.AssetsBuilder.build("sound"))
 | 
						|
private class SoundList {
 | 
						|
	public function new() {}
 | 
						|
 | 
						|
	public function get(name: String): Sound {
 | 
						|
		return Reflect.field(this, name);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
@:build(kha.internal.AssetsBuilder.build("blob"))
 | 
						|
private class BlobList {
 | 
						|
	public function new() {}
 | 
						|
 | 
						|
	public function get(name: String): Blob {
 | 
						|
		return Reflect.field(this, name);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
@:build(kha.internal.AssetsBuilder.build("font"))
 | 
						|
private class FontList {
 | 
						|
	public function new() {}
 | 
						|
 | 
						|
	public function get(name: String): Font {
 | 
						|
		return Reflect.field(this, name);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
@:build(kha.internal.AssetsBuilder.build("video"))
 | 
						|
private class VideoList {
 | 
						|
	public function new() {}
 | 
						|
 | 
						|
	public function get(name: String): Video {
 | 
						|
		return Reflect.field(this, name);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
class Assets {
 | 
						|
	public static var images: ImageList = new ImageList();
 | 
						|
	public static var sounds: SoundList = new SoundList();
 | 
						|
	public static var blobs: BlobList = new BlobList();
 | 
						|
	public static var fonts: FontList = new FontList();
 | 
						|
	public static var videos: VideoList = new VideoList();
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Moves from 0 to 1. Use for loading screens.
 | 
						|
	 */
 | 
						|
	public static var progress: Float;
 | 
						|
 | 
						|
	/**
 | 
						|
		Loads all assets which were detected by khamake. When running khamake (doing so is Kha's standard build behavior)
 | 
						|
		it creates a files.json in the build/{target}-resources directoy which contains information about all assets which were found.
 | 
						|
 | 
						|
		The `callback` parameter is always called after loading, even when some or all assets had failures.
 | 
						|
 | 
						|
		An optional callback parameter `failed` is called for each asset that failed to load.
 | 
						|
 | 
						|
		The filter parameter can be used to load assets selectively. The Dynamic parameter describes the asset,
 | 
						|
		it contains the very same objects which are listed in files.json.
 | 
						|
 | 
						|
		Additionally by default all sounds are decompressed. The uncompressSoundsFilter can be used to avoid that.
 | 
						|
		Uncompressed sounds can still be played using Audio.stream which is recommended for music.
 | 
						|
	 */
 | 
						|
	public static function loadEverything(callback: () -> Void, ?filter: (item: AssetData) -> Bool, ?uncompressSoundsFilter: (soundItem: AssetData) -> Bool,
 | 
						|
			?failed: (err: AssetError) -> Void): Void {
 | 
						|
		final lists: Array<Dynamic> = [ImageList, SoundList, BlobList, FontList, VideoList];
 | 
						|
		final listInstances: Array<Dynamic> = [images, sounds, blobs, fonts, videos];
 | 
						|
		var fileCount = 0;
 | 
						|
		var byteCount = 0;
 | 
						|
 | 
						|
		for (i in 0...lists.length) {
 | 
						|
			final list = lists[i];
 | 
						|
			for (file in Type.getInstanceFields(list)) {
 | 
						|
				if (file.endsWith("Description")) {
 | 
						|
					fileCount++;
 | 
						|
				}
 | 
						|
				else if (file.endsWith("Size")) {
 | 
						|
					var size: Int = Reflect.field(listInstances[i], file);
 | 
						|
					byteCount += size;
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (fileCount == 0) {
 | 
						|
			callback();
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		var filesLeft = fileCount;
 | 
						|
		var bytesLeft = byteCount;
 | 
						|
 | 
						|
		function onLoaded(bytes: Int): Void {
 | 
						|
			filesLeft--;
 | 
						|
			bytesLeft -= bytes;
 | 
						|
			progress = 1 - (bytesLeft / byteCount);
 | 
						|
			if (filesLeft == 0)
 | 
						|
				callback();
 | 
						|
		}
 | 
						|
 | 
						|
		function onError(err: AssetError, bytes: Int): Void {
 | 
						|
			reporter(failed)(err);
 | 
						|
			onLoaded(bytes);
 | 
						|
		}
 | 
						|
 | 
						|
		function loadFunc(desc: Dynamic, done: (bytes: Int) -> Void, failure: (err: AssetError, bytes: Int) -> Void): Void {
 | 
						|
			final name = desc.name;
 | 
						|
			final size = desc.file_sizes[0];
 | 
						|
			switch (desc.type) {
 | 
						|
				case "image":
 | 
						|
					Assets.loadImage(name, function(image: Image) done(size), function(err: AssetError) {
 | 
						|
						onError(err, size);
 | 
						|
					});
 | 
						|
				case "sound":
 | 
						|
					Assets.loadSound(name, function(sound: Sound) {
 | 
						|
						if (uncompressSoundsFilter == null || uncompressSoundsFilter(desc)) {
 | 
						|
							sound.uncompress(function() {
 | 
						|
								done(size);
 | 
						|
							});
 | 
						|
						}
 | 
						|
						else {
 | 
						|
							done(size);
 | 
						|
						}
 | 
						|
					}, function(err: AssetError) {
 | 
						|
						onError(err, size);
 | 
						|
					});
 | 
						|
				case "blob":
 | 
						|
					Assets.loadBlob(name, function(blob: Blob) done(size), function(err: AssetError) {
 | 
						|
						onError(err, size);
 | 
						|
					});
 | 
						|
				case "font":
 | 
						|
					Assets.loadFont(name, function(font: Font) done(size), function(err: AssetError) {
 | 
						|
						onError(err, size);
 | 
						|
					});
 | 
						|
				case "video":
 | 
						|
					Assets.loadVideo(name, function(video: Video) done(size), function(err: AssetError) {
 | 
						|
						onError(err, size);
 | 
						|
					});
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		for (i in 0...lists.length) {
 | 
						|
			final list = lists[i];
 | 
						|
			final listInstance = listInstances[i];
 | 
						|
			for (field in Type.getInstanceFields(list)) {
 | 
						|
				if (!field.endsWith("Description"))
 | 
						|
					continue;
 | 
						|
				final desc = Reflect.field(listInstance, field);
 | 
						|
				if (filter == null || filter(desc)) {
 | 
						|
					loadFunc(desc, onLoaded, onError);
 | 
						|
				}
 | 
						|
				else {
 | 
						|
					onLoaded(desc.file_sizes[0]);
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Loads an image by name which was preprocessed by khamake.
 | 
						|
	 *
 | 
						|
	 * @param	name The name as defined by the khafile.
 | 
						|
	 * @param	done A callback.
 | 
						|
	 */
 | 
						|
	public static function loadImage(name: String, done: Image->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = Reflect.field(images, name + "Description");
 | 
						|
		if (description == null) {
 | 
						|
			reporter(failed, pos)({url: name, error: "Name not found"});
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		LoaderImpl.loadImageFromDescription(description, function(image: Image) {
 | 
						|
			Reflect.setField(images, name, image);
 | 
						|
			done(image);
 | 
						|
		}, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Loads an image from a path. Most targets support PNG and JPEG formats.
 | 
						|
	 *
 | 
						|
	 * @param	path The path to the image file.
 | 
						|
	 * @param   readable If true, a copy of the image will be kept in main memory for image read operations.
 | 
						|
	 * @param	done A callback.
 | 
						|
	 */
 | 
						|
	public static function loadImageFromPath(path: String, readable: Bool, done: Image->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = {files: [path], readable: readable};
 | 
						|
		LoaderImpl.loadImageFromDescription(description, done, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static var imageFormats(get, null): Array<String>;
 | 
						|
 | 
						|
	static function get_imageFormats(): Array<String> {
 | 
						|
		return LoaderImpl.getImageFormats();
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadBlob(name: String, done: Blob->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = Reflect.field(blobs, name + "Description");
 | 
						|
		if (description == null) {
 | 
						|
			reporter(failed, pos)({url: name, error: "Name not found"});
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		LoaderImpl.loadBlobFromDescription(description, function(blob: Blob) {
 | 
						|
			Reflect.setField(blobs, name, blob);
 | 
						|
			done(blob);
 | 
						|
		}, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadBlobFromPath(path: String, done: Blob->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = {files: [path]};
 | 
						|
		LoaderImpl.loadBlobFromDescription(description, done, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadSound(name: String, done: Sound->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = Reflect.field(sounds, name + "Description");
 | 
						|
		if (description == null) {
 | 
						|
			reporter(failed, pos)({url: name, error: "Name not found"});
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		return LoaderImpl.loadSoundFromDescription(description, function(sound: Sound) {
 | 
						|
			Reflect.setField(sounds, name, sound);
 | 
						|
			done(sound);
 | 
						|
		}, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadSoundFromPath(path: String, done: Sound->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = {files: [path]};
 | 
						|
		return LoaderImpl.loadSoundFromDescription(description, done, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static var soundFormats(get, null): Array<String>;
 | 
						|
 | 
						|
	static function get_soundFormats(): Array<String> {
 | 
						|
		return LoaderImpl.getSoundFormats();
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadFont(name: String, done: Font->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = Reflect.field(fonts, name + "Description");
 | 
						|
		if (description == null) {
 | 
						|
			reporter(failed, pos)({url: name, error: "Name not found"});
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		return LoaderImpl.loadFontFromDescription(description, function(font: Font) {
 | 
						|
			Reflect.setField(fonts, name, font);
 | 
						|
			done(font);
 | 
						|
		}, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadFontFromPath(path: String, done: Font->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = {files: [path]};
 | 
						|
		return LoaderImpl.loadFontFromDescription(description, done, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static var fontFormats(get, null): Array<String>;
 | 
						|
 | 
						|
	static function get_fontFormats(): Array<String> {
 | 
						|
		return ["ttf"];
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadVideo(name: String, done: Video->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = Reflect.field(videos, name + "Description");
 | 
						|
		if (description == null) {
 | 
						|
			reporter(failed, pos)({url: name, error: "Name not found"});
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		return LoaderImpl.loadVideoFromDescription(description, function(video: Video) {
 | 
						|
			Reflect.setField(videos, name, video);
 | 
						|
			done(video);
 | 
						|
		}, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static function loadVideoFromPath(path: String, done: Video->Void, ?failed: AssetError->Void, ?pos: haxe.PosInfos): Void {
 | 
						|
		var description = {files: [path]};
 | 
						|
		return LoaderImpl.loadVideoFromDescription(description, done, reporter(failed, pos));
 | 
						|
	}
 | 
						|
 | 
						|
	public static var videoFormats(get, null): Array<String>;
 | 
						|
 | 
						|
	static function get_videoFormats(): Array<String> {
 | 
						|
		return LoaderImpl.getVideoFormats();
 | 
						|
	}
 | 
						|
 | 
						|
	public static function reporter(custom: AssetError->Void, ?pos: haxe.PosInfos) {
 | 
						|
		return custom != null ? custom : haxe.Log.trace.bind(_, pos);
 | 
						|
	}
 | 
						|
}
 |