@digabi/exam-engine-core 23.9.3-alpha.0 → 23.10.1-alpha.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/__tests__/tsconfig.tsbuildinfo +1 -1
- package/dist/components/exam/AudioAnswer.d.ts.map +1 -1
- package/dist/components/exam/AudioAnswer.js +4 -49
- package/dist/components/exam/AudioAnswer.js.map +1 -1
- package/dist/components/exam/internal/AudioRecorder.d.ts +11 -0
- package/dist/components/exam/internal/AudioRecorder.d.ts.map +1 -0
- package/dist/components/exam/internal/AudioRecorder.js +115 -0
- package/dist/components/exam/internal/AudioRecorder.js.map +1 -0
- package/dist/components/exam/internal/MyAudioRecorder.d.ts +10 -0
- package/dist/components/exam/internal/MyAudioRecorder.d.ts.map +1 -0
- package/dist/components/exam/internal/MyAudioRecorder.js +106 -0
- package/dist/components/exam/internal/MyAudioRecorder.js.map +1 -0
- package/dist/components/exam/internal/MyMediaRecorder.d.ts +10 -0
- package/dist/components/exam/internal/MyMediaRecorder.d.ts.map +1 -0
- package/dist/components/exam/internal/MyMediaRecorder.js +91 -0
- package/dist/components/exam/internal/MyMediaRecorder.js.map +1 -0
- package/dist/components/shared/Audio.js +2 -2
- package/dist/components/shared/Audio.js.map +1 -1
- package/dist/components/shared/AudioTest.js +2 -2
- package/dist/components/shared/AudioTest.js.map +1 -1
- package/dist/components/shared/internal/AudioError.d.ts +8 -0
- package/dist/components/shared/internal/AudioError.d.ts.map +1 -0
- package/dist/components/shared/internal/AudioError.js +17 -0
- package/dist/components/shared/internal/AudioError.js.map +1 -0
- package/dist/components/shared/internal/AudioErrorRemoveMe.d.ts +8 -0
- package/dist/components/shared/internal/AudioErrorRemoveMe.d.ts.map +1 -0
- package/dist/components/shared/internal/AudioErrorRemoveMe.js +17 -0
- package/dist/components/shared/internal/AudioErrorRemoveMe.js.map +1 -0
- package/dist/i18n/fi-FI.d.ts +2 -0
- package/dist/i18n/fi-FI.d.ts.map +1 -1
- package/dist/i18n/fi-FI.js +2 -0
- package/dist/i18n/fi-FI.js.map +1 -1
- package/dist/i18n/sv-FI.d.ts.map +1 -1
- package/dist/i18n/sv-FI.js +2 -0
- package/dist/i18n/sv-FI.js.map +1 -1
- package/dist/main-bundle.js +18 -18
- package/dist/main-bundle.js.LICENSE.txt +0 -7
- package/dist/main.css +1 -1
- package/dist/store/audio/actions.d.ts +3 -3
- package/dist/store/audio/actions.d.ts.map +1 -1
- package/dist/store/audio/actions.js.map +1 -1
- package/dist/store/audio/reducer.d.ts +2 -2
- package/dist/store/audio/reducer.d.ts.map +1 -1
- package/dist/store/index.d.ts +1 -1
- package/dist/store/selectors.d.ts +2 -2
- package/dist/store/selectors.d.ts.map +1 -1
- package/dist/store/selectors.js.map +1 -1
- package/dist/types/ExamServerAPI.d.ts +7 -3
- package/dist/types/ExamServerAPI.d.ts.map +1 -1
- package/package.json +2 -3
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"AudioAnswer.d.ts","sourceRoot":"","sources":["../../../src/components/exam/AudioAnswer.tsx"],"names":[],"mappings":"AAAA,OAAO,
|
1
|
+
{"version":3,"file":"AudioAnswer.d.ts","sourceRoot":"","sources":["../../../src/components/exam/AudioAnswer.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAqB,MAAM,OAAO,CAAA;AACzC,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAA;AAMjE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAA;AAGzD,iBAAS,WAAW,CAAC,gBAAgB,EAAE,kBAAkB,qBAoCxD;;AAED,wBAAsC"}
|
@@ -1,54 +1,9 @@
|
|
1
|
-
import React, { useContext
|
1
|
+
import React, { useContext } from 'react';
|
2
2
|
import { getNumericAttribute } from '../../dom-utils';
|
3
3
|
import { saveAnswer } from '../../store/answers/actions';
|
4
4
|
import { useDispatch, useSelector } from 'react-redux';
|
5
|
-
import { ReactMediaRecorder } from 'react-media-recorder';
|
6
5
|
import { ExamContext } from '../context/ExamContext';
|
7
|
-
import {
|
8
|
-
import { faMicrophone, faStop } from '@fortawesome/free-solid-svg-icons';
|
9
|
-
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
10
|
-
function AudioAnswerRecorder({ onSave, onDelete, bitsPerSecond, audioUrl }) {
|
11
|
-
const { t } = useExamTranslation();
|
12
|
-
const timer = useRef();
|
13
|
-
const [timeElapsed, setTimeElapsed] = useState(0);
|
14
|
-
function startTimer() {
|
15
|
-
const timerId = setInterval(() => {
|
16
|
-
var _a, _b;
|
17
|
-
const timeElapsedMs = new Date().getTime() - ((_b = (_a = timer === null || timer === void 0 ? void 0 : timer.current) === null || _a === void 0 ? void 0 : _a.startTime) !== null && _b !== void 0 ? _b : new Date()).getTime();
|
18
|
-
setTimeElapsed(Math.floor(timeElapsedMs / 1000));
|
19
|
-
}, 1000);
|
20
|
-
timer.current = { id: timerId, startTime: new Date() };
|
21
|
-
}
|
22
|
-
function saveAndStopTimer(blob) {
|
23
|
-
var _a;
|
24
|
-
clearInterval((_a = timer.current) === null || _a === void 0 ? void 0 : _a.id);
|
25
|
-
setTimeElapsed(0);
|
26
|
-
onSave(blob);
|
27
|
-
}
|
28
|
-
function renderTimeElapsed() {
|
29
|
-
const minutes = Math.floor(timeElapsed / 60);
|
30
|
-
const sec = timeElapsed - minutes * 60;
|
31
|
-
const seconds = `00${sec}`.slice(-2);
|
32
|
-
return `${minutes}:${seconds}`;
|
33
|
-
}
|
34
|
-
return (React.createElement(ReactMediaRecorder, { audio: true, mediaRecorderOptions: { audioBitsPerSecond: bitsPerSecond !== null && bitsPerSecond !== void 0 ? bitsPerSecond : 65536 }, blobPropertyBag: { type: 'audio/mpeg' }, onStart: startTimer, onStop: (_blobUrl, blob) => saveAndStopTimer(blob), render: ({ status, error, startRecording, stopRecording }) => (React.createElement(React.Fragment, null,
|
35
|
-
React.createElement("div", null,
|
36
|
-
error && React.createElement("p", null,
|
37
|
-
"error: ",
|
38
|
-
error),
|
39
|
-
React.createElement("p", null,
|
40
|
-
status != 'recording' && !audioUrl && (React.createElement("button", { className: "e-button", onClick: startRecording },
|
41
|
-
React.createElement(FontAwesomeIcon, { size: "sm", icon: faMicrophone, fixedWidth: true }),
|
42
|
-
t('start.recording'))),
|
43
|
-
status == 'recording' && (React.createElement(React.Fragment, null,
|
44
|
-
React.createElement("button", { className: "e-button", onClick: stopRecording },
|
45
|
-
React.createElement(FontAwesomeIcon, { size: "sm", icon: faStop, fixedWidth: true }),
|
46
|
-
t('stop.recording')),
|
47
|
-
React.createElement("span", { className: "time-elapsed" }, renderTimeElapsed()))))),
|
48
|
-
audioUrl && (React.createElement("div", { className: "audio-answer-controls" },
|
49
|
-
React.createElement("audio", { src: audioUrl, className: "e-column e-column--narrow", preload: "metadata", controls: true, controlsList: "nodownload" }),
|
50
|
-
React.createElement("button", { className: "e-button-reversed delete-recording", onClick: onDelete }, t('remove.recording')))))) }));
|
51
|
-
}
|
6
|
+
import { AudioRecorder } from './internal/AudioRecorder';
|
52
7
|
function AudioAnswer(audioAnswerProps) {
|
53
8
|
const { element } = audioAnswerProps;
|
54
9
|
const questionId = getNumericAttribute(element, 'question-id');
|
@@ -58,7 +13,7 @@ function AudioAnswer(audioAnswerProps) {
|
|
58
13
|
const { examServerApi } = useContext(ExamContext);
|
59
14
|
return (React.createElement("div", { className: "audio-answer" },
|
60
15
|
React.createElement("span", { className: "anchor", id: `question-nr-${displayNumber}` }),
|
61
|
-
React.createElement(
|
16
|
+
React.createElement(AudioRecorder, { audioUrl: (answer === null || answer === void 0 ? void 0 : answer.value) === '' ? undefined : answer === null || answer === void 0 ? void 0 : answer.value, onSave: audio => {
|
62
17
|
void (async function () {
|
63
18
|
const audioAttachmentUrl = await examServerApi.saveAudio(questionId, audio);
|
64
19
|
const answer = { questionId, type: 'audio', value: audioAttachmentUrl };
|
@@ -73,7 +28,7 @@ function AudioAnswer(audioAnswerProps) {
|
|
73
28
|
const answerObj = { questionId, type: 'audio', value: '' };
|
74
29
|
dispatch(saveAnswer(answerObj));
|
75
30
|
})();
|
76
|
-
} })));
|
31
|
+
}, audioRecorderOptions: { audioBitsPerSecond: 65536 } })));
|
77
32
|
}
|
78
33
|
export default React.memo(AudioAnswer);
|
79
34
|
//# sourceMappingURL=AudioAnswer.js.map
|
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"AudioAnswer.js","sourceRoot":"","sources":["../../../src/components/exam/AudioAnswer.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,
|
1
|
+
{"version":3,"file":"AudioAnswer.js","sourceRoot":"","sources":["../../../src/components/exam/AudioAnswer.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,OAAO,CAAA;AAEzC,OAAO,EAAE,mBAAmB,EAAE,MAAM,iBAAiB,CAAA;AACrD,OAAO,EAAE,UAAU,EAAE,MAAM,6BAA6B,CAAA;AACxD,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AACtD,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAA;AAGpD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAA;AAExD,SAAS,WAAW,CAAC,gBAAoC;IACvD,MAAM,EAAE,OAAO,EAAE,GAAG,gBAAgB,CAAA;IAEpC,MAAM,UAAU,GAAG,mBAAmB,CAAC,OAAO,EAAE,aAAa,CAAE,CAAA;IAC/D,MAAM,MAAM,GAAG,WAAW,CACxB,CAAC,KAAgC,EAAE,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,UAAU,CAA4B,CACvG,CAAA;IACD,MAAM,aAAa,GAAG,OAAO,CAAC,YAAY,CAAC,gBAAgB,CAAE,CAAA;IAC7D,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAA;IAC9B,MAAM,EAAE,aAAa,EAAE,GAAG,UAAU,CAAC,WAAW,CAAC,CAAA;IAEjD,OAAO,CACL,6BAAK,SAAS,EAAC,cAAc;QAC3B,8BAAM,SAAS,EAAC,QAAQ,EAAC,EAAE,EAAE,eAAe,aAAa,EAAE,GAAI;QAC/D,oBAAC,aAAa,IACZ,QAAQ,EAAE,CAAA,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,KAAK,MAAK,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,KAAK,EAC1D,MAAM,EAAE,KAAK,CAAC,EAAE;gBACd,KAAK,CAAC,KAAK;oBACT,MAAM,kBAAkB,GAAG,MAAM,aAAa,CAAC,SAAS,CAAC,UAAU,EAAE,KAAK,CAAC,CAAA;oBAC3E,MAAM,MAAM,GAAG,EAAE,UAAU,EAAE,IAAI,EAAE,OAAgB,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAA;oBAChF,QAAQ,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAA;gBAC9B,CAAC,CAAC,EAAE,CAAA;YACN,CAAC,EACD,QAAQ,EAAE,GAAG,EAAE;gBACb,IAAI,CAAC,MAAM;oBAAE,OAAM;gBACnB,KAAK,CAAC,KAAK;oBACT,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAG,CAAA;oBAC9C,MAAM,aAAa,CAAC,WAAW,CAAC,OAAO,CAAC,CAAA;oBACxC,MAAM,SAAS,GAAG,EAAE,UAAU,EAAE,IAAI,EAAE,OAAgB,EAAE,KAAK,EAAE,EAAE,EAAE,CAAA;oBACnE,QAAQ,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAA;gBACjC,CAAC,CAAC,EAAE,CAAA;YACN,CAAC,EACD,oBAAoB,EAAE,EAAE,kBAAkB,EAAE,KAAK,EAAE,GACnD,CACE,CACP,CAAA;AACH,CAAC;AAED,eAAe,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAA"}
|
@@ -0,0 +1,11 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
type AudioRecorderOptions = Omit<MediaRecorderOptions, 'bitsPerSecond' | 'videoBitsPerSecond'>;
|
3
|
+
interface AudioRecorderProps {
|
4
|
+
audioUrl?: string;
|
5
|
+
onSave: (blob: Blob) => void;
|
6
|
+
onDelete: () => void;
|
7
|
+
audioRecorderOptions?: AudioRecorderOptions;
|
8
|
+
}
|
9
|
+
export declare function AudioRecorder({ audioUrl, onSave, onDelete, audioRecorderOptions }: AudioRecorderProps): React.JSX.Element;
|
10
|
+
export {};
|
11
|
+
//# sourceMappingURL=AudioRecorder.d.ts.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"AudioRecorder.d.ts","sourceRoot":"","sources":["../../../../src/components/exam/internal/AudioRecorder.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAsC,MAAM,OAAO,CAAA;AAQ1D,KAAK,oBAAoB,GAAG,IAAI,CAAC,oBAAoB,EAAE,eAAe,GAAG,oBAAoB,CAAC,CAAA;AAE9F,UAAU,kBAAkB;IAC1B,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,MAAM,EAAE,CAAC,IAAI,EAAE,IAAI,KAAK,IAAI,CAAA;IAC5B,QAAQ,EAAE,MAAM,IAAI,CAAA;IACpB,oBAAoB,CAAC,EAAE,oBAAoB,CAAA;CAC5C;AAED,wBAAgB,aAAa,CAAC,EAAE,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,oBAAoB,EAAE,EAAE,kBAAkB,qBAuIrG"}
|
@@ -0,0 +1,115 @@
|
|
1
|
+
import React, { useEffect, useRef, useState } from 'react';
|
2
|
+
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
3
|
+
import { faMicrophone, faStop } from '@fortawesome/free-solid-svg-icons';
|
4
|
+
import { useExamTranslation } from '../../../i18n';
|
5
|
+
import AudioError from '../../shared/internal/AudioError';
|
6
|
+
import { NBSP } from '../../../dom-utils';
|
7
|
+
export function AudioRecorder({ audioUrl, onSave, onDelete, audioRecorderOptions }) {
|
8
|
+
const [mediaRecorder, setMediaRecorder] = useState();
|
9
|
+
const [timeElapsed, setTimeElapsed] = useState(0);
|
10
|
+
const [status, setStatus] = useState('inactive');
|
11
|
+
const [error, setError] = useState(null);
|
12
|
+
const timer = useRef();
|
13
|
+
const { t } = useExamTranslation();
|
14
|
+
useEffect(() => {
|
15
|
+
void (async function () {
|
16
|
+
var _a;
|
17
|
+
try {
|
18
|
+
if ((_a = navigator.mediaDevices) === null || _a === void 0 ? void 0 : _a.getUserMedia) {
|
19
|
+
const mediaStream = await navigator.mediaDevices.getUserMedia({ audio: true });
|
20
|
+
//const mediaStream = await navigator.mediaDevices.getUserMedia({ audio: { echoCancellation: true } })
|
21
|
+
setMediaRecorder(new MediaRecorder(mediaStream, audioRecorderOptions));
|
22
|
+
}
|
23
|
+
else {
|
24
|
+
showError(new Error('Cannot get media stream, possibly insecure context'));
|
25
|
+
}
|
26
|
+
}
|
27
|
+
catch (err) {
|
28
|
+
showError(err);
|
29
|
+
}
|
30
|
+
return () => stopRecording(); // this never happens while recording
|
31
|
+
})();
|
32
|
+
}, []);
|
33
|
+
function startRecording() {
|
34
|
+
try {
|
35
|
+
setError(null);
|
36
|
+
if (mediaRecorder) {
|
37
|
+
startTimer();
|
38
|
+
mediaRecorder.start();
|
39
|
+
setStatus(mediaRecorder.state);
|
40
|
+
//mediaRecorder.start(1000) // to save in chunks
|
41
|
+
mediaRecorder.ondataavailable = onData;
|
42
|
+
mediaRecorder.onerror = (ev) => showError(new Error(ev.message));
|
43
|
+
}
|
44
|
+
}
|
45
|
+
catch (err) {
|
46
|
+
showError(err);
|
47
|
+
}
|
48
|
+
}
|
49
|
+
function stopRecording() {
|
50
|
+
setError(null);
|
51
|
+
try {
|
52
|
+
if (mediaRecorder) {
|
53
|
+
stopTimer();
|
54
|
+
mediaRecorder.stop();
|
55
|
+
setStatus(mediaRecorder.state);
|
56
|
+
}
|
57
|
+
}
|
58
|
+
catch (err) {
|
59
|
+
showError(err);
|
60
|
+
}
|
61
|
+
}
|
62
|
+
function onData(blobEvent) {
|
63
|
+
onSave(blobEvent.data);
|
64
|
+
}
|
65
|
+
function startTimer() {
|
66
|
+
setTimeElapsed(0);
|
67
|
+
const timerId = setInterval(() => {
|
68
|
+
var _a, _b;
|
69
|
+
const timeElapsedMs = new Date().getTime() - ((_b = (_a = timer === null || timer === void 0 ? void 0 : timer.current) === null || _a === void 0 ? void 0 : _a.startTime) !== null && _b !== void 0 ? _b : new Date()).getTime();
|
70
|
+
setTimeElapsed(Math.floor(timeElapsedMs / 1000));
|
71
|
+
}, 1000);
|
72
|
+
timer.current = { id: timerId, startTime: new Date() };
|
73
|
+
}
|
74
|
+
function stopTimer() {
|
75
|
+
var _a;
|
76
|
+
clearInterval((_a = timer.current) === null || _a === void 0 ? void 0 : _a.id);
|
77
|
+
}
|
78
|
+
function renderTimeElapsed() {
|
79
|
+
const minutes = Math.floor(timeElapsed / 60);
|
80
|
+
const sec = timeElapsed - minutes * 60;
|
81
|
+
const seconds = `00${sec}`.slice(-2);
|
82
|
+
return `${minutes}:${seconds}`;
|
83
|
+
}
|
84
|
+
function showError(err) {
|
85
|
+
stopTimer();
|
86
|
+
const error = err instanceof Error ? err : new Error('unknown error');
|
87
|
+
switch (error.name) {
|
88
|
+
case 'NotAllowedError':
|
89
|
+
return setError('permission-denied');
|
90
|
+
default:
|
91
|
+
console.error(error.name, error.message);
|
92
|
+
return setError('other-recording-error');
|
93
|
+
}
|
94
|
+
}
|
95
|
+
function deleteRecording() {
|
96
|
+
setError(null);
|
97
|
+
onDelete();
|
98
|
+
}
|
99
|
+
return (React.createElement(React.Fragment, null,
|
100
|
+
React.createElement("div", null,
|
101
|
+
React.createElement("p", null,
|
102
|
+
status != 'recording' && !audioUrl && (React.createElement("button", { className: "e-button start-recording", onClick: () => void startRecording(), disabled: error != null },
|
103
|
+
React.createElement(FontAwesomeIcon, { size: "sm", icon: faMicrophone, fixedWidth: true }),
|
104
|
+
t('start.recording'))),
|
105
|
+
React.createElement(React.Fragment, null, status == 'recording' && (React.createElement(React.Fragment, null,
|
106
|
+
React.createElement("button", { className: "e-button stop-recording", onClick: stopRecording, disabled: error != null },
|
107
|
+
React.createElement(FontAwesomeIcon, { size: "sm", icon: faStop, fixedWidth: true }),
|
108
|
+
t('stop.recording')),
|
109
|
+
React.createElement("span", { className: "time-elapsed" }, renderTimeElapsed())))))),
|
110
|
+
audioUrl && (React.createElement("div", { className: "audio-answer-controls" },
|
111
|
+
React.createElement("audio", { src: audioUrl, className: "e-column e-column--narrow", preload: "metadata", controls: true, controlsList: "nodownload" }),
|
112
|
+
React.createElement("button", { className: "e-button-secondary delete-recording", onClick: deleteRecording }, t('remove.recording')))),
|
113
|
+
error && React.createElement(AudioError, { error: error }, NBSP)));
|
114
|
+
}
|
115
|
+
//# sourceMappingURL=AudioRecorder.js.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"AudioRecorder.js","sourceRoot":"","sources":["../../../../src/components/exam/internal/AudioRecorder.tsx"],"names":[],"mappings":"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"}
|
@@ -0,0 +1,10 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
interface MyAudioRecorderProps {
|
3
|
+
audioUrl?: string;
|
4
|
+
onSave: (blob: Blob) => void;
|
5
|
+
onDelete: () => void;
|
6
|
+
mediaRecorderOptions?: MediaRecorderOptions;
|
7
|
+
}
|
8
|
+
export declare function MyAudioRecorder({ audioUrl, onSave, onDelete, mediaRecorderOptions }: MyAudioRecorderProps): React.JSX.Element | null;
|
9
|
+
export {};
|
10
|
+
//# sourceMappingURL=MyAudioRecorder.d.ts.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"MyAudioRecorder.d.ts","sourceRoot":"","sources":["../../../../src/components/exam/internal/MyAudioRecorder.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAsC,MAAM,OAAO,CAAA;AAK1D,UAAU,oBAAoB;IAC5B,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,MAAM,EAAE,CAAC,IAAI,EAAE,IAAI,KAAK,IAAI,CAAA;IAC5B,QAAQ,EAAE,MAAM,IAAI,CAAA;IACpB,oBAAoB,CAAC,EAAE,oBAAoB,CAAA;CAC5C;AAED,wBAAgB,eAAe,CAAC,EAAE,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,oBAAoB,EAAE,EAAE,oBAAoB,4BAkIzG"}
|
@@ -0,0 +1,106 @@
|
|
1
|
+
import React, { useEffect, useRef, useState } from 'react';
|
2
|
+
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
3
|
+
import { faMicrophone, faStop } from '@fortawesome/free-solid-svg-icons';
|
4
|
+
import { useExamTranslation } from '../../../i18n';
|
5
|
+
export function MyAudioRecorder({ audioUrl, onSave, onDelete, mediaRecorderOptions }) {
|
6
|
+
const [mediaRecorder, setMediaRecorder] = useState();
|
7
|
+
const [timeElapsed, setTimeElapsed] = useState(0);
|
8
|
+
const [status, setStatus] = useState('');
|
9
|
+
const [error, setError] = useState();
|
10
|
+
const timer = useRef();
|
11
|
+
const { t } = useExamTranslation();
|
12
|
+
useEffect(() => {
|
13
|
+
void (async function () {
|
14
|
+
try {
|
15
|
+
const mediaStream = await navigator.mediaDevices.getUserMedia({ audio: true });
|
16
|
+
//const mediaStream = await navigator.mediaDevices.getUserMedia({ audio: { echoCancellation: true } })
|
17
|
+
const mediaRecorder = new MediaRecorder(mediaStream, mediaRecorderOptions);
|
18
|
+
setMediaRecorder(mediaRecorder);
|
19
|
+
}
|
20
|
+
catch (err) {
|
21
|
+
showError(err);
|
22
|
+
}
|
23
|
+
return () => stopRecording();
|
24
|
+
})();
|
25
|
+
}, []);
|
26
|
+
useEffect(() => {
|
27
|
+
var _a;
|
28
|
+
setStatus((_a = mediaRecorder === null || mediaRecorder === void 0 ? void 0 : mediaRecorder.state) !== null && _a !== void 0 ? _a : '');
|
29
|
+
}, [mediaRecorder === null || mediaRecorder === void 0 ? void 0 : mediaRecorder.state]);
|
30
|
+
function startRecording() {
|
31
|
+
try {
|
32
|
+
if (mediaRecorder) {
|
33
|
+
startTimer();
|
34
|
+
mediaRecorder.start();
|
35
|
+
//mediaRecorder.start(1000) // to save in chunks
|
36
|
+
mediaRecorder.ondataavailable = onData;
|
37
|
+
mediaRecorder.onerror = (ev) => showError(new Error(ev.message));
|
38
|
+
}
|
39
|
+
}
|
40
|
+
catch (err) {
|
41
|
+
showError(err);
|
42
|
+
}
|
43
|
+
}
|
44
|
+
function stopRecording() {
|
45
|
+
console.info('stop');
|
46
|
+
try {
|
47
|
+
if (mediaRecorder && mediaRecorder.state !== 'inactive') {
|
48
|
+
mediaRecorder.stop();
|
49
|
+
//saveAndStopTimer(new Blob())
|
50
|
+
}
|
51
|
+
}
|
52
|
+
catch (err) {
|
53
|
+
showError(err);
|
54
|
+
}
|
55
|
+
/*if (mediaStream) {
|
56
|
+
mediaStream.getTracks().forEach(track => track.stop())
|
57
|
+
mediaStream = null
|
58
|
+
}*/
|
59
|
+
}
|
60
|
+
function onData(blobEvent) {
|
61
|
+
var _a;
|
62
|
+
clearInterval((_a = timer.current) === null || _a === void 0 ? void 0 : _a.id);
|
63
|
+
setTimeElapsed(0);
|
64
|
+
console.info('save');
|
65
|
+
onSave(blobEvent.data);
|
66
|
+
}
|
67
|
+
/*function saveAndStopTimer(blob: Blob) {
|
68
|
+
console.info('save')
|
69
|
+
clearInterval(timer.current?.id)
|
70
|
+
setTimeElapsed(0)
|
71
|
+
onSave(blob)
|
72
|
+
}*/
|
73
|
+
function startTimer() {
|
74
|
+
const timerId = setInterval(() => {
|
75
|
+
var _a, _b;
|
76
|
+
const timeElapsedMs = new Date().getTime() - ((_b = (_a = timer === null || timer === void 0 ? void 0 : timer.current) === null || _a === void 0 ? void 0 : _a.startTime) !== null && _b !== void 0 ? _b : new Date()).getTime();
|
77
|
+
setTimeElapsed(Math.floor(timeElapsedMs / 1000));
|
78
|
+
}, 1000);
|
79
|
+
timer.current = { id: timerId, startTime: new Date() };
|
80
|
+
}
|
81
|
+
function renderTimeElapsed() {
|
82
|
+
const minutes = Math.floor(timeElapsed / 60);
|
83
|
+
const sec = timeElapsed - minutes * 60;
|
84
|
+
const seconds = `00${sec}`.slice(-2);
|
85
|
+
return `${minutes}:${seconds}`;
|
86
|
+
}
|
87
|
+
function showError(err) {
|
88
|
+
setError(err instanceof Error ? err : new Error('unknown error'));
|
89
|
+
}
|
90
|
+
console.info('status', status);
|
91
|
+
return error ? (React.createElement("div", null, error.message)) : status == '' ? null : (React.createElement(React.Fragment, null,
|
92
|
+
React.createElement("div", null,
|
93
|
+
React.createElement("p", null,
|
94
|
+
status != 'recording' && !audioUrl && (React.createElement("button", { className: "e-button start-recording", onClick: () => void startRecording() },
|
95
|
+
React.createElement(FontAwesomeIcon, { size: "sm", icon: faMicrophone, fixedWidth: true }),
|
96
|
+
t('start.recording'))),
|
97
|
+
React.createElement(React.Fragment, null,
|
98
|
+
status == 'recording' && (React.createElement("button", { className: "e-button stop-recording", onClick: stopRecording },
|
99
|
+
React.createElement(FontAwesomeIcon, { size: "sm", icon: faStop, fixedWidth: true }),
|
100
|
+
t('stop.recording'))),
|
101
|
+
React.createElement("span", { className: "time-elapsed" }, renderTimeElapsed())))),
|
102
|
+
audioUrl && (React.createElement("div", { className: "audio-answer-controls" },
|
103
|
+
React.createElement("audio", { src: audioUrl, className: "e-column e-column--narrow", preload: "metadata", controls: true, controlsList: "nodownload" }),
|
104
|
+
React.createElement("button", { className: "e-button-reversed delete-recording", onClick: onDelete }, t('remove.recording'))))));
|
105
|
+
}
|
106
|
+
//# sourceMappingURL=MyAudioRecorder.js.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"MyAudioRecorder.js","sourceRoot":"","sources":["../../../../src/components/exam/internal/MyAudioRecorder.tsx"],"names":[],"mappings":"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"}
|
@@ -0,0 +1,10 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
interface MyMediaRecorderProps {
|
3
|
+
audioUrl?: string;
|
4
|
+
onSave: (blob: Blob) => void;
|
5
|
+
onDelete: () => void;
|
6
|
+
bitsPerSecond?: number;
|
7
|
+
}
|
8
|
+
export declare function MyMediaRecorder({ onSave, onDelete }: MyMediaRecorderProps): React.JSX.Element;
|
9
|
+
export {};
|
10
|
+
//# sourceMappingURL=MyMediaRecorder.d.ts.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"MyMediaRecorder.d.ts","sourceRoot":"","sources":["../../../../src/components/exam/internal/MyMediaRecorder.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAsC,MAAM,OAAO,CAAA;AAK1D,UAAU,oBAAoB;IAC5B,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,MAAM,EAAE,CAAC,IAAI,EAAE,IAAI,KAAK,IAAI,CAAA;IAC5B,QAAQ,EAAE,MAAM,IAAI,CAAA;IACpB,aAAa,CAAC,EAAE,MAAM,CAAA;CACvB;AAED,wBAAgB,eAAe,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,EAAE,oBAAoB,qBAkHzE"}
|
@@ -0,0 +1,91 @@
|
|
1
|
+
import React, { useEffect, useRef, useState } from 'react';
|
2
|
+
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
3
|
+
import { faMicrophone, faStop } from '@fortawesome/free-solid-svg-icons';
|
4
|
+
import { useExamTranslation } from '../../../i18n';
|
5
|
+
export function MyMediaRecorder({ onSave, onDelete }) {
|
6
|
+
const [mediaRecorder, setMediaRecorder] = useState();
|
7
|
+
const [timeElapsed, setTimeElapsed] = useState(0);
|
8
|
+
const [status, setStatus] = useState('inactive');
|
9
|
+
const [audioUrl, setAudioUrl] = useState();
|
10
|
+
const [error, setError] = useState();
|
11
|
+
const timer = useRef();
|
12
|
+
const { t } = useExamTranslation();
|
13
|
+
useEffect(() => {
|
14
|
+
var _a;
|
15
|
+
setStatus((_a = mediaRecorder === null || mediaRecorder === void 0 ? void 0 : mediaRecorder.state) !== null && _a !== void 0 ? _a : 'inactive');
|
16
|
+
}, [mediaRecorder === null || mediaRecorder === void 0 ? void 0 : mediaRecorder.state]);
|
17
|
+
async function startRecording() {
|
18
|
+
try {
|
19
|
+
const mediaStream = await navigator.mediaDevices.getUserMedia({ audio: true });
|
20
|
+
setMediaRecorder(() => {
|
21
|
+
const mediaRecorder = new MediaRecorder(mediaStream);
|
22
|
+
startTimer();
|
23
|
+
mediaRecorder.start();
|
24
|
+
//mediaRecorder.start(1000)
|
25
|
+
mediaRecorder.ondataavailable = onData;
|
26
|
+
return mediaRecorder;
|
27
|
+
});
|
28
|
+
}
|
29
|
+
catch (err) {
|
30
|
+
setError(err instanceof Error ? err : new Error('unknown error'));
|
31
|
+
}
|
32
|
+
}
|
33
|
+
function stopRecording() {
|
34
|
+
try {
|
35
|
+
if (mediaRecorder && mediaRecorder.state !== 'inactive') {
|
36
|
+
mediaRecorder.stop();
|
37
|
+
saveAndStopTimer(new Blob());
|
38
|
+
}
|
39
|
+
}
|
40
|
+
catch (err) {
|
41
|
+
setError(err instanceof Error ? err : new Error('unknown error'));
|
42
|
+
}
|
43
|
+
/*if (mediaStream) {
|
44
|
+
mediaStream.getTracks().forEach(track => track.stop())
|
45
|
+
mediaStream = null
|
46
|
+
}*/
|
47
|
+
}
|
48
|
+
function onData(blobEvent) {
|
49
|
+
console.info('data', blobEvent.data);
|
50
|
+
}
|
51
|
+
function saveAndStopTimer(blob) {
|
52
|
+
var _a;
|
53
|
+
console.info('save');
|
54
|
+
clearInterval((_a = timer.current) === null || _a === void 0 ? void 0 : _a.id);
|
55
|
+
setTimeElapsed(0);
|
56
|
+
console.info('save blob', blob);
|
57
|
+
//onSave(blob)
|
58
|
+
}
|
59
|
+
function onDelete() { }
|
60
|
+
function startTimer() {
|
61
|
+
const timerId = setInterval(() => {
|
62
|
+
var _a, _b;
|
63
|
+
const timeElapsedMs = new Date().getTime() - ((_b = (_a = timer === null || timer === void 0 ? void 0 : timer.current) === null || _a === void 0 ? void 0 : _a.startTime) !== null && _b !== void 0 ? _b : new Date()).getTime();
|
64
|
+
setTimeElapsed(Math.floor(timeElapsedMs / 1000));
|
65
|
+
}, 1000);
|
66
|
+
timer.current = { id: timerId, startTime: new Date() };
|
67
|
+
}
|
68
|
+
function renderTimeElapsed() {
|
69
|
+
const minutes = Math.floor(timeElapsed / 60);
|
70
|
+
const sec = timeElapsed - minutes * 60;
|
71
|
+
const seconds = `00${sec}`.slice(-2);
|
72
|
+
return `${minutes}:${seconds}`;
|
73
|
+
}
|
74
|
+
console.info('status', status);
|
75
|
+
return (React.createElement(React.Fragment, null,
|
76
|
+
React.createElement("div", null,
|
77
|
+
React.createElement("p", null,
|
78
|
+
[undefined, 'inactive'].includes(mediaRecorder === null || mediaRecorder === void 0 ? void 0 : mediaRecorder.state) && (React.createElement("button", { className: "e-button start-recording", onClick: () => void startRecording() },
|
79
|
+
React.createElement(FontAwesomeIcon, { size: "sm", icon: faMicrophone, fixedWidth: true }),
|
80
|
+
t('start.recording'))),
|
81
|
+
React.createElement(React.Fragment, null,
|
82
|
+
(mediaRecorder === null || mediaRecorder === void 0 ? void 0 : mediaRecorder.state) == 'recording' && (React.createElement("button", { className: "e-button stop-recording", onClick: stopRecording },
|
83
|
+
React.createElement(FontAwesomeIcon, { size: "sm", icon: faStop, fixedWidth: true }),
|
84
|
+
t('stop.recording'))),
|
85
|
+
React.createElement("span", { className: "time-elapsed" }, renderTimeElapsed())))),
|
86
|
+
audioUrl && (React.createElement("div", { className: "audio-answer-controls" },
|
87
|
+
React.createElement("audio", { src: audioUrl, className: "e-column e-column--narrow", preload: "metadata", controls: true, controlsList: "nodownload" }),
|
88
|
+
React.createElement("button", { className: "e-button-reversed delete-recording", onClick: onDelete }, t('remove.recording')))),
|
89
|
+
error && React.createElement("div", null, error.message)));
|
90
|
+
}
|
91
|
+
//# sourceMappingURL=MyMediaRecorder.js.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"MyMediaRecorder.js","sourceRoot":"","sources":["../../../../src/components/exam/internal/MyMediaRecorder.tsx"],"names":[],"mappings":"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"}
|
@@ -5,7 +5,7 @@ import { getNumericAttribute } from '../../dom-utils';
|
|
5
5
|
import { useExamTranslation } from '../../i18n';
|
6
6
|
import { audioLabelId } from '../../ids';
|
7
7
|
import { getAudioPlaybackError, getPlaybackTimesRemaining } from '../../store/selectors';
|
8
|
-
import
|
8
|
+
import AudioError from './internal/AudioError';
|
9
9
|
import { CommonExamContext } from '../context/CommonExamContext';
|
10
10
|
import RestrictedAudioPlayer from './internal/RestrictedAudioPlayer';
|
11
11
|
function Audio({ element, className, renderChildNodes }) {
|
@@ -24,7 +24,7 @@ function Audio({ element, className, renderChildNodes }) {
|
|
24
24
|
restrictedAudioId != null && times != null ? (React.createElement(RestrictedAudioPlayer, { src, restrictedAudioId, duration, times, labelId })) : (React.createElement("audio", { className: "e-column e-column--narrow", "aria-describedby": labelId, preload: "metadata", controls: true, controlsList: "nodownload" },
|
25
25
|
React.createElement("source", { src: resolveAttachment(src) }))),
|
26
26
|
React.createElement("div", { className: "e-column", id: labelId },
|
27
|
-
React.createElement(
|
27
|
+
React.createElement(AudioError, { error: audioPlaybackError }, renderChildNodes(element)),
|
28
28
|
playbackTimesRemaining != null && React.createElement("em", null, t('listen-times-remaining', { count: playbackTimesRemaining })))));
|
29
29
|
}
|
30
30
|
export default React.memo(Audio);
|
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"Audio.js","sourceRoot":"","sources":["../../../src/components/shared/Audio.tsx"],"names":[],"mappings":"AAAA,OAAO,UAAU,MAAM,YAAY,CAAA;AACnC,OAAO,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,OAAO,CAAA;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAEzC,OAAO,EAAE,mBAAmB,EAAE,MAAM,iBAAiB,CAAA;AACrD,OAAO,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AAC/C,OAAO,EAAE,YAAY,EAAE,MAAM,WAAW,CAAA;AACxC,OAAO,EAAE,qBAAqB,EAAE,yBAAyB,EAAE,MAAM,uBAAuB,CAAA;AACxF,OAAO,
|
1
|
+
{"version":3,"file":"Audio.js","sourceRoot":"","sources":["../../../src/components/shared/Audio.tsx"],"names":[],"mappings":"AAAA,OAAO,UAAU,MAAM,YAAY,CAAA;AACnC,OAAO,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,OAAO,CAAA;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAEzC,OAAO,EAAE,mBAAmB,EAAE,MAAM,iBAAiB,CAAA;AACrD,OAAO,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AAC/C,OAAO,EAAE,YAAY,EAAE,MAAM,WAAW,CAAA;AACxC,OAAO,EAAE,qBAAqB,EAAE,yBAAyB,EAAE,MAAM,uBAAuB,CAAA;AACxF,OAAO,UAAU,MAAM,uBAAuB,CAAA;AAC9C,OAAO,EAAE,iBAAiB,EAAE,MAAM,8BAA8B,CAAA;AAChE,OAAO,qBAAqB,MAAM,kCAAkC,CAAA;AAEpE,SAAS,KAAK,CAAC,EAAE,OAAO,EAAE,SAAS,EAAE,gBAAgB,EAAsB;IACzE,MAAM,GAAG,GAAG,OAAO,CAAC,YAAY,CAAC,KAAK,CAAE,CAAA;IACxC,MAAM,iBAAiB,GAAG,mBAAmB,CAAC,OAAO,EAAE,qBAAqB,CAAC,CAAA;IAC7E,MAAM,KAAK,GAAG,mBAAmB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAA;IACnD,MAAM,QAAQ,GAAG,mBAAmB,CAAC,OAAO,EAAE,UAAU,CAAE,CAAA;IAC1D,MAAM,EAAE,CAAC,EAAE,GAAG,kBAAkB,EAAE,CAAA;IAClC,MAAM,kBAAkB,GACtB,KAAK,IAAI,IAAI,IAAI,iBAAiB,IAAI,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,qBAAqB,CAAC,GAAG,EAAE,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAA;IACrH,MAAM,sBAAsB,GAC1B,KAAK,IAAI,IAAI,IAAI,iBAAiB,IAAI,IAAI;QACxC,CAAC,CAAC,WAAW,CAAC,yBAAyB,CAAC,iBAAiB,EAAE,KAAK,CAAC,CAAC;QAClE,CAAC,CAAC,SAAS,CAAA;IACf,MAAM,EAAE,iBAAiB,EAAE,GAAG,UAAU,CAAC,iBAAiB,CAAC,CAAA;IAC3D,MAAM,OAAO,GAAG,YAAY,CAAC,OAAO,CAAC,CAAA;IAErC,OAAO,CACL,6BACE,SAAS,EAAE,UAAU,CAAC,gEAAgE,EAAE,SAAS,CAAC,EAClG,OAAO,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,EAAE;QAE/B,iBAAiB,IAAI,IAAI,IAAI,KAAK,IAAI,IAAI,CAAC,CAAC,CAAC,CAC5C,oBAAC,qBAAqB,IAAO,GAAG,EAAE,iBAAiB,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,GAAM,CACpF,CAAC,CAAC,CAAC,CACF,+BACE,SAAS,EAAC,2BAA2B,sBACnB,OAAO,EACzB,OAAO,EAAC,UAAU,EAClB,QAAQ,QACR,YAAY,EAAC,YAAY;YAEzB,gCAAQ,GAAG,EAAE,iBAAiB,CAAC,GAAG,CAAC,GAAI,CACjC,CACT;QACD,6BAAK,SAAS,EAAC,UAAU,EAAC,EAAE,EAAE,OAAO;YACnC,oBAAC,UAAU,IAAC,KAAK,EAAE,kBAAkB,IAAG,gBAAgB,CAAC,OAAO,CAAC,CAAc;YAC9E,sBAAsB,IAAI,IAAI,IAAI,gCAAK,CAAC,CAAC,wBAAwB,EAAE,EAAE,KAAK,EAAE,sBAAsB,EAAE,CAAC,CAAM,CACxG,CACF,CACP,CAAA;AACH,CAAC;AAED,eAAe,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA"}
|
@@ -6,7 +6,7 @@ import { getNumericAttribute, NBSP } from '../../dom-utils';
|
|
6
6
|
import { useExamTranslation } from '../../i18n';
|
7
7
|
import { playAudio } from '../../store/audio/actions';
|
8
8
|
import { getAudioPlaybackError, getAudioState } from '../../store/selectors';
|
9
|
-
import
|
9
|
+
import AudioError from './internal/AudioError';
|
10
10
|
import { CommonExamContext } from '../context/CommonExamContext';
|
11
11
|
function AudioTest({ element }) {
|
12
12
|
const src = element.getAttribute('src');
|
@@ -27,7 +27,7 @@ function AudioTest({ element }) {
|
|
27
27
|
React.createElement(FontAwesomeIcon, { className: "e-mrg-r-1", icon: faPlay, fixedWidth: true }),
|
28
28
|
t('audio-test.play'))),
|
29
29
|
React.createElement("div", { className: "e-column" },
|
30
|
-
React.createElement(
|
30
|
+
React.createElement(AudioError, { error: audioPlaybackError }, NBSP))),
|
31
31
|
React.createElement("div", null, t('audio-test.volume'))));
|
32
32
|
}
|
33
33
|
export default React.memo(AudioTest);
|
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"AudioTest.js","sourceRoot":"","sources":["../../../src/components/shared/AudioTest.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,mCAAmC,CAAA;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAA;AAChE,OAAO,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,OAAO,CAAA;AACjD,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAEtD,OAAO,EAAE,mBAAmB,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAA;AAC3D,OAAO,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AAC/C,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAA;AACrD,OAAO,EAAE,qBAAqB,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAA;AAC5E,OAAO,
|
1
|
+
{"version":3,"file":"AudioTest.js","sourceRoot":"","sources":["../../../src/components/shared/AudioTest.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,mCAAmC,CAAA;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAA;AAChE,OAAO,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,OAAO,CAAA;AACjD,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAEtD,OAAO,EAAE,mBAAmB,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAA;AAC3D,OAAO,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AAC/C,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAA;AACrD,OAAO,EAAE,qBAAqB,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAA;AAC5E,OAAO,UAAU,MAAM,uBAAuB,CAAA;AAC9C,OAAO,EAAE,iBAAiB,EAAE,MAAM,8BAA8B,CAAA;AAEhE,SAAS,SAAS,CAAC,EAAE,OAAO,EAAsB;IAChD,MAAM,GAAG,GAAG,OAAO,CAAC,YAAY,CAAC,KAAK,CAAE,CAAA;IACxC,MAAM,QAAQ,GAAG,mBAAmB,CAAC,OAAO,EAAE,UAAU,CAAE,CAAA;IAC1D,MAAM,EAAE,CAAC,EAAE,GAAG,kBAAkB,EAAE,CAAA;IAClC,MAAM,UAAU,GAAG,WAAW,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAA;IAClD,MAAM,kBAAkB,GAAG,WAAW,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAA;IAClE,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAA;IAC9B,MAAM,EAAE,OAAO,EAAE,iBAAiB,EAAE,GAAG,UAAU,CAAC,iBAAiB,CAAC,CAAA;IACpE,MAAM,QAAQ,GAAG,MAAM,CAA0B,IAAI,CAAC,CAAA;IAEtD,OAAO,CACL,6BAAK,SAAS,EAAC,6CAA6C;QAC1D,+BAAI,CAAC,CAAC,yBAAyB,CAAC,CAAK;QACrC,6BAAK,SAAS,EAAC,yCAAyC;YACtD,6BAAK,SAAS,EAAC,2BAA2B;gBACvC,OAAO,IAAI,CACV,+BAAO,GAAG,EAAE,QAAQ,EAAE,OAAO,EAAC,MAAM;oBAClC,gCAAQ,GAAG,EAAE,iBAAiB,CAAC,GAAG,CAAC,GAAI,CACjC,CACT;gBACD,gCACE,SAAS,EAAC,UAAU,EACpB,QAAQ,EAAE,UAAU,KAAK,SAAS,EAClC,OAAO,EAAE,GAAG,EAAE,CAAC,UAAU,KAAK,SAAS,IAAI,QAAQ,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC,CAAC;oBAE3F,oBAAC,eAAe,IAAC,SAAS,EAAC,WAAW,EAAC,IAAI,EAAE,MAAM,EAAE,UAAU,SAAG;oBACjE,CAAC,CAAC,iBAAiB,CAAC,CACd,CACL;YACN,6BAAK,SAAS,EAAC,UAAU;gBACvB,oBAAC,UAAU,IAAC,KAAK,EAAE,kBAAkB,IAAG,IAAI,CAAc,CACtD,CACF;QACN,iCAAM,CAAC,CAAC,mBAAmB,CAAC,CAAO,CAC/B,CACP,CAAA;AACH,CAAC;AAED,eAAe,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA"}
|
@@ -0,0 +1,8 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import { AudioError } from '../../../index';
|
3
|
+
declare const AudioError: React.FunctionComponent<{
|
4
|
+
error?: AudioError;
|
5
|
+
children: React.ReactNode[] | React.ReactNode;
|
6
|
+
}>;
|
7
|
+
export default AudioError;
|
8
|
+
//# sourceMappingURL=AudioError.d.ts.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"AudioError.d.ts","sourceRoot":"","sources":["../../../../src/components/shared/internal/AudioError.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,MAAM,OAAO,CAAA;AAEzB,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAA;AAG3C,QAAA,MAAM,UAAU,EAAE,KAAK,CAAC,iBAAiB,CAAC;IACxC,KAAK,CAAC,EAAE,UAAU,CAAA;IAClB,QAAQ,EAAE,KAAK,CAAC,SAAS,EAAE,GAAG,KAAK,CAAC,SAAS,CAAA;CAC9C,CAkBA,CAAA;AAED,eAAe,UAAU,CAAA"}
|
@@ -0,0 +1,17 @@
|
|
1
|
+
import { faExclamationTriangle } from '@fortawesome/free-solid-svg-icons';
|
2
|
+
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
3
|
+
import React from 'react';
|
4
|
+
import ReactCSSTransitionReplace from 'react-css-transition-replace';
|
5
|
+
import { useExamTranslation } from '../../../i18n';
|
6
|
+
const AudioError = ({ error, children }) => {
|
7
|
+
const { t } = useExamTranslation();
|
8
|
+
return (
|
9
|
+
// @types/react 18 removed children from implicit props. This component has not explicitly added them yet
|
10
|
+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
11
|
+
// @ts-ignore
|
12
|
+
React.createElement(ReactCSSTransitionReplace, { transitionName: "e-crossfade", transitionEnterTimeout: 500, transitionLeaveTimeout: 500 }, error != null ? (React.createElement("div", { className: "e-color-error", role: "alert" },
|
13
|
+
React.createElement(FontAwesomeIcon, { icon: faExclamationTriangle, className: "e-mrg-r-1" }),
|
14
|
+
t(`audio-errors.${error}`))) : (React.createElement("div", { key: "no-error" }, children))));
|
15
|
+
};
|
16
|
+
export default AudioError;
|
17
|
+
//# sourceMappingURL=AudioError.js.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"AudioError.js","sourceRoot":"","sources":["../../../../src/components/shared/internal/AudioError.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,qBAAqB,EAAE,MAAM,mCAAmC,CAAA;AACzE,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAA;AAChE,OAAO,KAAK,MAAM,OAAO,CAAA;AACzB,OAAO,yBAAyB,MAAM,8BAA8B,CAAA;AAEpE,OAAO,EAAE,kBAAkB,EAAE,MAAM,eAAe,CAAA;AAElD,MAAM,UAAU,GAGX,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE,EAAE;IAC3B,MAAM,EAAE,CAAC,EAAE,GAAG,kBAAkB,EAAE,CAAA;IAElC,OAAO;IACL,yGAAyG;IACzG,6DAA6D;IAC7D,aAAa;IACb,oBAAC,yBAAyB,IAAC,cAAc,EAAC,aAAa,EAAC,sBAAsB,EAAE,GAAG,EAAE,sBAAsB,EAAE,GAAG,IAC7G,KAAK,IAAI,IAAI,CAAC,CAAC,CAAC,CACf,6BAAK,SAAS,EAAC,eAAe,EAAC,IAAI,EAAC,OAAO;QACzC,oBAAC,eAAe,IAAC,IAAI,EAAE,qBAAqB,EAAE,SAAS,EAAC,WAAW,GAAG;QACrE,CAAC,CAAC,gBAAgB,KAAK,EAAW,CAAC,CAChC,CACP,CAAC,CAAC,CAAC,CACF,6BAAK,GAAG,EAAC,UAAU,IAAE,QAAQ,CAAO,CACrC,CACyB,CAC7B,CAAA;AACH,CAAC,CAAA;AAED,eAAe,UAAU,CAAA"}
|
@@ -0,0 +1,8 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import { AudioError } from '../../../index';
|
3
|
+
declare const AudioErrorRemoveMe: React.FunctionComponent<{
|
4
|
+
error?: AudioError;
|
5
|
+
children: React.ReactNode[] | React.ReactNode;
|
6
|
+
}>;
|
7
|
+
export default AudioErrorRemoveMe;
|
8
|
+
//# sourceMappingURL=AudioErrorRemoveMe.d.ts.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"AudioErrorRemoveMe.d.ts","sourceRoot":"","sources":["../../../../src/components/shared/internal/AudioErrorRemoveMe.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,MAAM,OAAO,CAAA;AAEzB,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAA;AAG3C,QAAA,MAAM,kBAAkB,EAAE,KAAK,CAAC,iBAAiB,CAAC;IAChD,KAAK,CAAC,EAAE,UAAU,CAAA;IAClB,QAAQ,EAAE,KAAK,CAAC,SAAS,EAAE,GAAG,KAAK,CAAC,SAAS,CAAA;CAC9C,CAkBA,CAAA;AAED,eAAe,kBAAkB,CAAA"}
|