From 990c2b2a8b8640af3f4269b551c8e06397287ef0 Mon Sep 17 00:00:00 2001 From: darrencruse Date: Sun, 1 Feb 2015 06:58:33 -0600 Subject: [PATCH 1/4] 'now matches' not 'not matches' --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index dd76c75..52f21c0 100644 --- a/README.md +++ b/README.md @@ -17,7 +17,7 @@ This fork of Whalesong differs from dyoo/whalesong in the following ways: * Adds parameters (require whalesong/lang/parameters) * Extended whalesong/image and whalesong/images - (more functions, bug fixes, not matches WeScheme) + (more functions, bug fixes, now matches WeScheme) Contributed by Emmanuel Schanzer Note: The implementation of parameters works fine, From 202d26fcb58536b583271950cd2e5b32e47dafd8 Mon Sep 17 00:00:00 2001 From: darrencruse Date: Sun, 1 Feb 2015 12:39:10 -0600 Subject: [PATCH 2/4] got play-sound working --- whalesong/image/private/js-impl.js | 52 ++++++++++++++++++++- whalesong/image/private/kernel.js | 73 ++++++++++++------------------ 2 files changed, 81 insertions(+), 44 deletions(-) diff --git a/whalesong/image/private/js-impl.js b/whalesong/image/private/js-impl.js index 3ca222d..3abe457 100644 --- a/whalesong/image/private/js-impl.js +++ b/whalesong/image/private/js-impl.js @@ -385,6 +385,56 @@ EXPORTS['video/url'] = ); }); +// We keep a cache of loaded sounds: +var audioCache = {}; + +EXPORTS['play-sound'] = + makeClosure( + 'play-sound', + 1, + function(MACHINE) { + var path = checkString(MACHINE, 'play-sound', 0); + var fileAudio = audioCache[path]; + if (fileAudio) { + // the sound was already loaded + finalizeClosureCall( + MACHINE, + fileAudio.play()); + } + else { + // this sound has never been played before + PAUSE( + function(restart) { + fileAudio = makeFileAudio(path.toString()); + audioCache[path] = fileAudio; + // let the audio file load before playing... + fileAudio.loading = true; + // (fileAudio.audio is the raw html5 Audio object) + fileAudio.audio.addEventListener('canplay', function() { + // ignore canplay events that follow the initial load + if(fileAudio.loading) { + restart(function(MACHINE) { + finalizeClosureCall( + MACHINE, + fileAudio.play()); + }); + fileAudio.loading = false; // we're done loading + } + }) + fileAudio.audio.addEventListener('error', function(e) { + restart(function(MACHINE) { + plt.baselib.exceptions.raiseFailure( + MACHINE, + plt.baselib.format.format( + "unable to load ~a: ~a", + [path, + e.message])); + }); + }); + }); + } + }); + EXPORTS['overlay'] = @@ -1398,4 +1448,4 @@ EXPORTS['name->color'] = var name = checkSymbolOrString(MACHINE, 'name->color', 0); var result = colorDb.get('' + name) || false; return result; - }); \ No newline at end of file + }); diff --git a/whalesong/image/private/kernel.js b/whalesong/image/private/kernel.js index 7796ebb..f465d14 100644 --- a/whalesong/image/private/kernel.js +++ b/whalesong/image/private/kernel.js @@ -456,7 +456,7 @@ FileImage.prototype.equals = function(other, aUnionFind) { }; ////////////////////////////////////////////////////////////////////// -// FileVideoe: String Node -> Video +// FileVideo: String Node -> Video var FileVideo = function(src, rawVideo) { BaseImage.call(this); var self = this; @@ -507,52 +507,39 @@ FileVideo.prototype.render = function(ctx, x, y) { ctx.drawImage(this.video, x, y); }; FileVideo.prototype.equals = function(other, aUnionFind) { - return (other instanceof FileVideo) && (this.src === other.src); + return (other instanceof FileVideo) && (this.src === other.src); }; ////////////////////////////////////////////////////////////////////// -// FileAudio: String Node -> Video +// FileAudio: String Boolean Audio -> FileAudio var FileAudio = function(src, loop, rawAudio) { - this.src = src; - var that = this; - if (rawAudio && (rawAudio.readyState===4)) { - that.audio = rawAudio; - that.audio.autoplay = true; - that.audio.autobuffer = true; - that.audio.currentTime = 0; - that.audio.loop = loop; - that.audio.play(); - } else { - // fixme: we may want to do something blocking here for - // onload, since we don't know at this time what the file size - // should be, nor will drawImage do the right thing until the - // file is loaded. - that.audio = document.createElement('audio'); - that.audio.src = src; - that.audio.addEventListener('canplay', function() { - that.audio.autoplay = true; - that.audio.autobuffer = true; - that.audio.currentTime = 0; - that.audio.loop = loop; - that.audio.play(); - }); - } - return true; + this.src = src; + if(rawAudio) { + this.audio = rawAudio; + } + else { + this.audio = document.createElement('audio'); + } + this.audio.src = src; + this.audio.autoplay = false; + this.audio.autobuffer = true; + // the below caused firefox to hang (but things work ok without it) + // this.audio.currentTime = 0; + this.audio.loop = loop; + return true; }; -var audioCache = {}; + FileAudio.makeInstance = function(path, loop, rawAudio) { - /* if (! (path in audioCache)) { - audioCache[path] = new FileAudio(path, loop, rawAudio, afterInit); - return audioCache[path]; - } else { - audioCache[path].audio.play(); - afterInit(audioCache[path]); - return audioCache[path]; - } - */ - return new FileAudio(path, loop, rawAudio); + return new FileAudio(path, loop, rawAudio); }; +FileAudio.prototype.play = function () { + this.audio.play(); + return true; +} + + + ////////////////////////////////////////////////////////////////////// // ImageDataImage: imageData -> image // Given an array of pixel data, create an image @@ -1384,8 +1371,8 @@ var makeFileImage = function(path, rawImage) { var makeFileVideo = function(path, rawVideo) { return FileVideo.makeInstance(path, rawVideo); }; -var makeFileAudio = function(path, rawAudio){ - return FileAudio.makeInstance(path, rawAudio) +var makeFileAudio = function(path, loop, rawAudio){ + return FileAudio.makeInstance(path, loop, rawAudio) }; @@ -1409,7 +1396,7 @@ var isFlipImage = function(x) { return x instanceof FlipImage; }; var isTextImage = function(x) { return x instanceof TextImage; }; var isFileImage = function(x) { return x instanceof FileImage; }; var isFileVideo = function(x) { return x instanceof FileVideo; }; - +var isFileAudio = function(x) { return x instanceof FileAudio; }; @@ -1503,7 +1490,7 @@ EXPORTS.isFlipImage = isFlipImage; EXPORTS.isTextImage = isTextImage; EXPORTS.isFileImage = isFileImage; EXPORTS.isFileVideo = isFileVideo; -//EXPORTS.isFileAudio = isFileAudio; +EXPORTS.isFileAudio = isFileAudio; EXPORTS.makeColor = makeColor; From 072322d417c6e9f36d6640f5bcfc325dab095cdd Mon Sep 17 00:00:00 2001 From: darrencruse Date: Sun, 1 Feb 2015 14:31:53 -0600 Subject: [PATCH 3/4] mention play-sound in the README --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 52f21c0..4f9e01b 100644 --- a/README.md +++ b/README.md @@ -19,6 +19,8 @@ This fork of Whalesong differs from dyoo/whalesong in the following ways: * Extended whalesong/image and whalesong/images (more functions, bug fixes, now matches WeScheme) Contributed by Emmanuel Schanzer + * Adds play-sound + (assumes a browser with html5 audio support) Note: The implementation of parameters works fine, as long as you don't mix parameterize with non-local-exits From 80db16648b119a12598dbae5ae0eafcc08596b48 Mon Sep 17 00:00:00 2001 From: darrencruse Date: Sun, 1 Feb 2015 14:40:14 -0600 Subject: [PATCH 4/4] play-sound was a team effort --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 4f9e01b..d9dcf4e 100644 --- a/README.md +++ b/README.md @@ -21,6 +21,7 @@ This fork of Whalesong differs from dyoo/whalesong in the following ways: Contributed by Emmanuel Schanzer * Adds play-sound (assumes a browser with html5 audio support) + Contributed by Emmanuel Schanzer and Darren Cruse Note: The implementation of parameters works fine, as long as you don't mix parameterize with non-local-exits