Use RecordRTC or Wami when adding audio to lp item - refs BT#11060

ofaj
Angel Fernando Quiroz Campos 9 years ago
parent 5c6c9e4f87
commit 810f8b5dcc
  1. 6
      app/Resources/public/css/base.css
  2. 4
      main/inc/ajax/lp.ajax.php
  3. 13
      main/inc/lib/display.lib.php
  4. 12
      main/inc/lib/wami-recorder/gui.js
  5. 12
      main/inc/lib/wami-recorder/record_document.php
  6. 4
      main/inc/lib/wami-recorder/recorder.js
  7. 14
      main/newscorm/lp_add_audio.php
  8. 360
      main/template/default/learnpath/record_voice.tpl

@ -5684,6 +5684,12 @@ a.sessionView {
.modal-body .sectioncomment{
color: #666;
}
.wami-container {
height: 144px;
position: relative;
}
/* CSS NEW TOP ******************************************************************************/
/* CSS Responsive */
@media (min-width: 1025px) and (max-width: 1200px) {

@ -147,11 +147,9 @@ switch ($action) {
$file['name'] = 'rec_'.date('Y-m-d_His').'_'.uniqid().'.'.$fileInfo['extension'];
$file['file'] = $file;
$lpPathInfo['dir'] = api_remove_trailing_slash($lpPathInfo['dir']);
$result = DocumentManager::upload_document(
$file,
$lpPathInfo['dir'],
'/audio',
$file['name'],
null,
0,

@ -2005,12 +2005,6 @@ class Display
$fileInfo = pathinfo($file);
switch ($fileInfo['extension']) {
case 'wav':
if (isset($params['url'])) {
$url = DocumentManager::generateAudioTempFile(basename($file), $file);
return DocumentManager::readNanogongFile($url);
}
break;
case 'mp3':
case 'webm':
$autoplay = null;
@ -2028,6 +2022,13 @@ class Display
</object>';
$html .= '</audio>';
return $html;
break;
case 'wav':
//no break;
case 'ogg':
$html = '<audio src="' . $params['url'] . '" controls class="skip"></audio>';
return $html;
break;
}

@ -34,13 +34,14 @@ Wami.GUI = function(options) {
}
function setupDOM() {
var guidiv = createDiv(null,
"position: absolute; width: 214px; height: 137px;");
var guidiv = createDiv(
null,
'position:absolute;width:214px;height:144px;padding-left:79px;padding-top:45px;left:50%;margin-left: -107px'
);
document.getElementById(options.id).appendChild(guidiv);
var rid = Wami.createID();
var recordDiv = createDiv(rid,
"position: absolute; left: 40px; top: 25px");
var recordDiv = createDiv(rid, '');
guidiv.appendChild(recordDiv);
recordButton = new Button(rid, RECORD_BUTTON, options.buttonUrl);
@ -51,8 +52,7 @@ Wami.GUI = function(options) {
//Chamilo hack single button
var pid = Wami.createID();
var playDiv = createDiv(pid,
"position: absolute; right: 40px; top: 25px");
var playDiv = createDiv(pid, '');
guidiv.appendChild(playDiv);
if (!options.singleButton) {
playButton = new Button(pid, PLAY_BUTTON, options.buttonUrl);

@ -56,7 +56,6 @@ if (!is_dir($saveDir)) {
//avoid duplicates
$waminame_to_save = $waminame;
$title_to_save = str_replace('_', ' ', $waminame);
$waminame_noex = basename($waminame, ".wav");
if (file_exists($saveDir.'/'.$waminame_noex.'.'.$ext)) {
$i = 1;
@ -64,8 +63,6 @@ if (file_exists($saveDir.'/'.$waminame_noex.'.'.$ext)) {
$i++;
}
$waminame_to_save = $waminame_noex.'_'.$i.'.'.$ext;
$title_to_save = $waminame_noex.'_'.$i.'.'.$ext;
$title_to_save = str_replace('_', ' ', $title_to_save);
}
$documentPath = $saveDir.'/'.$waminame_to_save;
@ -83,11 +80,16 @@ $file = array(
'name' => $fileInfo['basename'],
'tmp_name' => $documentPath,
'size' => filesize($documentPath),
'type' => 'audio/wav',
'from_file' => true
)
);
$output = true;
ob_start();
// Strangely the file path changes with a double extension
copy($documentPath, $documentPath . '.wav');
$documentData = DocumentManager::upload_document(
$file,
$wamidir,
@ -130,6 +132,10 @@ if (!empty($documentData)) {
);
}
}
// Strangely the file path changes with a double extension
// Remove file with one extension
unlink($documentPath);
} else {
Display::addFlash($contents);
}

@ -127,6 +127,8 @@ Wami.setup = function(options) {
var container = document.createElement('div');
container.style.position = 'absolute';
container.style.marginLeft = '-107px';
container.style.left = '50%';
_options.cid = Wami.createID();
container.setAttribute('id', _options.cid);
@ -240,7 +242,7 @@ Wami.setup = function(options) {
var augmentedfn = Wami.nameCallback(function() {
checkRemembered(finishedfn);
container.style.cssText = "position: absolute;";
container.style.cssText = "position: absolute; left:50%; margin-left:-107px";
});
container.style.cssText = "position: absolute; z-index: 99999";

@ -93,8 +93,8 @@ if (isset($lp_item->audio) && !empty($lp_item->audio)) {
$urlFile = api_get_path(WEB_COURSE_PATH).$courseInfo['path'].'/document/audio/'.$lp_item->audio.'?'.api_get_cidreq();
if (!file_exists($file)) {
$file = api_get_path(SYS_COURSE_PATH).$courseInfo['path'].'/document'.$lpPathInfo['dir'].$lp_item->audio;
$urlFile = api_get_path(WEB_COURSE_PATH).$courseInfo['path'].'/document'.$lpPathInfo['dir'].$lp_item->audio.'?'.api_get_cidreq();
$file = api_get_path(SYS_COURSE_PATH).$courseInfo['path'].'/document'.$lpPathInfo['dir'] . '/' . $lp_item->audio;
$urlFile = api_get_path(WEB_COURSE_PATH).$courseInfo['path'].'/document'.$lpPathInfo['dir'] . '/' . $lp_item->audio.'?'.api_get_cidreq();
}
}
@ -110,20 +110,24 @@ $recordVoiceForm = Display::page_subheader(get_lang('RecordYourVoice'));
$page .= '<div id="doc_form" class="col-md-8">';
$htmlHeadXtra[] = '<script src="' . api_get_path(WEB_LIBRARY_JS_PATH) . 'rtc/RecordRTC.js"></script>';
$htmlHeadXtra[] = '<script src="' . api_get_path(WEB_LIBRARY_PATH) . 'wami-recorder/recorder.js"></script>';
$htmlHeadXtra[] = '<script src="' . api_get_path(WEB_LIBRARY_PATH) . 'wami-recorder/gui.js"></script>';
$htmlHeadXtra[] = '<script type="text/javascript" src="' . api_get_path(WEB_LIBRARY_PATH) . 'swfobject/swfobject.js"></script>';
$tpl = new Template(null);
$tpl->assign('unique_file_id', api_get_unique_id());
$tpl->assign('course_code', api_get_course_id());
$tpl->assign('php_session_id', session_id());
$tpl->assign('filename', $lp_item->get_title().'_nano.wav');
$tpl->assign('enable_nanogong', api_get_setting('enable_nanogong') == 'true' ? 1 : 0);
$tpl->assign('enable_wami', api_get_setting('enable_wami_record') == 'true' ? 1 : 0);
$tpl->assign('enable_record_audio', api_get_setting('enable_record_audio') === 'true');
$tpl->assign('cur_dir_path', '/audio');
$tpl->assign('lp_item_id', $lp_item_id);
$tpl->assign('lp_dir', api_remove_trailing_slash($lpPathInfo['dir']));
$recordVoiceForm .= $tpl->fetch('default/learnpath/record_voice.tpl');
$form->addElement('header', get_lang('Or'));
$form->addElement('header', get_lang('AudioFile'));
$form->addElement('html', sprintf(get_lang('AudioFileForItemX'), $lp_item->get_title()));
$form->addLabel(null, sprintf(get_lang('AudioFileForItemX'), $lp_item->get_title()));
if (!empty($file)) {
$audioPlayer = '<div id="preview">'.

@ -1,263 +1,133 @@
{# <script type="text/javascript" src="{{ _p.web_lib }}javascript/rtc/RecordRTC.js"></script> #}
<script type="text/javascript" src="{{ _p.web_lib }}swfobject/swfobject.js"></script>
<script type="text/javascript" src="{{ _p.web_lib }}wami-recorder/recorder.js"></script>
<script type="text/javascript" src="{{ _p.web_lib }}wami-recorder/gui.js"></script>
<div id="rtc">
<audio class="skip" id="audio" autoplay="" loop="" controls=""></audio>
<span id="progress-info"></span>
<br />
<a id="record" class="btn btn-danger" >{{ 'RecordAudio' | get_lang }}</a>
<button id="stop" class="btn btn-default" disabled="">{{ 'StopRecordingAudio' | get_lang }}</button>
</div>
<div id="wami" style="float:left; z-index:6000">
<a id="record-wami" class="btn btn-info">{{ 'ActivateAudioRecorder' | get_lang }}</a>
<br />
<div id="wami-recorder">
</div>
<div id="start-recording" class="alert" style="display:none">
{{ "StartSpeaking" | get_lang }}
</div>
</div>
<div id="nanogong">
<applet id="nanogongApplet" archive="{{ _p.web_lib }}nanogong/nanogong.jar" code="gong.NanoGong" width="250" height="95">
<param name="ShowTime" value="true" />
<param name="AudioFormat" value="ImaADPCM" />
<param name="ShowSpeedButton" value="false" />
</applet>
<form name="form_nanogong">
<input id="status" type="hidden" name="status" value="0">
<button class="upload" type="submit" value="{{ 'Send' | get_lang }}" onclick="submitVoice()">
{{ 'Send' | get_lang }}
</button>
</form>
<span id="nanogong_result" ></span>
<div id="record-audio-recordrtc" class="text-center">
<p>
<span class="fa fa-microphone fa-5x fa-fw" aria-hidden="true"></span>
<span class="sr-only">{{ 'RecordAudio'|get_lang }}</span>
</p>
<button class="btn btn-primary" type="button" id="btn-start-record">
<span class="fa fa-circle fa-fw" aria-hidden="true"></span> {{ 'StartRecordingAudio'|get_lang }}
</button>
<button class="btn btn-success" type="button" id="btn-stop-record" disabled>
<span class="fa fa-square fa-fw" aria-hidden="true"></span> {{ 'StopRecordingAudio'|get_lang }}
</button>
</div>
<span id="record-result"></span>
<div class="clear"></div>
<div id="record-audio-wami" class="wami-container"></div>
<script>
$(document).on('ready', function () {
function useRecordRTC(){
$('#record-audio-recordrtc').show();
var mediaConstraints = {audio: true},
recordRTC = null,
btnStart = $('#btn-start-record'),
btnStop = $('#btn-stop-record');
btnStart.on('click', function () {
navigator.getUserMedia = navigator.getUserMedia ||
navigator.mozGetUserMedia ||
navigator.webkitGetUserMedia;
if (navigator.getUserMedia) {
navigator.getUserMedia(mediaConstraints, successCallback, errorCallback);
} else if (navigator.mediaDevices.getUserMedia) {
navigator.mediaDevices.getUserMedia(mediaConstraints)
.then(successCallback).error(errorCallback);
}
function successCallback(stream) {
recordRTC = RecordRTC(stream, {
numberOfAudioChannels: 1,
type: 'audio'
});
recordRTC.startRecording();
btnStop.prop('disabled', false);
btnStart.prop('disabled', true);
}
function errorCallback(error) {
alert(error.message);
}
});
function checkNanoGongEnable() {
var nanogongEnabled = {{ enable_nanogong }};
if (nanogongEnabled != 1) {
return false;
}
return recorder = document.getElementById("nanogongApplet");
}
function checkWamiEnable() {
var wamiEnabled = {{ enable_wami }};
if (wamiEnabled != 1) {
return false;
}
if (swfobject.hasFlashPlayerVersion("1")) {
return true;
} else {
return false;
}
}
function submitVoice() {
// lang vars
var lang_no_applet = "{{ 'NanogongNoApplet' | get_lang }}";
var lang_record_before_save = "{{ 'NanogongRecordBeforeSave' | get_lang }}";
var lang_give_a_title = "{{ 'NanogongGiveTitle' | get_lang }}";
var lang_failled_to_submit = "{{ 'NanogongFailledToSubmit' | get_lang }}";
var lang_submitted = "{{'NanogongSubmitted' | get_lang }}";
// user and group id
//path, url and filename
var filename = "{{ filename }}"; //adding name file, tag and extension
var filename = encodeURIComponent(filename);
var fileInput = 'file';
var urlNanoGong = "{{ _p.web_lib }}nanogong/upload_nanogong_file.php?lp_item_id={{ lp_item_id }}&filename="+filename+"&file_field="+fileInput+"&cidReq={{ course_code }}&path={{ cur_dir_path }}";
var cookie= "ch_sid=" + "{{ php_session_id }}";
// Check
var recorder;
if (!(recorder = document.getElementById("nanogongApplet"))) {
alert(lang_no_applet);
return
}
var duration = parseInt(recorder.sendGongRequest("GetMediaDuration", "audio")) || 0
if (duration <= 0) {
alert(lang_record_before_save);
return;
}
var ret = recorder.sendGongRequest("PostToForm", urlNanoGong, fileInput, cookie, filename);
if (ret == null) {
alert(lang_failled_to_submit);
} else {
alert(lang_submitted);
$("#status").attr('value', '1');
}
}
function setupRecorder() {
$('#wami-recorder').css('margin-bottom', '150px');
Wami.setup({
id : "wami",
onReady : setupGUI,
swfUrl : "{{ _p.web_lib }}wami-recorder/Wami.swf"
});
}
function setupGUI() {
var uniq = 'rec_' + (new Date()).getTime() + ".wav";
var gui = new Wami.GUI({
id : "wami-recorder",
singleButton : true,
recordUrl : "{{ _p.web_lib }}wami-recorder/record_document.php?lp_item_id={{ lp_item_id }}&waminame="+uniq+"&wamidir={{ cur_dir_path }}&wamiuserid={{ _u.user_id }}",
buttonUrl : "{{ _p.web_lib }}wami-recorder/buttons.png",
buttonNoUrl : "{{ _p.web_img }}blank.gif",
onRecordStart : function() {
$('#start-recording').show();
},
onRecordFinish: function() {
$('#start-recording').hide();
window.location.reload();
},
onError : function() {
btnStop.on('click', function () {
if (!recordRTC) {
return;
}
recordRTC.stopRecording(function (audioURL) {
var recordedBlob = recordRTC.getBlob(),
fileName = Math.round(Math.random() * 99999999) + 99999999,
fileExtension = '.' + recordedBlob.type.split('/')[1];
var formData = new FormData();
formData.append('audio-filename', fileName + fileExtension);
formData.append('audio-blob', recordedBlob, 'audio' + fileExtension);
$.ajax({
url: '{{ _p.web_ajax }}lp.ajax.php?a=record_audio&lp_item_id={{ lp_item_id }}',
data: formData,
processData: false,
contentType: false,
type: 'POST',
success: function (fileURL) {
if (!fileURL) {
return;
}
window.location.reload();
}
});
btnStop.prop('disabled', true);
btnStart.prop('disabled', false);
});
});
}
});
gui.setPlayEnabled(true);
}
var rtcEnabled = false;
$(document).ready(function() {
var isChrome = navigator.webkitGetUserMedia;
$('#rtc').hide();
$('#wami').hide();
$('#nanogong').hide();
if (rtcEnabled && isChrome) {
$('#rtc').show();
} else if (checkNanoGongEnable()) {
$('#nanogong').show();
} else if (checkWamiEnable()) {
$('#wami').show();
var recordWami = $('#record-wami');
recordWami.on('click', function() {
setupRecorder();
return false;
});
} else {
$('#record-result').html("{{ 'RecordIsNotAvailable' | get_lang }}");
}
// Start web RTC code
var format = 'webm'; // or wav
var record = $('#record');
var stop = document.getElementById('stop');
var preview = document.getElementById('audio');
var progressInfo = document.getElementById('progress-info');
var previewBlock = document.getElementById('preview');
function postBlob(blob, fileType, fileName) {
// FormData
var formData = new FormData();
formData.append(fileType + '-filename', fileName);
formData.append(fileType + '-blob', blob);
// progress-bar
var hr = document.createElement('hr');
progressInfo.appendChild(hr);
var strong = document.createElement('strong');
strong.innerHTML = fileType + ' upload progress: ';
progressInfo.appendChild(strong);
var progress = document.createElement('progress');
progressInfo.appendChild(progress);
// POST the Blob
$.ajax({
url:'{{ _p.web_ajax }}lp.ajax.php?a=record_audio&lp_item_id={{ lp_item_id }}',
data: formData,
processData: false,
contentType: false,
type: 'POST',
success:function(fileURL) {
window.location.reload();
progressInfo.appendChild(document.createElement('hr'));
var mediaElement = document.createElement(fileType);
mediaElement.src = fileURL;
mediaElement.controls = true;
var uniq = 'id' + (new Date()).getTime();
mediaElement.id = uniq;
$(previewBlock).html('');
function useWami(){
$('#record-audio-wami').show();
function setupGUI() {
var gui = new Wami.GUI({
id : 'record-audio-wami',
singleButton : true,
recordUrl : '{{ _p.web_lib }}wami-recorder/record_document.php?' + $.param({
waminame: 'rec_' + (new Date()).getTime() + '.wav',
wamidir: '{{ cur_dir_path }}',
wamiuserid: {{ _u.user_id }},
lp_item_id: {{ lp_item_id }}
}),
buttonUrl : '{{ _p.web_lib }}wami-recorder/buttons.png',
buttonNoUrl: '{{ _p.web_img }}blank.gif',
onRecordFinish: function() {
$('#start-recording').hide();
window.location.reload();
}
});
previewBlock.appendChild(mediaElement);
mediaElement.play();
$(progressInfo).html('');
window.location.reload();
gui.setPlayEnabled(false);
}
});
}
var recordAudio, recordVideo;
record.on('click', function() {
record.disabled = true;
var myURL = (window.URL || window.webkitURL || {});
if (navigator.getUserMedia) {
navigator.getUserMedia({
audio: true,
video: false
},
function(stream) {
if (window.IsChrome) stream = new window.MediaStream(stream.getAudioTracks());
preview.src = myURL.createObjectURL(stream);
preview.play();
recordAudio = RecordRTC(stream, {
type: 'audio'
});
recordAudio.startRecording();
/*recordVideo = RecordRTC(stream, {
type: 'video'
});*/
//recordVideo.startRecording();
stop.disabled = false;
},
function(err) {
console.log("The following error occured: " + err);
return false;
Wami.setup({
id : "record-audio-wami",
onReady : setupGUI,
swfUrl: '{{ _p.web_lib }}wami-recorder/Wami.swf'
});
}
});
stop.onclick = function() {
record.disabled = false;
stop.disabled = true;
var fileName = Math.round(Math.random() * 99999999) + 99999999;
recordAudio.stopRecording();
$('#record-audio-recordrtc, #record-audio-wami').hide();
postBlob(recordAudio.getBlob(), 'audio', fileName + '.' + format);
var webRTCIsEnabled = navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.getUserMedia ||
navigator.mediaDevices.getUserMedia;
//recordVideo.stopRecording();
//PostBlob(recordVideo.getBlob(), 'video', fileName + '.webm');
preview.src = '';
};
if (webRTCIsEnabled) {
useRecordRTC();
// End web RTC code
});
return;
}
useWami();
});
</script>

Loading…
Cancel
Save