@clarionhq/recorder 0.0.1 → 0.1.1

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.
Files changed (94) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +110 -0
  3. package/android/CMakeLists.txt +9 -0
  4. package/android/build.gradle +97 -0
  5. package/android/gradle.properties +4 -0
  6. package/android/proguard-rules.pro +1 -0
  7. package/android/src/main/AndroidManifest.xml +4 -0
  8. package/android/src/main/cpp/cpp-adapter.cpp +7 -0
  9. package/android/src/main/java/com/clarionhq/recorder/AudioLevelMeter.kt +38 -0
  10. package/android/src/main/java/com/clarionhq/recorder/ClarionRecorderPackage.kt +27 -0
  11. package/android/src/main/java/com/clarionhq/recorder/EncodeFile.kt +62 -0
  12. package/android/src/main/java/com/clarionhq/recorder/RecorderConfig.kt +33 -0
  13. package/android/src/main/java/com/clarionhq/recorder/RecorderConstants.kt +16 -0
  14. package/android/src/main/java/com/clarionhq/recorder/RecorderSession.kt +336 -0
  15. package/android/src/main/java/com/clarionhq/recorder/RecorderTypes.kt +29 -0
  16. package/android/src/main/java/com/margelo/nitro/clarion/recorder/HybridClarionRecorder.kt +174 -0
  17. package/clarionhq-recorder.podspec +31 -0
  18. package/ios/AudioLevelMeter.swift +37 -0
  19. package/ios/EncodeFile.swift +69 -0
  20. package/ios/HybridClarionRecorder.swift +186 -0
  21. package/ios/RecorderConstants.swift +11 -0
  22. package/ios/RecorderSession.swift +278 -0
  23. package/ios/RecorderTypes.swift +41 -0
  24. package/lib/RecorderEngine.d.ts +31 -0
  25. package/lib/RecorderEngine.d.ts.map +1 -0
  26. package/lib/RecorderEngine.js +245 -0
  27. package/lib/RecorderEngine.js.map +1 -0
  28. package/lib/index.d.ts +4 -0
  29. package/lib/index.d.ts.map +1 -0
  30. package/lib/index.js +2 -0
  31. package/lib/index.js.map +1 -0
  32. package/lib/native.d.ts +3 -0
  33. package/lib/native.d.ts.map +1 -0
  34. package/lib/native.js +3 -0
  35. package/lib/native.js.map +1 -0
  36. package/lib/specs/ClarionRecorder.nitro.d.ts +49 -0
  37. package/lib/specs/ClarionRecorder.nitro.d.ts.map +1 -0
  38. package/lib/specs/ClarionRecorder.nitro.js +2 -0
  39. package/lib/specs/ClarionRecorder.nitro.js.map +1 -0
  40. package/nitro.json +24 -0
  41. package/nitrogen/generated/android/ClarionRecorder+autolinking.cmake +81 -0
  42. package/nitrogen/generated/android/ClarionRecorder+autolinking.gradle +27 -0
  43. package/nitrogen/generated/android/ClarionRecorderOnLoad.cpp +62 -0
  44. package/nitrogen/generated/android/ClarionRecorderOnLoad.hpp +34 -0
  45. package/nitrogen/generated/android/c++/JFunc_void_NativeRecorderError.hpp +78 -0
  46. package/nitrogen/generated/android/c++/JFunc_void_double_double.hpp +75 -0
  47. package/nitrogen/generated/android/c++/JFunc_void_std__string.hpp +76 -0
  48. package/nitrogen/generated/android/c++/JFunc_void_std__string_double_double_double.hpp +76 -0
  49. package/nitrogen/generated/android/c++/JHybridClarionRecorderSpec.cpp +207 -0
  50. package/nitrogen/generated/android/c++/JHybridClarionRecorderSpec.hpp +75 -0
  51. package/nitrogen/generated/android/c++/JNativeRecorderConfig.hpp +90 -0
  52. package/nitrogen/generated/android/c++/JNativeRecorderError.hpp +65 -0
  53. package/nitrogen/generated/android/c++/JNativeRecorderResult.hpp +77 -0
  54. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/ClarionRecorderOnLoad.kt +35 -0
  55. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/Func_void_NativeRecorderError.kt +80 -0
  56. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/Func_void_double_double.kt +80 -0
  57. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/Func_void_std__string.kt +80 -0
  58. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/Func_void_std__string_double_double_double.kt +80 -0
  59. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/HybridClarionRecorderSpec.kt +125 -0
  60. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/NativeRecorderConfig.kt +91 -0
  61. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/NativeRecorderError.kt +61 -0
  62. package/nitrogen/generated/android/kotlin/com/margelo/nitro/clarion/recorder/NativeRecorderResult.kt +76 -0
  63. package/nitrogen/generated/ios/ClarionRecorder+autolinking.rb +62 -0
  64. package/nitrogen/generated/ios/ClarionRecorder-Swift-Cxx-Bridge.cpp +89 -0
  65. package/nitrogen/generated/ios/ClarionRecorder-Swift-Cxx-Bridge.hpp +297 -0
  66. package/nitrogen/generated/ios/ClarionRecorder-Swift-Cxx-Umbrella.hpp +56 -0
  67. package/nitrogen/generated/ios/ClarionRecorderAutolinking.mm +33 -0
  68. package/nitrogen/generated/ios/ClarionRecorderAutolinking.swift +26 -0
  69. package/nitrogen/generated/ios/c++/HybridClarionRecorderSpecSwift.cpp +11 -0
  70. package/nitrogen/generated/ios/c++/HybridClarionRecorderSpecSwift.hpp +188 -0
  71. package/nitrogen/generated/ios/swift/Func_void.swift +46 -0
  72. package/nitrogen/generated/ios/swift/Func_void_NativeRecorderError.swift +46 -0
  73. package/nitrogen/generated/ios/swift/Func_void_NativeRecorderResult.swift +46 -0
  74. package/nitrogen/generated/ios/swift/Func_void_double_double.swift +46 -0
  75. package/nitrogen/generated/ios/swift/Func_void_std__exception_ptr.swift +46 -0
  76. package/nitrogen/generated/ios/swift/Func_void_std__string.swift +46 -0
  77. package/nitrogen/generated/ios/swift/Func_void_std__string_double_double_double.swift +46 -0
  78. package/nitrogen/generated/ios/swift/HybridClarionRecorderSpec.swift +67 -0
  79. package/nitrogen/generated/ios/swift/HybridClarionRecorderSpec_cxx.swift +354 -0
  80. package/nitrogen/generated/ios/swift/NativeRecorderConfig.swift +108 -0
  81. package/nitrogen/generated/ios/swift/NativeRecorderError.swift +39 -0
  82. package/nitrogen/generated/ios/swift/NativeRecorderResult.swift +54 -0
  83. package/nitrogen/generated/shared/c++/HybridClarionRecorderSpec.cpp +34 -0
  84. package/nitrogen/generated/shared/c++/HybridClarionRecorderSpec.hpp +84 -0
  85. package/nitrogen/generated/shared/c++/NativeRecorderConfig.hpp +116 -0
  86. package/nitrogen/generated/shared/c++/NativeRecorderError.hpp +91 -0
  87. package/nitrogen/generated/shared/c++/NativeRecorderResult.hpp +103 -0
  88. package/package.json +68 -8
  89. package/react-native.config.js +10 -0
  90. package/src/RecorderEngine.ts +298 -0
  91. package/src/index.ts +8 -0
  92. package/src/native.ts +5 -0
  93. package/src/specs/ClarionRecorder.nitro.ts +58 -0
  94. package/index.js +0 -1
@@ -0,0 +1,298 @@
1
+ import {
2
+ ClarionEmitter,
3
+ ClarionError,
4
+ DEFAULT_AUDIO_FORMAT,
5
+ assertTransition,
6
+ type ClarionEngine,
7
+ type ClarionEvent,
8
+ type EngineKind,
9
+ type EngineState,
10
+ type Listener,
11
+ type RecorderEngineConfig,
12
+ type RecorderResult,
13
+ type Unsubscribe,
14
+ } from '@clarionhq/core';
15
+
16
+ import { createNativeRecorder } from './native';
17
+ import type {
18
+ ClarionRecorder,
19
+ NativeRecorderConfig,
20
+ NativeRecorderError,
21
+ NativeRecorderResult,
22
+ } from './specs/ClarionRecorder.nitro';
23
+
24
+ const VALID_STATES: readonly EngineState[] = [
25
+ 'idle',
26
+ 'preparing',
27
+ 'ready',
28
+ 'starting',
29
+ 'recording',
30
+ 'paused',
31
+ 'stopping',
32
+ 'error',
33
+ 'released',
34
+ ];
35
+
36
+ const isEngineState = (s: string): s is EngineState =>
37
+ (VALID_STATES as readonly string[]).includes(s);
38
+
39
+ // 32 kbps is the highest AAC-LC bitrate universally supported across iOS
40
+ // (Simulator + device) and Android for 16 kHz mono. Higher bitrates work
41
+ // on Android but iOS's AAC encoder rejects them with
42
+ // kAudioFormatUnsupportedDataFormatError. Callers can override for higher
43
+ // sample rates / stereo where larger values are accepted.
44
+ const DEFAULT_AAC_BITRATE = 32_000;
45
+ const DEFAULT_AUDIO_LEVEL_INTERVAL_MS = 50;
46
+
47
+ export interface RecorderEngineOptions extends RecorderEngineConfig {
48
+ aacBitrate?: number;
49
+ }
50
+
51
+ export class RecorderEngine implements ClarionEngine {
52
+ readonly kind: EngineKind = 'recorder';
53
+
54
+ private readonly emitter = new ClarionEmitter();
55
+ private readonly native: ClarionRecorder;
56
+ private readonly listenerIds: number[] = [];
57
+ private readonly options: RecorderEngineOptions;
58
+
59
+ private currentState: EngineState = 'idle';
60
+
61
+ constructor(options: RecorderEngineOptions = {}) {
62
+ this.options = options;
63
+ this.native = createNativeRecorder();
64
+ this.bindNativeListeners();
65
+ }
66
+
67
+ get state(): EngineState {
68
+ return this.currentState;
69
+ }
70
+
71
+ on(listener: Listener<ClarionEvent>): Unsubscribe {
72
+ return this.emitter.on(listener);
73
+ }
74
+
75
+ async prepare(): Promise<void> {
76
+ this.transitionTo('preparing');
77
+ try {
78
+ const config = this.buildNativeConfig();
79
+ await this.native.prepare(config);
80
+ // Native side fires state='ready' via the state listener — no JS-side transition needed.
81
+ } catch (err) {
82
+ this.handleNativeError(err, 'prepare');
83
+ throw err;
84
+ }
85
+ }
86
+
87
+ async start(): Promise<void> {
88
+ // Auto-recover from a previous error — caller can retry without manual reset.
89
+ if (this.currentState === 'error') {
90
+ this.setState('idle');
91
+ }
92
+ // Auto-prepare if no session yet — callers don't need to know about prepare().
93
+ if (this.currentState === 'idle') {
94
+ await this.prepare();
95
+ }
96
+ if (this.currentState !== 'ready') {
97
+ throw new ClarionError({
98
+ code: 'INVALID_STATE',
99
+ message: `Cannot start from state '${this.currentState}'`,
100
+ });
101
+ }
102
+ this.transitionTo('starting');
103
+ try {
104
+ await this.native.start();
105
+ // Native side fires state='recording' via the state listener.
106
+ } catch (err) {
107
+ this.handleNativeError(err, 'start');
108
+ throw err;
109
+ }
110
+ }
111
+
112
+ async pause(): Promise<void> {
113
+ if (this.currentState !== 'recording') {
114
+ throw new ClarionError({
115
+ code: 'INVALID_STATE',
116
+ message: `Cannot pause from state '${this.currentState}'`,
117
+ });
118
+ }
119
+ try {
120
+ await this.native.pause();
121
+ } catch (err) {
122
+ this.handleNativeError(err, 'pause');
123
+ throw err;
124
+ }
125
+ }
126
+
127
+ async resume(): Promise<void> {
128
+ if (this.currentState !== 'paused') {
129
+ throw new ClarionError({
130
+ code: 'INVALID_STATE',
131
+ message: `Cannot resume from state '${this.currentState}'`,
132
+ });
133
+ }
134
+ try {
135
+ await this.native.resume();
136
+ } catch (err) {
137
+ this.handleNativeError(err, 'resume');
138
+ throw err;
139
+ }
140
+ }
141
+
142
+ async stop(): Promise<void> {
143
+ this.transitionTo('stopping');
144
+ try {
145
+ const result = await this.native.stop();
146
+ this.emitter.emit({
147
+ type: 'recording-complete',
148
+ result: this.toRecorderResult(result),
149
+ });
150
+ } catch (err) {
151
+ this.handleNativeError(err, 'stop');
152
+ throw err;
153
+ }
154
+ }
155
+
156
+ async discard(): Promise<void> {
157
+ try {
158
+ await this.native.discard();
159
+ } catch (err) {
160
+ this.handleNativeError(err, 'discard');
161
+ throw err;
162
+ }
163
+ }
164
+
165
+ async release(): Promise<void> {
166
+ try {
167
+ this.native.removeAllListeners();
168
+ await this.native.release();
169
+ } finally {
170
+ this.listenerIds.length = 0;
171
+ this.emitter.removeAll();
172
+ this.currentState = 'released';
173
+ }
174
+ }
175
+
176
+ private buildNativeConfig(): NativeRecorderConfig {
177
+ const format = DEFAULT_AUDIO_FORMAT;
178
+ const cfg: NativeRecorderConfig = {
179
+ sampleRate: format.sampleRate,
180
+ channels: format.channels,
181
+ bitDepth: format.bitDepth,
182
+ emitAudioLevel: this.options.emitAudioLevel ?? false,
183
+ audioLevelIntervalMs:
184
+ this.options.audioLevelIntervalMs ?? DEFAULT_AUDIO_LEVEL_INTERVAL_MS,
185
+ aacBitrate: this.options.aacBitrate ?? DEFAULT_AAC_BITRATE,
186
+ };
187
+ if (this.options.outputDirectory !== undefined) {
188
+ cfg.outputDirectory = this.options.outputDirectory;
189
+ }
190
+ if (this.options.filenamePrefix !== undefined) {
191
+ cfg.filenamePrefix = this.options.filenamePrefix;
192
+ }
193
+ if (this.options.rotateAfterMs !== undefined) {
194
+ cfg.rotateAfterMs = this.options.rotateAfterMs;
195
+ }
196
+ return cfg;
197
+ }
198
+
199
+ private bindNativeListeners(): void {
200
+ this.listenerIds.push(
201
+ this.native.addStateListener((nativeState) => {
202
+ if (!isEngineState(nativeState)) return;
203
+ this.setState(nativeState);
204
+ }),
205
+ );
206
+
207
+ this.listenerIds.push(
208
+ this.native.addAudioLevelListener((rms, peak) => {
209
+ this.emitter.emit({ type: 'audio-level', rms, peak });
210
+ }),
211
+ );
212
+
213
+ this.listenerIds.push(
214
+ this.native.addChunkListener((uri, startMs, endMs, sizeBytes) => {
215
+ this.emitter.emit({
216
+ type: 'chunk',
217
+ uri,
218
+ startMs,
219
+ endMs,
220
+ sizeBytes,
221
+ });
222
+ }),
223
+ );
224
+
225
+ this.listenerIds.push(
226
+ this.native.addErrorListener((err) => {
227
+ this.emitter.emit({
228
+ type: 'error',
229
+ error: this.toClarionError(err),
230
+ });
231
+ }),
232
+ );
233
+ }
234
+
235
+ private transitionTo(next: EngineState): void {
236
+ if (this.currentState === next) return;
237
+ assertTransition(this.currentState, next);
238
+ this.setState(next);
239
+ }
240
+
241
+ private setState(next: EngineState): void {
242
+ if (this.currentState === next) return;
243
+ this.currentState = next;
244
+ this.emitter.emit({ type: 'state', state: next });
245
+ }
246
+
247
+ private toRecorderResult(r: NativeRecorderResult): RecorderResult {
248
+ return {
249
+ uri: r.uri,
250
+ durationMs: r.durationMs,
251
+ sizeBytes: r.sizeBytes,
252
+ container: 'm4a',
253
+ audioFormat: {
254
+ sampleRate: r.sampleRate as RecorderResult['audioFormat']['sampleRate'],
255
+ channels: r.channels as RecorderResult['audioFormat']['channels'],
256
+ bitDepth: r.bitDepth as RecorderResult['audioFormat']['bitDepth'],
257
+ },
258
+ };
259
+ }
260
+
261
+ private toClarionError(err: NativeRecorderError): ClarionError {
262
+ return new ClarionError({
263
+ code: this.mapErrorCode(err.code),
264
+ message: err.message,
265
+ recoverable: err.recoverable,
266
+ });
267
+ }
268
+
269
+ private mapErrorCode(
270
+ code: string,
271
+ ): ClarionError['code'] {
272
+ const known: Record<string, ClarionError['code']> = {
273
+ PERMISSION_DENIED: 'PERMISSION_DENIED',
274
+ AUDIO_BUSY: 'AUDIO_BUSY',
275
+ IO_ERROR: 'IO_ERROR',
276
+ INTERRUPTED: 'INTERRUPTED',
277
+ CANCELLED: 'CANCELLED',
278
+ ENGINE_NOT_READY: 'ENGINE_NOT_READY',
279
+ INVALID_STATE: 'INVALID_STATE',
280
+ UNSUPPORTED_FORMAT: 'UNSUPPORTED_FORMAT',
281
+ INTERNAL_ERROR: 'INTERNAL_ERROR',
282
+ };
283
+ return known[code] ?? 'UNKNOWN';
284
+ }
285
+
286
+ private handleNativeError(err: unknown, where: string): void {
287
+ const error =
288
+ err instanceof ClarionError
289
+ ? err
290
+ : new ClarionError({
291
+ code: 'INTERNAL_ERROR',
292
+ message: `Recorder ${where} failed: ${String(err)}`,
293
+ cause: err,
294
+ });
295
+ this.emitter.emit({ type: 'error', error });
296
+ this.setState('error');
297
+ }
298
+ }
package/src/index.ts ADDED
@@ -0,0 +1,8 @@
1
+ export { RecorderEngine } from './RecorderEngine';
2
+ export type { RecorderEngineOptions } from './RecorderEngine';
3
+ export type {
4
+ ClarionRecorder,
5
+ NativeRecorderConfig,
6
+ NativeRecorderResult,
7
+ NativeRecorderError,
8
+ } from './specs/ClarionRecorder.nitro';
package/src/native.ts ADDED
@@ -0,0 +1,5 @@
1
+ import { NitroModules } from 'react-native-nitro-modules';
2
+ import type { ClarionRecorder } from './specs/ClarionRecorder.nitro';
3
+
4
+ export const createNativeRecorder = (): ClarionRecorder =>
5
+ NitroModules.createHybridObject<ClarionRecorder>('ClarionRecorder');
@@ -0,0 +1,58 @@
1
+ import type { HybridObject } from 'react-native-nitro-modules';
2
+
3
+ export interface NativeRecorderConfig {
4
+ sampleRate: number;
5
+ channels: number;
6
+ bitDepth: number;
7
+ outputDirectory?: string;
8
+ filenamePrefix?: string;
9
+ rotateAfterMs?: number;
10
+ emitAudioLevel: boolean;
11
+ audioLevelIntervalMs: number;
12
+ aacBitrate: number;
13
+ }
14
+
15
+ export interface NativeRecorderResult {
16
+ uri: string;
17
+ durationMs: number;
18
+ sizeBytes: number;
19
+ sampleRate: number;
20
+ channels: number;
21
+ bitDepth: number;
22
+ }
23
+
24
+ export interface NativeRecorderError {
25
+ code: string;
26
+ message: string;
27
+ recoverable: boolean;
28
+ }
29
+
30
+ export type StateListener = (state: string) => void;
31
+ export type AudioLevelListener = (rms: number, peak: number) => void;
32
+ export type ChunkListener = (
33
+ uri: string,
34
+ startMs: number,
35
+ endMs: number,
36
+ sizeBytes: number,
37
+ ) => void;
38
+ export type ErrorListener = (error: NativeRecorderError) => void;
39
+
40
+ export interface ClarionRecorder
41
+ extends HybridObject<{ ios: 'swift'; android: 'kotlin' }> {
42
+ readonly state: string;
43
+
44
+ prepare(config: NativeRecorderConfig): Promise<void>;
45
+ start(): Promise<void>;
46
+ pause(): Promise<void>;
47
+ resume(): Promise<void>;
48
+ stop(): Promise<NativeRecorderResult>;
49
+ discard(): Promise<void>;
50
+ release(): Promise<void>;
51
+
52
+ addStateListener(listener: StateListener): number;
53
+ addAudioLevelListener(listener: AudioLevelListener): number;
54
+ addChunkListener(listener: ChunkListener): number;
55
+ addErrorListener(listener: ErrorListener): number;
56
+ removeListener(id: number): void;
57
+ removeAllListeners(): void;
58
+ }
package/index.js DELETED
@@ -1 +0,0 @@
1
- // @clarionhq/recorder — placeholder. See https://github.com/clarionhq/clarion