From dcce17e6a5bc938ef1c21c6f3026dfcebf228bd4 Mon Sep 17 00:00:00 2001
From: llaffont <llaffont@git-test.afi-sa.fr>
Date: Wed, 21 Nov 2012 09:16:22 +0000
Subject: [PATCH] =?UTF-8?q?Int=C3=A9gration=20Player=20AudioJS?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

---
 .gitattributes                                |   8 ++
 .../ZendAfi/Controller/Plugin/AdminAuth.php   |   1 -
 library/ZendAfi/View/Helper/AudioJsPlayer.php |  36 +++++++
 public/admin/js/audiojs/LICENSE               |  19 ++++
 public/admin/js/audiojs/audiojs/audio.min.js  |  24 +++++
 public/admin/js/audiojs/audiojs/audiojs.swf   | Bin 0 -> 1644 bytes
 .../js/audiojs/audiojs/player-graphics.gif    | Bin 0 -> 4499 bytes
 public/admin/js/audiojs/includes/index.css    |  44 ++++++++
 public/admin/js/audiojs/index.html            |  96 ++++++++++++++++++
 .../ZendAfi/View/Helper/AudioJSPlayerTest.php |  48 +++++++++
 10 files changed, 275 insertions(+), 1 deletion(-)
 create mode 100644 library/ZendAfi/View/Helper/AudioJsPlayer.php
 create mode 100644 public/admin/js/audiojs/LICENSE
 create mode 100644 public/admin/js/audiojs/audiojs/audio.min.js
 create mode 100644 public/admin/js/audiojs/audiojs/audiojs.swf
 create mode 100644 public/admin/js/audiojs/audiojs/player-graphics.gif
 create mode 100644 public/admin/js/audiojs/includes/index.css
 create mode 100644 public/admin/js/audiojs/index.html
 create mode 100644 tests/library/ZendAfi/View/Helper/AudioJSPlayerTest.php

diff --git a/.gitattributes b/.gitattributes
index 6f9e4496c69..8f12855d8dc 100644
--- a/.gitattributes
+++ b/.gitattributes
@@ -2223,6 +2223,7 @@ library/ZendAfi/View/Helper/Album/RssFeedVisitor.php -text
 library/ZendAfi/View/Helper/Album/XspfPlaylistVisitor.php -text
 library/ZendAfi/View/Helper/AlbumRessourceInfos.php -text
 library/ZendAfi/View/Helper/Article/FormulairesCsvVisitor.php -text
+library/ZendAfi/View/Helper/AudioJsPlayer.php -text
 library/ZendAfi/View/Helper/Avis.php -text
 library/ZendAfi/View/Helper/AvisCms.php -text
 library/ZendAfi/View/Helper/BarreNav.php -text
@@ -3004,6 +3005,12 @@ public/admin/images/zone/zone_4.jpg -text
 public/admin/images/zone/zone_5.jpg -text
 public/admin/images/zone/zone_6.jpg -text
 public/admin/images/zone/zone_7.jpg -text
+public/admin/js/audiojs/LICENSE -text
+public/admin/js/audiojs/audiojs/audio.min.js -text
+public/admin/js/audiojs/audiojs/audiojs.swf -text svneol=unset#unset
+public/admin/js/audiojs/audiojs/player-graphics.gif -text svneol=unset#unset
+public/admin/js/audiojs/includes/index.css -text
+public/admin/js/audiojs/index.html -text
 public/admin/js/cacheimages.js -text
 public/admin/js/calendar.js -text
 public/admin/js/cfg.accueil.js -text
@@ -4745,6 +4752,7 @@ tests/library/ZendAfi/View/Helper/Accueil/SitoTest.php -text
 tests/library/ZendAfi/View/Helper/Accueil/TagsTest.php -text
 tests/library/ZendAfi/View/Helper/Admin/AdminHelpLinkTest.php -text
 tests/library/ZendAfi/View/Helper/Admin/ImageViewersOptionsTest.php -text
+tests/library/ZendAfi/View/Helper/AudioJSPlayerTest.php -text
 tests/library/ZendAfi/View/Helper/AvisTest.php -text
 tests/library/ZendAfi/View/Helper/CkEditorTest.php -text
 tests/library/ZendAfi/View/Helper/ComboCodificationTest.php -text
diff --git a/library/ZendAfi/Controller/Plugin/AdminAuth.php b/library/ZendAfi/Controller/Plugin/AdminAuth.php
index eba7bb5e3a7..1e01534c33c 100644
--- a/library/ZendAfi/Controller/Plugin/AdminAuth.php
+++ b/library/ZendAfi/Controller/Plugin/AdminAuth.php
@@ -88,7 +88,6 @@ class ZendAfi_Controller_Plugin_AdminAuth extends Zend_Controller_Plugin_Abstrac
 		if (!$acl->has($resource)) $resource = null;
 				
 		// Test du role et redirection vers opac si pas autorisé
-		xdebug_break();
 		return $acl->isAllowed($role, $resource);
 	}
 }
diff --git a/library/ZendAfi/View/Helper/AudioJsPlayer.php b/library/ZendAfi/View/Helper/AudioJsPlayer.php
new file mode 100644
index 00000000000..6102c220a3e
--- /dev/null
+++ b/library/ZendAfi/View/Helper/AudioJsPlayer.php
@@ -0,0 +1,36 @@
+<?php
+/**
+ * Copyright (c) 2012, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * AFI-OPAC 2.0 is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * AFI-OPAC 2.0 is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with AFI-OPAC 2.0; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA 
+ */
+
+
+class ZendAfi_View_Helper_AudioJsPlayer extends Zend_View_Helper_HtmlElement {
+	/**
+	 * Exemple:
+	 * $this->audioJsPlayer();
+	 * Voir: http://kolber.github.com/audiojs/
+	 */
+	public function audioJsPlayer() {
+		return Class_ScriptLoader::getInstance()
+			->addAdminScript('audiojs/audiojs/audiomin.js')
+			->addJQueryReady('audiojs.events.ready(function() {audiojs.createAll();})');
+	}
+}
+
+?>
\ No newline at end of file
diff --git a/public/admin/js/audiojs/LICENSE b/public/admin/js/audiojs/LICENSE
new file mode 100644
index 00000000000..4fbf29dde7c
--- /dev/null
+++ b/public/admin/js/audiojs/LICENSE
@@ -0,0 +1,19 @@
+Copyright (c) 2010 Anthony Kolber (http://aestheticallyloyal.com)
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+THE SOFTWARE.
\ No newline at end of file
diff --git a/public/admin/js/audiojs/audiojs/audio.min.js b/public/admin/js/audiojs/audiojs/audio.min.js
new file mode 100644
index 00000000000..e5fde2b1fdf
--- /dev/null
+++ b/public/admin/js/audiojs/audiojs/audio.min.js
@@ -0,0 +1,24 @@
+(function(h,o,g){var p=function(){for(var b=/audio(.min)?.js.*/,a=document.getElementsByTagName("script"),c=0,d=a.length;c<d;c++){var e=a[c].getAttribute("src");if(b.test(e))return e.replace(b,"")}}();g[h]={instanceCount:0,instances:{},flashSource:'      <object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" id="$1" width="1" height="1" name="$1" style="position: absolute; left: -1px;">         <param name="movie" value="$2?playerInstance='+h+'.instances[\'$1\']&datetime=$3">         <param name="allowscriptaccess" value="always">         <embed name="$1" src="$2?playerInstance='+
+h+'.instances[\'$1\']&datetime=$3" width="1" height="1" allowscriptaccess="always">       </object>',settings:{autoplay:false,loop:false,preload:true,imageLocation:p+"player-graphics.gif",swfLocation:p+"audiojs.swf",useFlash:function(){var b=document.createElement("audio");return!(b.canPlayType&&b.canPlayType("audio/mpeg;").replace(/no/,""))}(),hasFlash:function(){if(navigator.plugins&&navigator.plugins.length&&navigator.plugins["Shockwave Flash"])return true;else if(navigator.mimeTypes&&navigator.mimeTypes.length){var b=
+navigator.mimeTypes["application/x-shockwave-flash"];return b&&b.enabledPlugin}else try{new ActiveXObject("ShockwaveFlash.ShockwaveFlash");return true}catch(a){}return false}(),createPlayer:{markup:'          <div class="play-pause">             <p class="play"></p>             <p class="pause"></p>             <p class="loading"></p>             <p class="error"></p>           </div>           <div class="scrubber">             <div class="progress"></div>             <div class="loaded"></div>           </div>           <div class="time">             <em class="played">00:00</em>/<strong class="duration">00:00</strong>           </div>           <div class="error-message"></div>',
+playPauseClass:"play-pause",scrubberClass:"scrubber",progressClass:"progress",loaderClass:"loaded",timeClass:"time",durationClass:"duration",playedClass:"played",errorMessageClass:"error-message",playingClass:"playing",loadingClass:"loading",errorClass:"error"},css:'        .audiojs audio { position: absolute; left: -1px; }         .audiojs { width: 460px; height: 36px; background: #404040; overflow: hidden; font-family: monospace; font-size: 12px;           background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0, #444), color-stop(0.5, #555), color-stop(0.51, #444), color-stop(1, #444));           background-image: -moz-linear-gradient(center top, #444 0%, #555 50%, #444 51%, #444 100%);           -webkit-box-shadow: 1px 1px 8px rgba(0, 0, 0, 0.3); -moz-box-shadow: 1px 1px 8px rgba(0, 0, 0, 0.3);           -o-box-shadow: 1px 1px 8px rgba(0, 0, 0, 0.3); box-shadow: 1px 1px 8px rgba(0, 0, 0, 0.3); }         .audiojs .play-pause { width: 25px; height: 40px; padding: 4px 6px; margin: 0px; float: left; overflow: hidden; border-right: 1px solid #000; }         .audiojs p { display: none; width: 25px; height: 40px; margin: 0px; cursor: pointer; }         .audiojs .play { display: block; }         .audiojs .scrubber { position: relative; float: left; width: 280px; background: #5a5a5a; height: 14px; margin: 10px; border-top: 1px solid #3f3f3f; border-left: 0px; border-bottom: 0px; overflow: hidden; }         .audiojs .progress { position: absolute; top: 0px; left: 0px; height: 14px; width: 0px; background: #ccc; z-index: 1;           background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0, #ccc), color-stop(0.5, #ddd), color-stop(0.51, #ccc), color-stop(1, #ccc));           background-image: -moz-linear-gradient(center top, #ccc 0%, #ddd 50%, #ccc 51%, #ccc 100%); }         .audiojs .loaded { position: absolute; top: 0px; left: 0px; height: 14px; width: 0px; background: #000;           background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0, #222), color-stop(0.5, #333), color-stop(0.51, #222), color-stop(1, #222));           background-image: -moz-linear-gradient(center top, #222 0%, #333 50%, #222 51%, #222 100%); }         .audiojs .time { float: left; height: 36px; line-height: 36px; margin: 0px 0px 0px 6px; padding: 0px 6px 0px 12px; border-left: 1px solid #000; color: #ddd; text-shadow: 1px 1px 0px rgba(0, 0, 0, 0.5); }         .audiojs .time em { padding: 0px 2px 0px 0px; color: #f9f9f9; font-style: normal; }         .audiojs .time strong { padding: 0px 0px 0px 2px; font-weight: normal; }         .audiojs .error-message { float: left; display: none; margin: 0px 10px; height: 36px; width: 400px; overflow: hidden; line-height: 36px; white-space: nowrap; color: #fff;           text-overflow: ellipsis; -o-text-overflow: ellipsis; -icab-text-overflow: ellipsis; -khtml-text-overflow: ellipsis; -moz-text-overflow: ellipsis; -webkit-text-overflow: ellipsis; }         .audiojs .error-message a { color: #eee; text-decoration: none; padding-bottom: 1px; border-bottom: 1px solid #999; white-space: wrap; }                 .audiojs .play { background: url("$1") -2px -1px no-repeat; }         .audiojs .loading { background: url("$1") -2px -31px no-repeat; }         .audiojs .error { background: url("$1") -2px -61px no-repeat; }         .audiojs .pause { background: url("$1") -2px -91px no-repeat; }                 .playing .play, .playing .loading, .playing .error { display: none; }         .playing .pause { display: block; }                 .loading .play, .loading .pause, .loading .error { display: none; }         .loading .loading { display: block; }                 .error .time, .error .play, .error .pause, .error .scrubber, .error .loading { display: none; }         .error .error { display: block; }         .error .play-pause p { cursor: auto; }         .error .error-message { display: block; }',
+trackEnded:function(){},flashError:function(){var b=this.settings.createPlayer,a=j(b.errorMessageClass,this.wrapper),c='Missing <a href="http://get.adobe.com/flashplayer/">flash player</a> plugin.';if(this.mp3)c+=' <a href="'+this.mp3+'">Download audio file</a>.';g[h].helpers.removeClass(this.wrapper,b.loadingClass);g[h].helpers.addClass(this.wrapper,b.errorClass);a.innerHTML=c},loadError:function(){var b=this.settings.createPlayer,a=j(b.errorMessageClass,this.wrapper);g[h].helpers.removeClass(this.wrapper,
+b.loadingClass);g[h].helpers.addClass(this.wrapper,b.errorClass);a.innerHTML='Error loading: "'+this.mp3+'"'},init:function(){g[h].helpers.addClass(this.wrapper,this.settings.createPlayer.loadingClass)},loadStarted:function(){var b=this.settings.createPlayer,a=j(b.durationClass,this.wrapper),c=Math.floor(this.duration/60),d=Math.floor(this.duration%60);g[h].helpers.removeClass(this.wrapper,b.loadingClass);a.innerHTML=(c<10?"0":"")+c+":"+(d<10?"0":"")+d},loadProgress:function(b){var a=this.settings.createPlayer,
+c=j(a.scrubberClass,this.wrapper);j(a.loaderClass,this.wrapper).style.width=c.offsetWidth*b+"px"},playPause:function(){this.playing?this.settings.play():this.settings.pause()},play:function(){g[h].helpers.addClass(this.wrapper,this.settings.createPlayer.playingClass)},pause:function(){g[h].helpers.removeClass(this.wrapper,this.settings.createPlayer.playingClass)},updatePlayhead:function(b){var a=this.settings.createPlayer,c=j(a.scrubberClass,this.wrapper);j(a.progressClass,this.wrapper).style.width=
+c.offsetWidth*b+"px";a=j(a.playedClass,this.wrapper);c=this.duration*b;b=Math.floor(c/60);c=Math.floor(c%60);a.innerHTML=(b<10?"0":"")+b+":"+(c<10?"0":"")+c}},create:function(b,a){a=a||{};return b.length?this.createAll(a,b):this.newInstance(b,a)},createAll:function(b,a){var c=a||document.getElementsByTagName("audio"),d=[];b=b||{};for(var e=0,i=c.length;e<i;e++)d.push(this.newInstance(c[e],b));return d},newInstance:function(b,a){var c=this.helpers.clone(this.settings),d="audiojs"+this.instanceCount,
+e="audiojs_wrapper"+this.instanceCount;this.instanceCount++;if(b.getAttribute("autoplay")!=null)c.autoplay=true;if(b.getAttribute("loop")!=null)c.loop=true;if(b.getAttribute("preload")=="none")c.preload=false;a&&this.helpers.merge(c,a);if(c.createPlayer.markup)b=this.createPlayer(b,c.createPlayer,e);else b.parentNode.setAttribute("id",e);e=new g[o](b,c);c.css&&this.helpers.injectCss(e,c.css);if(c.useFlash&&c.hasFlash){this.injectFlash(e,d);this.attachFlashEvents(e.wrapper,e)}else c.useFlash&&!c.hasFlash&&
+this.settings.flashError.apply(e);if(!c.useFlash||c.useFlash&&c.hasFlash)this.attachEvents(e.wrapper,e);return this.instances[d]=e},createPlayer:function(b,a,c){var d=document.createElement("div"),e=b.cloneNode(true);d.setAttribute("class","audiojs");d.setAttribute("className","audiojs");d.setAttribute("id",c);if(e.outerHTML&&!document.createElement("audio").canPlayType){e=this.helpers.cloneHtml5Node(b);d.innerHTML=a.markup;d.appendChild(e);b.outerHTML=d.outerHTML;d=document.getElementById(c)}else{d.appendChild(e);
+d.innerHTML+=a.markup;b.parentNode.replaceChild(d,b)}return d.getElementsByTagName("audio")[0]},attachEvents:function(b,a){if(a.settings.createPlayer){var c=a.settings.createPlayer,d=j(c.playPauseClass,b),e=j(c.scrubberClass,b);g[h].events.addListener(d,"click",function(){a.playPause.apply(a)});g[h].events.addListener(e,"click",function(i){i=i.clientX;var f=this,k=0;if(f.offsetParent){do k+=f.offsetLeft;while(f=f.offsetParent)}a.skipTo((i-k)/e.offsetWidth)});if(!a.settings.useFlash){g[h].events.trackLoadProgress(a);
+g[h].events.addListener(a.element,"timeupdate",function(){a.updatePlayhead.apply(a)});g[h].events.addListener(a.element,"ended",function(){a.trackEnded.apply(a)});g[h].events.addListener(a.source,"error",function(){clearInterval(a.readyTimer);clearInterval(a.loadTimer);a.settings.loadError.apply(a)})}}},attachFlashEvents:function(b,a){a.swfReady=false;a.load=function(c){a.mp3=c;a.swfReady&&a.element.load(c)};a.loadProgress=function(c,d){a.loadedPercent=c;a.duration=d;a.settings.loadStarted.apply(a);
+a.settings.loadProgress.apply(a,[c])};a.skipTo=function(c){if(!(c>a.loadedPercent)){a.updatePlayhead.call(a,[c]);a.element.skipTo(c)}};a.updatePlayhead=function(c){a.settings.updatePlayhead.apply(a,[c])};a.play=function(){if(!a.settings.preload){a.settings.preload=true;a.element.init(a.mp3)}a.playing=true;a.element.pplay();a.settings.play.apply(a)};a.pause=function(){a.playing=false;a.element.ppause();a.settings.pause.apply(a)};a.setVolume=function(c){a.element.setVolume(c)};a.loadStarted=function(){a.swfReady=
+true;a.settings.preload&&a.element.init(a.mp3);a.settings.autoplay&&a.play.apply(a)}},injectFlash:function(b,a){var c=this.flashSource.replace(/\$1/g,a);c=c.replace(/\$2/g,b.settings.swfLocation);c=c.replace(/\$3/g,+new Date+Math.random());var d=b.wrapper.innerHTML,e=document.createElement("div");e.innerHTML=c+d;b.wrapper.innerHTML=e.innerHTML;b.element=this.helpers.getSwf(a)},helpers:{merge:function(b,a){for(attr in a)if(b.hasOwnProperty(attr)||a.hasOwnProperty(attr))b[attr]=a[attr]},clone:function(b){if(b==
+null||typeof b!=="object")return b;var a=new b.constructor,c;for(c in b)a[c]=arguments.callee(b[c]);return a},addClass:function(b,a){RegExp("(\\s|^)"+a+"(\\s|$)").test(b.className)||(b.className+=" "+a)},removeClass:function(b,a){b.className=b.className.replace(RegExp("(\\s|^)"+a+"(\\s|$)")," ")},injectCss:function(b,a){for(var c="",d=document.getElementsByTagName("style"),e=a.replace(/\$1/g,b.settings.imageLocation),i=0,f=d.length;i<f;i++){var k=d[i].getAttribute("title");if(k&&~k.indexOf("audiojs")){f=
+d[i];if(f.innerHTML===e)return;c=f.innerHTML;break}}d=document.getElementsByTagName("head")[0];i=d.firstChild;f=document.createElement("style");if(d){f.setAttribute("type","text/css");f.setAttribute("title","audiojs");if(f.styleSheet)f.styleSheet.cssText=c+e;else f.appendChild(document.createTextNode(c+e));i?d.insertBefore(f,i):d.appendChild(styleElement)}},cloneHtml5Node:function(b){var a=document.createDocumentFragment(),c=a.createElement?a:document;c.createElement("audio");c=c.createElement("div");
+a.appendChild(c);c.innerHTML=b.outerHTML;return c.firstChild},getSwf:function(b){b=document[b]||window[b];return b.length>1?b[b.length-1]:b}},events:{memoryLeaking:false,listeners:[],addListener:function(b,a,c){if(b.addEventListener)b.addEventListener(a,c,false);else if(b.attachEvent){this.listeners.push(b);if(!this.memoryLeaking){window.attachEvent("onunload",function(){if(this.listeners)for(var d=0,e=this.listeners.length;d<e;d++)g[h].events.purge(this.listeners[d])});this.memoryLeaking=true}b.attachEvent("on"+
+a,function(){c.call(b,window.event)})}},trackLoadProgress:function(b){if(b.settings.preload){var a,c;b=b;var d=/(ipod|iphone|ipad)/i.test(navigator.userAgent);a=setInterval(function(){if(b.element.readyState>-1)d||b.init.apply(b);if(b.element.readyState>1){b.settings.autoplay&&b.play.apply(b);clearInterval(a);c=setInterval(function(){b.loadProgress.apply(b);b.loadedPercent>=1&&clearInterval(c)})}},10);b.readyTimer=a;b.loadTimer=c}},purge:function(b){var a=b.attributes,c;if(a)for(c=0;c<a.length;c+=
+1)if(typeof b[a[c].name]==="function")b[a[c].name]=null;if(a=b.childNodes)for(c=0;c<a.length;c+=1)purge(b.childNodes[c])},ready:function(){return function(b){var a=window,c=false,d=true,e=a.document,i=e.documentElement,f=e.addEventListener?"addEventListener":"attachEvent",k=e.addEventListener?"removeEventListener":"detachEvent",n=e.addEventListener?"":"on",m=function(l){if(!(l.type=="readystatechange"&&e.readyState!="complete")){(l.type=="load"?a:e)[k](n+l.type,m,false);if(!c&&(c=true))b.call(a,l.type||
+l)}},q=function(){try{i.doScroll("left")}catch(l){setTimeout(q,50);return}m("poll")};if(e.readyState=="complete")b.call(a,"lazy");else{if(e.createEventObject&&i.doScroll){try{d=!a.frameElement}catch(r){}d&&q()}e[f](n+"DOMContentLoaded",m,false);e[f](n+"readystatechange",m,false);a[f](n+"load",m,false)}}}()}};g[o]=function(b,a){this.element=b;this.wrapper=b.parentNode;this.source=b.getElementsByTagName("source")[0]||b;this.mp3=function(c){var d=c.getElementsByTagName("source")[0];return c.getAttribute("src")||
+(d?d.getAttribute("src"):null)}(b);this.settings=a;this.loadStartedCalled=false;this.loadedPercent=0;this.duration=1;this.playing=false};g[o].prototype={updatePlayhead:function(){this.settings.updatePlayhead.apply(this,[this.element.currentTime/this.duration])},skipTo:function(b){if(!(b>this.loadedPercent)){this.element.currentTime=this.duration*b;this.updatePlayhead()}},load:function(b){this.loadStartedCalled=false;this.source.setAttribute("src",b);this.element.load();this.mp3=b;g[h].events.trackLoadProgress(this)},
+loadError:function(){this.settings.loadError.apply(this)},init:function(){this.settings.init.apply(this)},loadStarted:function(){if(!this.element.duration)return false;this.duration=this.element.duration;this.updatePlayhead();this.settings.loadStarted.apply(this)},loadProgress:function(){if(this.element.buffered!=null&&this.element.buffered.length){if(!this.loadStartedCalled)this.loadStartedCalled=this.loadStarted();this.loadedPercent=this.element.buffered.end(this.element.buffered.length-1)/this.duration;
+this.settings.loadProgress.apply(this,[this.loadedPercent])}},playPause:function(){this.playing?this.pause():this.play()},play:function(){/(ipod|iphone|ipad)/i.test(navigator.userAgent)&&this.element.readyState==0&&this.init.apply(this);if(!this.settings.preload){this.settings.preload=true;this.element.setAttribute("preload","auto");g[h].events.trackLoadProgress(this)}this.playing=true;this.element.play();this.settings.play.apply(this)},pause:function(){this.playing=false;this.element.pause();this.settings.pause.apply(this)},
+setVolume:function(b){this.element.volume=b},trackEnded:function(){this.skipTo.apply(this,[0]);this.settings.loop||this.pause.apply(this);this.settings.trackEnded.apply(this)}};var j=function(b,a){var c=[];a=a||document;if(a.getElementsByClassName)c=a.getElementsByClassName(b);else{var d,e,i=a.getElementsByTagName("*"),f=RegExp("(^|\\s)"+b+"(\\s|$)");d=0;for(e=i.length;d<e;d++)f.test(i[d].className)&&c.push(i[d])}return c.length>1?c:c[0]}})("audiojs","audiojsInstance",this);
diff --git a/public/admin/js/audiojs/audiojs/audiojs.swf b/public/admin/js/audiojs/audiojs/audiojs.swf
new file mode 100644
index 0000000000000000000000000000000000000000..483599fc8273d0ccaaa935aa115a65e05ffe27ec
GIT binary patch
literal 1644
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

literal 0
HcmV?d00001

diff --git a/public/admin/js/audiojs/audiojs/player-graphics.gif b/public/admin/js/audiojs/audiojs/player-graphics.gif
new file mode 100644
index 0000000000000000000000000000000000000000..3e4d9d4e512052d49bfa872c37d14f2a1c211b69
GIT binary patch
literal 4499
zcmd6q`(ILN0>%%BfZz?pqPzqO-Zk@<*<^rP;6Y_j)6&c=n=&)gYRYs%K+DDo0Z~&s
zT9{dCsL{nsSyJmIOG`_gZqu}l+SF82O_!aSJ)lnQ?q~N$XU|{op6C63zR&Y6<t*`C
zu-Y2f0X*RX@RyLV#=*hC&(DuUBDGz(=<n};P*QLhf;>Gv2?WB9ojaLKrm?ZfsZ*+`
zsAvoZgTvvng+eZuJ9q9}D=RBHojx=?+|i+dpo+f!n=LJ^Ha0d<QBiC*=exW2u3Wv^
z)!nlxC1rTzyO!40qspq96DP@Ja?iEijT@6)TwG)_d3bnuYU&neXBUxJtW;K|rKK-f
zy42d*T2D`}xuvDHuHM+#_`tz~>07t1U7KiOVX=2_UPMIXikO(o9h%(SJy1pEvSrIc
zLqk`tT*+WC%gV|V6W1LoEiF_Og@y5+{5b04<D>o0|I3g12R<|?B0PY}3-WMt*F(UE
zq`pTP-~cY*qCK$ongGln0<KUb&dJ@gH*ep!`TGyR7bKO*;U6r{pN%LHfuOTtIRx4|
zmM9^+BZ?Zx!Jw`NYh>C8Qmp}8Lm@;|&nTX)2iUh2n<C4v^<KXLU(~IE!P|F+;A@d0
zjFrY^h4^~NRc-=d%>i>P<RW{3Z6R?<bP(H90GZ>&cA3bb%?7|Ou|Goc_=YJ~Bz;1q
zkyfu(0#GnFzVRg=9~J1{u;bS;>|zi}0-_g)N_&dj*L0V0nvV6B-m+HBrls5znUD<)
zZHlsqdWIklOF#P<Km?<4w89r-fS{UJwX7{8Z(NEuQXTlXb#)iIa!skvkc719Y#l24
zh+WVOM^R+$mkR@r2Qpk_D1z@pOYS+<_Z@r9oa4CrN%r|rX4Y&9;Gy9t9b(ROrluwG
z%kXafhKKl1Zs0<8-LK8?8t*Az*>vnQvE2iqp7r#8Z_GN%^oMOS6}I10*fgWd(uuTj
zgiZ+@_E80JZch>?hh?V7EDk(aK_P>teWPz{G$=c>3$C*O9zQ<~^aG-OIA|mvyKjq1
z$wCN<aO|{FUD&?q<9EBvM)a16G$9!;5Tb16V-H?}U^OTE=wpQ?h0(=8R1^KJ0ET)z
zvMzRR;>xs2=AA8&o6B%2scrymv8wD$Aqj9noD6->>{h8y4jL1j2);curnqns9QL(y
z4L|%%VjE_ArRKT)DhQ2Lxk!kPy=90U*IP^EqZ|m_?dR&aZM5Ga^6MobAH1Ipow0AT
z><Vm8>1r3@FV4bFf95dY#rPT;pwQtB5ER<DMF_l1OSTzgDTRPn*x!CRt7KJXf!#qF
zGkJXrb0o-2r+L|3;>{rXv25V-V6`!z2ua8mkc0`pIT`R&DMjp*KLPhmXYFcFimSrR
zaVL{$ZhCYySwSK@rF0>Xazhf&>HAv<fOj-hz3&ZwkUn$v+PZ<)m!(DlT`M-Jm>Wh?
zVxtV$k^5ZuxhI!bZC}4MuRYq%z(zU1nF9P;^lZma;^OxoycmqfVmUkVg5f;rx*nxp
zz=vxKJQMr$o|U`R`Pn92tDC>4eju$xpxB@O^nQq^>GWsqJ7;dWJH+%98oSH!uuX7W
z8^}*a0pY}jb|4MT+U+@d(L2(m2SmULmxYY)3!v`6&<BU?lT8;{7~1N;i(qILW`O9s
z#hQ-6YD+luS4<(vVC6!W<$HmOb>e+@(_p@vs&7CH5iaMgV2~0jx6W*jkC!wX%E|H3
z{EC`pItI)0;0aEM!!W@5dn4wxL*S~S*!AvbZVuwXwLV_k@MkWePo&c~X`B?!k!uS)
z#8y4(0A?_1#Dim6`JfNx8$8TC-}v{QTkI_(Et)&k2&C^<UStoin*IWI_KXmYdMu}I
z@x=qaUEWVe8aK=0vYPq5axUs0SGld{ULMY#ev2#Szz(;m<tkm}2yk(m83uN^F^DCI
zNbavD6mACn_SzM}sO+-`q;JhX%;)jHK<Q%^1(L@uRTIsMxqHuQg4)^hVzb5phCY23
zmQ77X2t;&q9MBmNFD>mY)rmzLe7|+R8ER7JD^#=z+IihuCuo=GXpe46d4qe)zXQ`Q
zu8VjB(;l2KoB`vW6w4W{6%IO?FIO6_>PULVI7rM^bBTee2B&pm(FXs@i(mH2?d6*d
z0-%><w*7am)9H5lVP?2ZVe{+p{uA~Mc3CgHlSYgZJGO^#>;#F(%oyv?)^pbx)6TnW
zF5I0(h<exJAYE@3Pn9M}$u)y;QizZX0q<OEcn%Mwf&$CV^!aQmQY^XZ>Il09l>?%G
znsm!3C!zAWnJ~f9#efG?R3QuQCQOzm9D62=c}1#Bhk2inKFeWVyGqIEKMER~R;;aH
zgd0G{KymP3((Sq;mD~9~!AAP=nTlkG44;i6qmbC)VCnHf#^A9|o5bVJ$CEI-<=Emx
zDTUQtt}V|ApH~+=qHW1L6{d%Z$c;8>TE{P=7Ie3k+#J0DgzR!#H0HH2d)Ak8BCI~D
zl?rMJw%zLd*`FVH!+T}g*9xAZ^e!uOuZO-Ug!y&@tZ9&c;i48N?hJj9c3J*WXaf<7
zvyhar9FalJ@2*txScjBI!e-fCG9?}eqg%TZUT(_UBalkuF;YFoMYbPH()eZa9=fq9
z(AnCM|F4{Xnl1mq`9o)l*6PtskAC@YoFh7keSdNeq)l-C5ctHohd0dmH?MS@;~fiL
z88#;<`yozAuDJR1==B=Rg*euVH$ROu+N;m}#hmL#qKN_?iSy)$$&u(Gdw`i7iB8fv
zpg9T~@=h5~Gy>jYvygoo9ujj)E>H#JJOma4>*d`UO%Rh~)9nh6y$3Yv&jaUy_DfPg
zA4qU=@2Z1G;-p@})|*5dJqjGVgfT>;3wjuNh+Mk9R2q{;c}UaKuLyn@cw>C9HWsm?
z>h!j*9P}FdkT%uYfSbYLt4zFxfH#}Ys!%ExycPqZn9<s`7@sV}y6$$G+AZMWCHumg
z!!Gz6nWny{()pze!}C#xJZt(T!QcK~JN%k;8)QMbt{>mf2AdqOpj6!U=i1$J<1gAR
zQ^R~J^Zz}}Pe~XeX&foQVf}p;$MfC1k{6r00HK_rgq8jSO}iD_QLtNNpSxwNwnDc|
zxW)ARWQ97MpRCXtZH2PSSD2r?4MxEgs@it98V~xAyz%Nw=wlm~x9p@d3LP8Svl>&X
zMyM=0GO>iVSCv*hG0e4Y*`V~T`uO<iEdU{-9zLC4G7ywo=DFK#w4sR=taST!iPebb
z)0-#f!zQmu<k%sT7vJw~`MP)dO)j;eC@D4y-+G7v6HaS}y%Hev25A7{GHezfxI?W8
z;=s&i>X;d@T5LEY_R*0(y_|vzR35Vv;RZ-&WR6qU3$M_vk~tp2hZNaC43Jt8$jGjk
zX`2R6`Zl2VTYbJ<3O`>iSErWCRXS#UsE6c~Kpf_%PU*#Le^PY{bn`L=$HsH$772%z
zHD^@7%O#Co)!`?bST5!owH`x}Y!(>3;FDOi1^RZfK*{#7SPsD^SF+~Dvi#Iz*N;={
zQ(Rl-{B-lk=9XiqieEhtZIPc9sQ$FRKD7^`NwG}5K4Gy;y*{0!+KmpsKFM(GyTpcX
zWB8Slb~-@og&~jAuo3?B5_hadia^<_DLImCBG6;kz)1bxMuxf!uXKj!bD!UapP?Fq
zd;cyL0YKM@Wh)>UJh}XQMw?D7tDJRW;j3d;StJWSEaNXZ5!av-%bim?u~44%U@Zlo
z#ByxOzFYU`I<X8VGJDH7J97h*(!UC>3UDryfAHGaS3mQc2qS`ba(^K(8fVA&byO>s
KA7@~Jh5rN$ej<?o

literal 0
HcmV?d00001

diff --git a/public/admin/js/audiojs/includes/index.css b/public/admin/js/audiojs/includes/index.css
new file mode 100644
index 00000000000..bb076576afc
--- /dev/null
+++ b/public/admin/js/audiojs/includes/index.css
@@ -0,0 +1,44 @@
+@font-face { font-family: 'bpreplay'; font-weight: normal; font-style: normal; src: url('BPreplay-webfont.eot'); src: local('☺'), url('BPreplay-webfont.woff') format('woff'), url('BPreplay-webfont.ttf') format('truetype'), url('BPreplay-webfont.svg#webfontgzJ8Cr4g') format('svg'); }
+
+body { width: 480px; margin: 40px auto; font-family: sans-serif; background: #f6f6f0; color: #330; line-height: 1.5; }
+header { display: block; background: url('./divider.png') bottom center no-repeat; padding: 10px 0px; margin: 0px 0px 20px; text-align: center; }
+h1 { display: inline; position: relative; margin: 0ex 0ex 1ex 1.2ex; font-family: 'bpreplay', sans-serif; font-weight: normal; font-size: 42px; letter-spacing: 1px; }
+h1:before { content: '♬ '; position: absolute; top: 16px; left: -3.1ex; color: #da1c5c; font-size: 0.4em; }
+
+.audiojs { margin: 15px 0px 20px; }
+h2 { font-family: 'bpreplay', sans-serif; font-weight: normal; font-size: 24px; margin-bottom: 10px; line-height: 1.6; }
+
+.download { background: url('./divider.png') top center no-repeat, url('./divider.png') bottom center no-repeat; margin: 30px 0px; padding: 20px 0px; text-align: center; }
+.download a em { color: rgba(255, 255, 255, 0.8); font-style: normal; }
+.download a em:before { content: '⬇ '; font-size: 0.8em; padding: 0px 4px 0px 0px; }
+.download a { display: block; color: #f6f6f0; width: 190px; margin: 5px auto; padding: 8px 2px 8px 0px; font-size: 14px; border-bottom: 0px; background: #da1c5c;
+  border-radius: 5px; -webkit-border-radius: 5px; -moz-border-radius: 5px; -o-border-radius: 5px;
+  background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0.2, #da1c5c), color-stop(0.5, #C81C5C), color-stop(0.51, #da1c5c), color-stop(1, #da1c5c));
+  background-image: -moz-linear-gradient(center top, #da1c5c 0%, #C81C5C 50%, #da1c5c 51%, #da1c5c 100%); }
+  
+h3 { margin: 30px 0px 8px; }
+
+p { font-size: 13px; margin: 0px 0px 5px; padding: 0px 0px 1px; }
+em { font-style: normal; color: #b1aca0; padding-right: 3px; }
+a { color: #330; text-decoration: none; border-bottom: 1px solid; }
+blockquote { font-size: 13px; font-style: italic; border-left: 2px solid #b1aca0; padding: 0px 0px 2px 10px; margin: 0px 0px 10px; }
+cite:before { content: "From:\00a0"; }
+cite { border-top: 1px dotted #999; padding-top: 0.5em; display: block; word-wrap: break-word; }
+
+ol { padding: 0px; font-size: 12px; color: #b1aca0; }
+ol li { margin: 0px 0px 20px; }
+ol li p { color: #330; }
+
+ul { font-size: 13px; margin: 6px 0px; padding: 0px; list-style: none; }
+ul li { position: relative; }
+ul li:before { content: '\2022'; position: absolute; left: -1.8ex; }
+
+pre,
+code { background: #EFEEE6; color: #552; font-family: Monaco, Consolas, 'Lucida Console', monospace; font-size: 12px; padding: 0px 3px; }
+pre { margin: 0px; padding: 4px 8px; }
+
+footer { display: block; margin: 30px 0px 0px; color: #b1aca0; text-align: center; margin: 50px 0px 0px; }
+footer a { color: #b1aca0; }
+footer em { display: block; font-size: 1.2em; font-style: normal; }
+footer .ab-c { font-family: 'Andale Mono', 'Courier New', courier; font-size: 0.75em; margin: 30px 0px 0px; padding: 20px 0px; border-top: 1px solid #EFEEE6; }
+footer .ab-c a { padding: 0px; }
\ No newline at end of file
diff --git a/public/admin/js/audiojs/index.html b/public/admin/js/audiojs/index.html
new file mode 100644
index 00000000000..fdad851f794
--- /dev/null
+++ b/public/admin/js/audiojs/index.html
@@ -0,0 +1,96 @@
+<!DOCTYPE html>
+<html lang="en">
+  <head>
+    <meta charset="utf-8">
+    <title>audio.js</title>
+    <script>var _gaq=[['_setAccount','UA-20257902-1'],['_trackPageview']];(function(d,t){ var g=d.createElement(t),s=d.getElementsByTagName(t)[0]; g.async=1;g.src='//www.google-analytics.com/ga.js';s.parentNode.insertBefore(g,s)}(document,'script'))</script>
+    <script src="./audiojs/audio.min.js"></script>
+    <link rel="stylesheet" href="./includes/index.css" media="screen">
+    <script>
+      audiojs.events.ready(function() {
+        audiojs.createAll();
+      });
+    </script>
+  </head>
+  <body>
+    <header>
+      <h1>audio.js</h1>
+    </header>
+
+    <audio src="http://s3.amazonaws.com/audiojs/02-juicy-r.mp3" preload="auto"></audio>
+
+    <h2>audio.js is a drop-in javascript library that allows HTML5’s &lt;audio&gt; tag to be used anywhere.</h2>
+
+    <p>It uses native &lt;audio&gt; where available and an invisible flash player to emulate &lt;audio&gt; for other browsers. It provides a consistent html player UI to all browsers which can be styled used standard css.</audio>
+
+    <div class="download">
+      <a href="http://kolber.github.com/audiojs/audiojs.zip" class="button"><em>Download</em> audio.js</a>
+    </div>
+
+    <h3>Installation</h3>
+    <ol>
+      <li>
+        <p>Put <code>audio.js</code>, <code>player-graphics.gif</code> & <code>audiojs.swf</code> in the same folder.</p>
+      </li>
+      <li>
+        <p>Include the <code>audio.js</code> file:</p>
+        <pre><code>&lt;script src="/audiojs/audio.min.js"&gt;&lt;/script&gt;</code></pre>
+      </li>
+      <li>
+        <p>Initialise audio.js:</p>
+        <pre><code>&lt;script&gt;
+  audiojs.events.ready(function() {
+    var as = audiojs.createAll();
+  });
+&lt;/script&gt;</code></pre>
+      </li>
+      <li>
+        <p>Then you can use <code>&lt;audio&gt;</code> wherever you like in your HTML:</p>
+        <pre><code>&lt;audio src="/mp3/juicy.mp3" preload="auto" /&gt;</code></pre>
+      </li>
+    </ol>
+
+    <h3>Examples</h3>
+    <p>A series of API tests & examples for using and extending audio.js</p>
+    <p><em>Example 1</em> <a href="http://kolber.github.com/audiojs/demos/test1.html">Test multiple load types</a></p>
+    <p><em>Example 2</em> <a href="http://kolber.github.com/audiojs/demos/test2.html">Custom markup/css</a></p>
+    <p><em>Example 3</em> <a href="http://kolber.github.com/audiojs/demos/test3.html">Multiple players, testing <code>preload</code>, <code>loop</code> & <code>autoplay</code> attributes</a></p>
+    <p><em>Example 4</em> <a href="http://kolber.github.com/audiojs/demos/test5.html">Customised player</a></p>
+    <p><em>Example 5</em> <a href="http://kolber.github.com/audiojs/demos/test6.html">Customised playlist player</a></p>
+    <!--<p><em>Example 6</em> <a href="http://kolber.github.com/audiojs/demos/test4.html">.ogg fallback</a></p>-->
+
+    <h3>Browser & format support</h3>
+    <p>With Flash as a fallback, it should work pretty much anywhere.<br>
+      It has been verified to work across:</p>
+    <ul>
+      <li>Mobile Safari <em>(iOS 3+)</em></li>
+      <li>Android <em>(2.2+, w/Flash)</em></li>
+      <li>Safari <em>(4+)</em></li>
+      <li>Chrome <em>(7+)</em></li>
+      <li>Firefox <em>(3+, w/ Flash)</em></li>
+      <li>Opera <em>(10+, w/ Flash)</em></li>
+      <li>IE <em>(6, 7, 8, w/ Flash)</em></li>
+    </ul>
+    <p><strong>ogg</strong></p>
+    <p>audio.js focuses on playing mp3s. It doesn’t currently support the ogg format. As mp3 is the current defacto music transfer format, ogg support is lower on our list of priorities.</p>
+
+    <h3>Flash local security</h3>
+    <blockquote>
+      <p><strong>Note:</strong> For local content running in a browser, calls to the <code>ExternalInterface.addCallback()</code> method work only if the SWF file and the containing web page are in the local-trusted security sandbox.</p>
+      <cite><a href="http://help.adobe.com/en_US/FlashPlatform/reference/actionscript/3/flash/external/ExternalInterface.html#addCallback()">http://help.adobe.com/en_US/FlashPlatform/reference/actionscript/3/flash/external/ExternalInterface.html#addCallback()</a></cite>
+    </blockquote>
+    <p>This means that unless you have gone through the rigmarole of setting up your <a href="http://kb2.adobe.com/cps/093/4c093f20.html#main_blocked">flash player security settings for local files</a>, <code>ExternalInterface</code> calls will only work when the page is loaded from a ‘domain’. <code>http://localhost</code> counts, but any <code>file://</code> requests don’t.</p>
+
+    <h3>Source code</h3>
+    <p>All efforts have been made to keep the source as clean and readable as possible. Until we release more detailed documentation, the annotated source is the best reference for usage.</p>
+    <p><a href="http://kolber.github.com/audiojs/docs/">Annotated source</a> / <a href="http://github.com/kolber/audiojs">Source on Github</a></p>
+
+    <h3>License</h3>
+    <p>audio.js is released under an <a href="http://www.opensource.org/licenses/mit-license.php">MIT License</a>, so do with it what you will.</p>
+    <footer>
+      <p>All example audio files are from <a href="http://waitwhatmusic.com/">wait what</a>’s <a href="http://soundcloud.com/wait-what/sets/the-notorious-xx">notorious xx album</a> and used with permission.</p>
+      <p>This site is ©copyright <a href="http://aestheticallyloyal.com">Anthony Kolber</a>, 2010.</p>
+      <p class="ab-c"><em>&real;</em> Another <a href="http://ab-c.com.au">ab+c</a> joint</p>
+    </footer>
+  </body>
+</html>
\ No newline at end of file
diff --git a/tests/library/ZendAfi/View/Helper/AudioJSPlayerTest.php b/tests/library/ZendAfi/View/Helper/AudioJSPlayerTest.php
new file mode 100644
index 00000000000..955be89c650
--- /dev/null
+++ b/tests/library/ZendAfi/View/Helper/AudioJSPlayerTest.php
@@ -0,0 +1,48 @@
+<?php
+/**
+ * Copyright (c) 2012, Agence Française Informatique (AFI). All rights reserved.
+ *
+ * AFI-OPAC 2.0 is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by
+ * the Free Software Foundation.
+ *
+ * There are special exceptions to the terms and conditions of the AGPL as it
+ * is applied to this software (see README file).
+ *
+ * AFI-OPAC 2.0 is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
+ *
+ * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE
+ * along with AFI-OPAC 2.0; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA 
+ */
+
+class ZendAfi_View_Helper_AudioJsPlayerTest extends PHPUnit_Framework_TestCase {
+	/**
+	 * @var ZendAfi_View_Helper_AudioJsPlayer
+	 */
+	private $_helper;
+
+	protected function setUp() {
+		Class_ScriptLoader::resetInstance();
+		$this->_helper = new ZendAfi_View_Helper_AudioJsPlayer();
+		$this->_helper->setView(new Zend_View());
+		$this->_helper->audioJsPlayer();
+	}
+
+	/** @test */
+	public function scriptLoaderShouldContainsAudioJS() {
+		$this->assertContains('<script src="'.BASE_URL.'/public/admin/js/audiojs/audiojs/audiomin.js?', 
+													Class_ScriptLoader::getInstance()->html());
+	}
+
+
+	/** @test */
+	public function audioJSShouldBeCreated() {
+		$this->assertContains('audiojs.createAll()', 		
+	}
+}
+
+?>
\ No newline at end of file
-- 
GitLab