react-native-nitro-audio-record 0.1.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/LICENSE +21 -0
- package/NitroAudioRecord.podspec +33 -0
- package/README.md +92 -0
- package/android/CMakeLists.txt +32 -0
- package/android/build.gradle +148 -0
- package/android/fix-prefab.gradle +51 -0
- package/android/gradle.properties +5 -0
- package/android/src/main/AndroidManifest.xml +2 -0
- package/android/src/main/cpp/cpp-adapter.cpp +9 -0
- package/android/src/main/java/com/nitroaudiorecord/HybridNitroAudioRecord.kt +189 -0
- package/android/src/main/java/com/nitroaudiorecord/NitroAudioRecordPackage.kt +20 -0
- package/ios/Bridge.h +8 -0
- package/ios/HybridNitroAudioRecord.swift +108 -0
- package/lib/commonjs/index.js +9 -0
- package/lib/commonjs/index.js.map +1 -0
- package/lib/commonjs/package.json +1 -0
- package/lib/commonjs/specs/nitro-audio-record.nitro.js +6 -0
- package/lib/commonjs/specs/nitro-audio-record.nitro.js.map +1 -0
- package/lib/module/index.js +5 -0
- package/lib/module/index.js.map +1 -0
- package/lib/module/specs/nitro-audio-record.nitro.js +4 -0
- package/lib/module/specs/nitro-audio-record.nitro.js.map +1 -0
- package/lib/typescript/src/index.d.ts +3 -0
- package/lib/typescript/src/index.d.ts.map +1 -0
- package/lib/typescript/src/specs/nitro-audio-record.nitro.d.ts +18 -0
- package/lib/typescript/src/specs/nitro-audio-record.nitro.d.ts.map +1 -0
- package/nitro.json +30 -0
- package/nitrogen/generated/.gitattributes +1 -0
- package/nitrogen/generated/android/NitroAudioRecord+autolinking.cmake +81 -0
- package/nitrogen/generated/android/NitroAudioRecord+autolinking.gradle +27 -0
- package/nitrogen/generated/android/NitroAudioRecordOnLoad.cpp +56 -0
- package/nitrogen/generated/android/NitroAudioRecordOnLoad.hpp +34 -0
- package/nitrogen/generated/android/c++/JAudioRecordOptions.hpp +74 -0
- package/nitrogen/generated/android/c++/JFunc_void_std__shared_ptr_ArrayBuffer_.hpp +77 -0
- package/nitrogen/generated/android/c++/JHybridNitroAudioRecordSpec.cpp +87 -0
- package/nitrogen/generated/android/c++/JHybridNitroAudioRecordSpec.hpp +66 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/nitroaudiorecord/AudioRecordOptions.kt +50 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/nitroaudiorecord/Func_void_std__shared_ptr_ArrayBuffer_.kt +80 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/nitroaudiorecord/HybridNitroAudioRecordSpec.kt +73 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/nitroaudiorecord/NitroAudioRecordOnLoad.kt +35 -0
- package/nitrogen/generated/ios/NitroAudioRecord+autolinking.rb +60 -0
- package/nitrogen/generated/ios/NitroAudioRecord-Swift-Cxx-Bridge.cpp +57 -0
- package/nitrogen/generated/ios/NitroAudioRecord-Swift-Cxx-Bridge.hpp +177 -0
- package/nitrogen/generated/ios/NitroAudioRecord-Swift-Cxx-Umbrella.hpp +51 -0
- package/nitrogen/generated/ios/NitroAudioRecordAutolinking.mm +33 -0
- package/nitrogen/generated/ios/NitroAudioRecordAutolinking.swift +26 -0
- package/nitrogen/generated/ios/c++/HybridNitroAudioRecordSpecSwift.cpp +11 -0
- package/nitrogen/generated/ios/c++/HybridNitroAudioRecordSpecSwift.hpp +109 -0
- package/nitrogen/generated/ios/swift/AudioRecordOptions.swift +75 -0
- package/nitrogen/generated/ios/swift/Func_void_std__exception_ptr.swift +46 -0
- package/nitrogen/generated/ios/swift/Func_void_std__shared_ptr_ArrayBuffer_.swift +46 -0
- package/nitrogen/generated/ios/swift/Func_void_std__string.swift +46 -0
- package/nitrogen/generated/ios/swift/HybridNitroAudioRecordSpec.swift +58 -0
- package/nitrogen/generated/ios/swift/HybridNitroAudioRecordSpec_cxx.swift +183 -0
- package/nitrogen/generated/shared/c++/AudioRecordOptions.hpp +100 -0
- package/nitrogen/generated/shared/c++/HybridNitroAudioRecordSpec.cpp +24 -0
- package/nitrogen/generated/shared/c++/HybridNitroAudioRecordSpec.hpp +70 -0
- package/package.json +123 -0
- package/src/index.ts +5 -0
- package/src/specs/nitro-audio-record.nitro.ts +17 -0
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import Foundation
|
|
2
|
+
import NitroModules
|
|
3
|
+
import AVFoundation
|
|
4
|
+
import AudioToolbox
|
|
5
|
+
|
|
6
|
+
struct AQRecordState {
|
|
7
|
+
var mDataFormat = AudioStreamBasicDescription()
|
|
8
|
+
var mQueue: AudioQueueRef?
|
|
9
|
+
var mBuffers = [AudioQueueBufferRef?](repeating: nil, count: 3)
|
|
10
|
+
var mAudioFile: AudioFileID?
|
|
11
|
+
var bufferByteSize: UInt32 = 0
|
|
12
|
+
var mCurrentPacket: Int64 = 0
|
|
13
|
+
var mIsRunning = false
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
class HybridNitroAudioRecord: HybridNitroAudioRecordSpec {
|
|
17
|
+
private var recordState = AQRecordState()
|
|
18
|
+
private var filePath: String = ""
|
|
19
|
+
private var onDataCallback: ((ArrayBuffer) -> Void)?
|
|
20
|
+
|
|
21
|
+
func setup(options: AudioRecordOptions) throws {
|
|
22
|
+
let sampleRate = options.sampleRate
|
|
23
|
+
let bitsPerSample = UInt32(options.bitsPerSample)
|
|
24
|
+
let channels = UInt32(options.channels)
|
|
25
|
+
|
|
26
|
+
recordState.mDataFormat.mSampleRate = sampleRate
|
|
27
|
+
recordState.mDataFormat.mBitsPerChannel = bitsPerSample
|
|
28
|
+
recordState.mDataFormat.mChannelsPerFrame = channels
|
|
29
|
+
recordState.mDataFormat.mBytesPerPacket = (bitsPerSample / 8) * channels
|
|
30
|
+
recordState.mDataFormat.mBytesPerFrame = recordState.mDataFormat.mBytesPerPacket
|
|
31
|
+
recordState.mDataFormat.mFramesPerPacket = 1
|
|
32
|
+
recordState.mDataFormat.mReserved = 0
|
|
33
|
+
recordState.mDataFormat.mFormatID = kAudioFormatLinearPCM
|
|
34
|
+
recordState.mDataFormat.mFormatFlags = bitsPerSample == 8 ? kLinearPCMFormatFlagIsPacked : (kLinearPCMFormatFlagIsSignedInteger | kLinearPCMFormatFlagIsPacked)
|
|
35
|
+
|
|
36
|
+
recordState.bufferByteSize = 2048
|
|
37
|
+
|
|
38
|
+
let fileName = options.wavFile ?? "audio.wav"
|
|
39
|
+
let docDir = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first!
|
|
40
|
+
filePath = "\(docDir)/\(fileName)"
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
func start() throws {
|
|
44
|
+
// Set AudioSession
|
|
45
|
+
try AVAudioSession.sharedInstance().setCategory(.record)
|
|
46
|
+
try AVAudioSession.sharedInstance().setActive(true)
|
|
47
|
+
|
|
48
|
+
recordState.mIsRunning = true
|
|
49
|
+
recordState.mCurrentPacket = 0
|
|
50
|
+
|
|
51
|
+
let url = URL(fileURLWithPath: filePath)
|
|
52
|
+
let status = AudioFileCreateWithURL(url as CFURL, kAudioFileWAVEType, &recordState.mDataFormat, .eraseFile, &recordState.mAudioFile)
|
|
53
|
+
if status != noErr {
|
|
54
|
+
throw NSError(domain: "NitroAudioRecord", code: Int(status), userInfo: [NSLocalizedDescriptionKey: "Failed to create audio file"])
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
let selfPointer = Unmanaged.passUnretained(self).toOpaque()
|
|
58
|
+
|
|
59
|
+
AudioQueueNewInput(&recordState.mDataFormat, { (inUserData, inAQ, inBuffer, inStartTime, inNumPackets, inPacketDesc) in
|
|
60
|
+
let recorder = Unmanaged<HybridNitroAudioRecord>.fromOpaque(inUserData!).takeUnretainedValue()
|
|
61
|
+
|
|
62
|
+
if !recorder.recordState.mIsRunning { return }
|
|
63
|
+
|
|
64
|
+
var numPackets = inNumPackets
|
|
65
|
+
if AudioFileWritePackets(recorder.recordState.mAudioFile!, false, inBuffer.pointee.mAudioDataByteSize, inPacketDesc, recorder.recordState.mCurrentPacket, &numPackets, inBuffer.pointee.mAudioData) == noErr {
|
|
66
|
+
recorder.recordState.mCurrentPacket += Int64(numPackets)
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Notify callback
|
|
70
|
+
if let callback = recorder.onDataCallback {
|
|
71
|
+
let data = Data(bytes: inBuffer.pointee.mAudioData, count: Int(inBuffer.pointee.mAudioDataByteSize))
|
|
72
|
+
// Create ArrayBuffer from Data using the helper extension
|
|
73
|
+
do {
|
|
74
|
+
let arrayBuffer = try ArrayBuffer.copy(data: data)
|
|
75
|
+
callback(arrayBuffer)
|
|
76
|
+
} catch {
|
|
77
|
+
print("Failed to notify audio data: \(error)")
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
AudioQueueEnqueueBuffer(inAQ, inBuffer, 0, nil)
|
|
82
|
+
}, selfPointer, nil, nil, 0, &recordState.mQueue)
|
|
83
|
+
|
|
84
|
+
for i in 0..<3 {
|
|
85
|
+
AudioQueueAllocateBuffer(recordState.mQueue!, recordState.bufferByteSize, &recordState.mBuffers[i])
|
|
86
|
+
AudioQueueEnqueueBuffer(recordState.mQueue!, recordState.mBuffers[i]!, 0, nil)
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
AudioQueueStart(recordState.mQueue!, nil)
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
func stop() throws -> Promise<String> {
|
|
93
|
+
if recordState.mIsRunning {
|
|
94
|
+
recordState.mIsRunning = false
|
|
95
|
+
AudioQueueStop(recordState.mQueue!, true)
|
|
96
|
+
AudioQueueDispose(recordState.mQueue!, true)
|
|
97
|
+
AudioFileClose(recordState.mAudioFile!)
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
let promise = Promise<String>()
|
|
101
|
+
promise.resolve(withResult: filePath)
|
|
102
|
+
return promise
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
func onData(callback: @escaping (ArrayBuffer) -> Void) throws {
|
|
106
|
+
onDataCallback = callback
|
|
107
|
+
}
|
|
108
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.NitroAudioRecord = void 0;
|
|
7
|
+
var _reactNativeNitroModules = require("react-native-nitro-modules");
|
|
8
|
+
const NitroAudioRecord = exports.NitroAudioRecord = _reactNativeNitroModules.NitroModules.createHybridObject('NitroAudioRecord');
|
|
9
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["_reactNativeNitroModules","require","NitroAudioRecord","exports","NitroModules","createHybridObject"],"sourceRoot":"../../src","sources":["index.ts"],"mappings":";;;;;;AAAA,IAAAA,wBAAA,GAAAC,OAAA;AAGO,MAAMC,gBAAgB,GAAAC,OAAA,CAAAD,gBAAA,GAC3BE,qCAAY,CAACC,kBAAkB,CAAuB,kBAAkB,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"type":"commonjs"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":[],"sourceRoot":"../../../src","sources":["specs/nitro-audio-record.nitro.ts"],"mappings":"","ignoreList":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["NitroModules","NitroAudioRecord","createHybridObject"],"sourceRoot":"../../src","sources":["index.ts"],"mappings":";;AAAA,SAASA,YAAY,QAAQ,4BAA4B;AAGzD,OAAO,MAAMC,gBAAgB,GAC3BD,YAAY,CAACE,kBAAkB,CAAuB,kBAAkB,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":[],"sourceRoot":"../../../src","sources":["specs/nitro-audio-record.nitro.ts"],"mappings":"","ignoreList":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/index.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,gBAAgB,IAAI,oBAAoB,EAAE,MAAM,kCAAkC,CAAA;AAEhG,eAAO,MAAM,gBAAgB,sBAC8C,CAAA"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { HybridObject } from 'react-native-nitro-modules';
|
|
2
|
+
export interface AudioRecordOptions {
|
|
3
|
+
sampleRate: number;
|
|
4
|
+
channels: 1 | 2;
|
|
5
|
+
bitsPerSample: 8 | 16;
|
|
6
|
+
audioSource?: number;
|
|
7
|
+
wavFile?: string;
|
|
8
|
+
}
|
|
9
|
+
export interface NitroAudioRecord extends HybridObject<{
|
|
10
|
+
ios: 'swift';
|
|
11
|
+
android: 'kotlin';
|
|
12
|
+
}> {
|
|
13
|
+
setup(options: AudioRecordOptions): void;
|
|
14
|
+
start(): void;
|
|
15
|
+
stop(): Promise<string>;
|
|
16
|
+
onData(callback: (data: ArrayBuffer) => void): void;
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=nitro-audio-record.nitro.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"nitro-audio-record.nitro.d.ts","sourceRoot":"","sources":["../../../../src/specs/nitro-audio-record.nitro.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,4BAA4B,CAAA;AAE9D,MAAM,WAAW,kBAAkB;IACjC,UAAU,EAAE,MAAM,CAAA;IAClB,QAAQ,EAAE,CAAC,GAAG,CAAC,CAAA;IACf,aAAa,EAAE,CAAC,GAAG,EAAE,CAAA;IACrB,WAAW,CAAC,EAAE,MAAM,CAAA;IACpB,OAAO,CAAC,EAAE,MAAM,CAAA;CACjB;AAED,MAAM,WAAW,gBACf,SAAQ,YAAY,CAAC;IAAE,GAAG,EAAE,OAAO,CAAC;IAAC,OAAO,EAAE,QAAQ,CAAA;CAAE,CAAC;IACzD,KAAK,CAAC,OAAO,EAAE,kBAAkB,GAAG,IAAI,CAAA;IACxC,KAAK,IAAI,IAAI,CAAA;IACb,IAAI,IAAI,OAAO,CAAC,MAAM,CAAC,CAAA;IACvB,MAAM,CAAC,QAAQ,EAAE,CAAC,IAAI,EAAE,WAAW,KAAK,IAAI,GAAG,IAAI,CAAA;CACpD"}
|
package/nitro.json
ADDED
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
{
|
|
2
|
+
"$schema": "https://nitro.margelo.com/nitro.schema.json",
|
|
3
|
+
"cxxNamespace": [
|
|
4
|
+
"nitroaudiorecord"
|
|
5
|
+
],
|
|
6
|
+
"ios": {
|
|
7
|
+
"iosModuleName": "NitroAudioRecord"
|
|
8
|
+
},
|
|
9
|
+
"android": {
|
|
10
|
+
"androidNamespace": [
|
|
11
|
+
"nitroaudiorecord"
|
|
12
|
+
],
|
|
13
|
+
"androidCxxLibName": "NitroAudioRecord"
|
|
14
|
+
},
|
|
15
|
+
"autolinking": {
|
|
16
|
+
"NitroAudioRecord": {
|
|
17
|
+
"ios": {
|
|
18
|
+
"language": "swift",
|
|
19
|
+
"implementationClassName": "HybridNitroAudioRecord"
|
|
20
|
+
},
|
|
21
|
+
"android": {
|
|
22
|
+
"language": "kotlin",
|
|
23
|
+
"implementationClassName": "HybridNitroAudioRecord"
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
},
|
|
27
|
+
"ignorePaths": [
|
|
28
|
+
"**/node_modules"
|
|
29
|
+
]
|
|
30
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
** linguist-generated=true
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
#
|
|
2
|
+
# NitroAudioRecord+autolinking.cmake
|
|
3
|
+
# This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
# https://github.com/mrousavy/nitro
|
|
5
|
+
# Copyright © Marc Rousavy @ Margelo
|
|
6
|
+
#
|
|
7
|
+
|
|
8
|
+
# This is a CMake file that adds all files generated by Nitrogen
|
|
9
|
+
# to the current CMake project.
|
|
10
|
+
#
|
|
11
|
+
# To use it, add this to your CMakeLists.txt:
|
|
12
|
+
# ```cmake
|
|
13
|
+
# include(${CMAKE_SOURCE_DIR}/../nitrogen/generated/android/NitroAudioRecord+autolinking.cmake)
|
|
14
|
+
# ```
|
|
15
|
+
|
|
16
|
+
# Define a flag to check if we are building properly
|
|
17
|
+
add_definitions(-DBUILDING_NITROAUDIORECORD_WITH_GENERATED_CMAKE_PROJECT)
|
|
18
|
+
|
|
19
|
+
# Enable Raw Props parsing in react-native (for Nitro Views)
|
|
20
|
+
add_definitions(-DRN_SERIALIZABLE_STATE)
|
|
21
|
+
|
|
22
|
+
# Add all headers that were generated by Nitrogen
|
|
23
|
+
include_directories(
|
|
24
|
+
"../nitrogen/generated/shared/c++"
|
|
25
|
+
"../nitrogen/generated/android/c++"
|
|
26
|
+
"../nitrogen/generated/android/"
|
|
27
|
+
)
|
|
28
|
+
|
|
29
|
+
# Add all .cpp sources that were generated by Nitrogen
|
|
30
|
+
target_sources(
|
|
31
|
+
# CMake project name (Android C++ library name)
|
|
32
|
+
NitroAudioRecord PRIVATE
|
|
33
|
+
# Autolinking Setup
|
|
34
|
+
../nitrogen/generated/android/NitroAudioRecordOnLoad.cpp
|
|
35
|
+
# Shared Nitrogen C++ sources
|
|
36
|
+
../nitrogen/generated/shared/c++/HybridNitroAudioRecordSpec.cpp
|
|
37
|
+
# Android-specific Nitrogen C++ sources
|
|
38
|
+
../nitrogen/generated/android/c++/JHybridNitroAudioRecordSpec.cpp
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
# From node_modules/react-native/ReactAndroid/cmake-utils/folly-flags.cmake
|
|
42
|
+
# Used in node_modules/react-native/ReactAndroid/cmake-utils/ReactNative-application.cmake
|
|
43
|
+
target_compile_definitions(
|
|
44
|
+
NitroAudioRecord PRIVATE
|
|
45
|
+
-DFOLLY_NO_CONFIG=1
|
|
46
|
+
-DFOLLY_HAVE_CLOCK_GETTIME=1
|
|
47
|
+
-DFOLLY_USE_LIBCPP=1
|
|
48
|
+
-DFOLLY_CFG_NO_COROUTINES=1
|
|
49
|
+
-DFOLLY_MOBILE=1
|
|
50
|
+
-DFOLLY_HAVE_RECVMMSG=1
|
|
51
|
+
-DFOLLY_HAVE_PTHREAD=1
|
|
52
|
+
# Once we target android-23 above, we can comment
|
|
53
|
+
# the following line. NDK uses GNU style stderror_r() after API 23.
|
|
54
|
+
-DFOLLY_HAVE_XSI_STRERROR_R=1
|
|
55
|
+
)
|
|
56
|
+
|
|
57
|
+
# Add all libraries required by the generated specs
|
|
58
|
+
find_package(fbjni REQUIRED) # <-- Used for communication between Java <-> C++
|
|
59
|
+
find_package(ReactAndroid REQUIRED) # <-- Used to set up React Native bindings (e.g. CallInvoker/TurboModule)
|
|
60
|
+
find_package(react-native-nitro-modules REQUIRED) # <-- Used to create all HybridObjects and use the Nitro core library
|
|
61
|
+
|
|
62
|
+
# Link all libraries together
|
|
63
|
+
target_link_libraries(
|
|
64
|
+
NitroAudioRecord
|
|
65
|
+
fbjni::fbjni # <-- Facebook C++ JNI helpers
|
|
66
|
+
ReactAndroid::jsi # <-- RN: JSI
|
|
67
|
+
react-native-nitro-modules::NitroModules # <-- NitroModules Core :)
|
|
68
|
+
)
|
|
69
|
+
|
|
70
|
+
# Link react-native (different prefab between RN 0.75 and RN 0.76)
|
|
71
|
+
if(ReactAndroid_VERSION_MINOR GREATER_EQUAL 76)
|
|
72
|
+
target_link_libraries(
|
|
73
|
+
NitroAudioRecord
|
|
74
|
+
ReactAndroid::reactnative # <-- RN: Native Modules umbrella prefab
|
|
75
|
+
)
|
|
76
|
+
else()
|
|
77
|
+
target_link_libraries(
|
|
78
|
+
NitroAudioRecord
|
|
79
|
+
ReactAndroid::react_nativemodule_core # <-- RN: TurboModules Core
|
|
80
|
+
)
|
|
81
|
+
endif()
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// NitroAudioRecord+autolinking.gradle
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
/// This is a Gradle file that adds all files generated by Nitrogen
|
|
9
|
+
/// to the current Gradle project.
|
|
10
|
+
///
|
|
11
|
+
/// To use it, add this to your build.gradle:
|
|
12
|
+
/// ```gradle
|
|
13
|
+
/// apply from: '../nitrogen/generated/android/NitroAudioRecord+autolinking.gradle'
|
|
14
|
+
/// ```
|
|
15
|
+
|
|
16
|
+
logger.warn("[NitroModules] 🔥 NitroAudioRecord is boosted by nitro!")
|
|
17
|
+
|
|
18
|
+
android {
|
|
19
|
+
sourceSets {
|
|
20
|
+
main {
|
|
21
|
+
java.srcDirs += [
|
|
22
|
+
// Nitrogen files
|
|
23
|
+
"${project.projectDir}/../nitrogen/generated/android/kotlin"
|
|
24
|
+
]
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// NitroAudioRecordOnLoad.cpp
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
#ifndef BUILDING_NITROAUDIORECORD_WITH_GENERATED_CMAKE_PROJECT
|
|
9
|
+
#error NitroAudioRecordOnLoad.cpp is not being built with the autogenerated CMakeLists.txt project. Is a different CMakeLists.txt building this?
|
|
10
|
+
#endif
|
|
11
|
+
|
|
12
|
+
#include "NitroAudioRecordOnLoad.hpp"
|
|
13
|
+
|
|
14
|
+
#include <jni.h>
|
|
15
|
+
#include <fbjni/fbjni.h>
|
|
16
|
+
#include <NitroModules/HybridObjectRegistry.hpp>
|
|
17
|
+
|
|
18
|
+
#include "JHybridNitroAudioRecordSpec.hpp"
|
|
19
|
+
#include "JFunc_void_std__shared_ptr_ArrayBuffer_.hpp"
|
|
20
|
+
#include <NitroModules/DefaultConstructableObject.hpp>
|
|
21
|
+
|
|
22
|
+
namespace margelo::nitro::nitroaudiorecord {
|
|
23
|
+
|
|
24
|
+
int initialize(JavaVM* vm) {
|
|
25
|
+
return facebook::jni::initialize(vm, []() {
|
|
26
|
+
::margelo::nitro::nitroaudiorecord::registerAllNatives();
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
struct JHybridNitroAudioRecordSpecImpl: public jni::JavaClass<JHybridNitroAudioRecordSpecImpl, JHybridNitroAudioRecordSpec::JavaPart> {
|
|
31
|
+
static constexpr auto kJavaDescriptor = "Lcom/nitroaudiorecord/HybridNitroAudioRecord;";
|
|
32
|
+
static std::shared_ptr<JHybridNitroAudioRecordSpec> create() {
|
|
33
|
+
static const auto constructorFn = javaClassStatic()->getConstructor<JHybridNitroAudioRecordSpecImpl::javaobject()>();
|
|
34
|
+
jni::local_ref<JHybridNitroAudioRecordSpec::JavaPart> javaPart = javaClassStatic()->newObject(constructorFn);
|
|
35
|
+
return javaPart->getJHybridNitroAudioRecordSpec();
|
|
36
|
+
}
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
void registerAllNatives() {
|
|
40
|
+
using namespace margelo::nitro;
|
|
41
|
+
using namespace margelo::nitro::nitroaudiorecord;
|
|
42
|
+
|
|
43
|
+
// Register native JNI methods
|
|
44
|
+
margelo::nitro::nitroaudiorecord::JHybridNitroAudioRecordSpec::CxxPart::registerNatives();
|
|
45
|
+
margelo::nitro::nitroaudiorecord::JFunc_void_std__shared_ptr_ArrayBuffer__cxx::registerNatives();
|
|
46
|
+
|
|
47
|
+
// Register Nitro Hybrid Objects
|
|
48
|
+
HybridObjectRegistry::registerHybridObjectConstructor(
|
|
49
|
+
"NitroAudioRecord",
|
|
50
|
+
[]() -> std::shared_ptr<HybridObject> {
|
|
51
|
+
return JHybridNitroAudioRecordSpecImpl::create();
|
|
52
|
+
}
|
|
53
|
+
);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
} // namespace margelo::nitro::nitroaudiorecord
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// NitroAudioRecordOnLoad.hpp
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
#include <jni.h>
|
|
9
|
+
#include <functional>
|
|
10
|
+
#include <NitroModules/NitroDefines.hpp>
|
|
11
|
+
|
|
12
|
+
namespace margelo::nitro::nitroaudiorecord {
|
|
13
|
+
|
|
14
|
+
[[deprecated("Use registerNatives() instead.")]]
|
|
15
|
+
int initialize(JavaVM* vm);
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Register the native (C++) part of NitroAudioRecord, and autolinks all Hybrid Objects.
|
|
19
|
+
* Call this in your `JNI_OnLoad` function (probably inside `cpp-adapter.cpp`),
|
|
20
|
+
* inside a `facebook::jni::initialize(vm, ...)` call.
|
|
21
|
+
* Example:
|
|
22
|
+
* ```cpp (cpp-adapter.cpp)
|
|
23
|
+
* JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void*) {
|
|
24
|
+
* return facebook::jni::initialize(vm, []() {
|
|
25
|
+
* // register all NitroAudioRecord HybridObjects
|
|
26
|
+
* margelo::nitro::nitroaudiorecord::registerNatives();
|
|
27
|
+
* // any other custom registrations go here.
|
|
28
|
+
* });
|
|
29
|
+
* }
|
|
30
|
+
* ```
|
|
31
|
+
*/
|
|
32
|
+
void registerAllNatives();
|
|
33
|
+
|
|
34
|
+
} // namespace margelo::nitro::nitroaudiorecord
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// JAudioRecordOptions.hpp
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
#pragma once
|
|
9
|
+
|
|
10
|
+
#include <fbjni/fbjni.h>
|
|
11
|
+
#include "AudioRecordOptions.hpp"
|
|
12
|
+
|
|
13
|
+
#include <optional>
|
|
14
|
+
#include <string>
|
|
15
|
+
|
|
16
|
+
namespace margelo::nitro::nitroaudiorecord {
|
|
17
|
+
|
|
18
|
+
using namespace facebook;
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* The C++ JNI bridge between the C++ struct "AudioRecordOptions" and the the Kotlin data class "AudioRecordOptions".
|
|
22
|
+
*/
|
|
23
|
+
struct JAudioRecordOptions final: public jni::JavaClass<JAudioRecordOptions> {
|
|
24
|
+
public:
|
|
25
|
+
static constexpr auto kJavaDescriptor = "Lcom/margelo/nitro/nitroaudiorecord/AudioRecordOptions;";
|
|
26
|
+
|
|
27
|
+
public:
|
|
28
|
+
/**
|
|
29
|
+
* Convert this Java/Kotlin-based struct to the C++ struct AudioRecordOptions by copying all values to C++.
|
|
30
|
+
*/
|
|
31
|
+
[[maybe_unused]]
|
|
32
|
+
[[nodiscard]]
|
|
33
|
+
AudioRecordOptions toCpp() const {
|
|
34
|
+
static const auto clazz = javaClassStatic();
|
|
35
|
+
static const auto fieldSampleRate = clazz->getField<double>("sampleRate");
|
|
36
|
+
double sampleRate = this->getFieldValue(fieldSampleRate);
|
|
37
|
+
static const auto fieldChannels = clazz->getField<double>("channels");
|
|
38
|
+
double channels = this->getFieldValue(fieldChannels);
|
|
39
|
+
static const auto fieldBitsPerSample = clazz->getField<double>("bitsPerSample");
|
|
40
|
+
double bitsPerSample = this->getFieldValue(fieldBitsPerSample);
|
|
41
|
+
static const auto fieldAudioSource = clazz->getField<jni::JDouble>("audioSource");
|
|
42
|
+
jni::local_ref<jni::JDouble> audioSource = this->getFieldValue(fieldAudioSource);
|
|
43
|
+
static const auto fieldWavFile = clazz->getField<jni::JString>("wavFile");
|
|
44
|
+
jni::local_ref<jni::JString> wavFile = this->getFieldValue(fieldWavFile);
|
|
45
|
+
return AudioRecordOptions(
|
|
46
|
+
sampleRate,
|
|
47
|
+
channels,
|
|
48
|
+
bitsPerSample,
|
|
49
|
+
audioSource != nullptr ? std::make_optional(audioSource->value()) : std::nullopt,
|
|
50
|
+
wavFile != nullptr ? std::make_optional(wavFile->toStdString()) : std::nullopt
|
|
51
|
+
);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
public:
|
|
55
|
+
/**
|
|
56
|
+
* Create a Java/Kotlin-based struct by copying all values from the given C++ struct to Java.
|
|
57
|
+
*/
|
|
58
|
+
[[maybe_unused]]
|
|
59
|
+
static jni::local_ref<JAudioRecordOptions::javaobject> fromCpp(const AudioRecordOptions& value) {
|
|
60
|
+
using JSignature = JAudioRecordOptions(double, double, double, jni::alias_ref<jni::JDouble>, jni::alias_ref<jni::JString>);
|
|
61
|
+
static const auto clazz = javaClassStatic();
|
|
62
|
+
static const auto create = clazz->getStaticMethod<JSignature>("fromCpp");
|
|
63
|
+
return create(
|
|
64
|
+
clazz,
|
|
65
|
+
value.sampleRate,
|
|
66
|
+
value.channels,
|
|
67
|
+
value.bitsPerSample,
|
|
68
|
+
value.audioSource.has_value() ? jni::JDouble::valueOf(value.audioSource.value()) : nullptr,
|
|
69
|
+
value.wavFile.has_value() ? jni::make_jstring(value.wavFile.value()) : nullptr
|
|
70
|
+
);
|
|
71
|
+
}
|
|
72
|
+
};
|
|
73
|
+
|
|
74
|
+
} // namespace margelo::nitro::nitroaudiorecord
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// JFunc_void_std__shared_ptr_ArrayBuffer_.hpp
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
#pragma once
|
|
9
|
+
|
|
10
|
+
#include <fbjni/fbjni.h>
|
|
11
|
+
#include <functional>
|
|
12
|
+
|
|
13
|
+
#include <NitroModules/ArrayBuffer.hpp>
|
|
14
|
+
#include <functional>
|
|
15
|
+
#include <NitroModules/JNICallable.hpp>
|
|
16
|
+
#include <NitroModules/JArrayBuffer.hpp>
|
|
17
|
+
|
|
18
|
+
namespace margelo::nitro::nitroaudiorecord {
|
|
19
|
+
|
|
20
|
+
using namespace facebook;
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Represents the Java/Kotlin callback `(data: ArrayBuffer) -> Unit`.
|
|
24
|
+
* This can be passed around between C++ and Java/Kotlin.
|
|
25
|
+
*/
|
|
26
|
+
struct JFunc_void_std__shared_ptr_ArrayBuffer_: public jni::JavaClass<JFunc_void_std__shared_ptr_ArrayBuffer_> {
|
|
27
|
+
public:
|
|
28
|
+
static constexpr auto kJavaDescriptor = "Lcom/margelo/nitro/nitroaudiorecord/Func_void_std__shared_ptr_ArrayBuffer_;";
|
|
29
|
+
|
|
30
|
+
public:
|
|
31
|
+
/**
|
|
32
|
+
* Invokes the function this `JFunc_void_std__shared_ptr_ArrayBuffer_` instance holds through JNI.
|
|
33
|
+
*/
|
|
34
|
+
void invoke(const std::shared_ptr<ArrayBuffer>& data) const {
|
|
35
|
+
static const auto method = javaClassStatic()->getMethod<void(jni::alias_ref<JArrayBuffer::javaobject> /* data */)>("invoke");
|
|
36
|
+
method(self(), JArrayBuffer::wrap(data));
|
|
37
|
+
}
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* An implementation of Func_void_std__shared_ptr_ArrayBuffer_ that is backed by a C++ implementation (using `std::function<...>`)
|
|
42
|
+
*/
|
|
43
|
+
class JFunc_void_std__shared_ptr_ArrayBuffer__cxx final: public jni::HybridClass<JFunc_void_std__shared_ptr_ArrayBuffer__cxx, JFunc_void_std__shared_ptr_ArrayBuffer_> {
|
|
44
|
+
public:
|
|
45
|
+
static jni::local_ref<JFunc_void_std__shared_ptr_ArrayBuffer_::javaobject> fromCpp(const std::function<void(const std::shared_ptr<ArrayBuffer>& /* data */)>& func) {
|
|
46
|
+
return JFunc_void_std__shared_ptr_ArrayBuffer__cxx::newObjectCxxArgs(func);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
public:
|
|
50
|
+
/**
|
|
51
|
+
* Invokes the C++ `std::function<...>` this `JFunc_void_std__shared_ptr_ArrayBuffer__cxx` instance holds.
|
|
52
|
+
*/
|
|
53
|
+
void invoke_cxx(jni::alias_ref<JArrayBuffer::javaobject> data) {
|
|
54
|
+
_func(data->cthis()->getArrayBuffer());
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
public:
|
|
58
|
+
[[nodiscard]]
|
|
59
|
+
inline const std::function<void(const std::shared_ptr<ArrayBuffer>& /* data */)>& getFunction() const {
|
|
60
|
+
return _func;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
public:
|
|
64
|
+
static constexpr auto kJavaDescriptor = "Lcom/margelo/nitro/nitroaudiorecord/Func_void_std__shared_ptr_ArrayBuffer__cxx;";
|
|
65
|
+
static void registerNatives() {
|
|
66
|
+
registerHybrid({makeNativeMethod("invoke_cxx", JFunc_void_std__shared_ptr_ArrayBuffer__cxx::invoke_cxx)});
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
private:
|
|
70
|
+
explicit JFunc_void_std__shared_ptr_ArrayBuffer__cxx(const std::function<void(const std::shared_ptr<ArrayBuffer>& /* data */)>& func): _func(func) { }
|
|
71
|
+
|
|
72
|
+
private:
|
|
73
|
+
friend HybridBase;
|
|
74
|
+
std::function<void(const std::shared_ptr<ArrayBuffer>& /* data */)> _func;
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
} // namespace margelo::nitro::nitroaudiorecord
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// JHybridNitroAudioRecordSpec.cpp
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
#include "JHybridNitroAudioRecordSpec.hpp"
|
|
9
|
+
|
|
10
|
+
// Forward declaration of `AudioRecordOptions` to properly resolve imports.
|
|
11
|
+
namespace margelo::nitro::nitroaudiorecord { struct AudioRecordOptions; }
|
|
12
|
+
|
|
13
|
+
#include <string>
|
|
14
|
+
#include <NitroModules/Promise.hpp>
|
|
15
|
+
#include <NitroModules/JPromise.hpp>
|
|
16
|
+
#include "AudioRecordOptions.hpp"
|
|
17
|
+
#include "JAudioRecordOptions.hpp"
|
|
18
|
+
#include <optional>
|
|
19
|
+
#include <NitroModules/ArrayBuffer.hpp>
|
|
20
|
+
#include <functional>
|
|
21
|
+
#include "JFunc_void_std__shared_ptr_ArrayBuffer_.hpp"
|
|
22
|
+
#include <NitroModules/JNICallable.hpp>
|
|
23
|
+
#include <NitroModules/JArrayBuffer.hpp>
|
|
24
|
+
|
|
25
|
+
namespace margelo::nitro::nitroaudiorecord {
|
|
26
|
+
|
|
27
|
+
std::shared_ptr<JHybridNitroAudioRecordSpec> JHybridNitroAudioRecordSpec::JavaPart::getJHybridNitroAudioRecordSpec() {
|
|
28
|
+
auto hybridObject = JHybridObject::JavaPart::getJHybridObject();
|
|
29
|
+
auto castHybridObject = std::dynamic_pointer_cast<JHybridNitroAudioRecordSpec>(hybridObject);
|
|
30
|
+
if (castHybridObject == nullptr) [[unlikely]] {
|
|
31
|
+
throw std::runtime_error("Failed to downcast JHybridObject to JHybridNitroAudioRecordSpec!");
|
|
32
|
+
}
|
|
33
|
+
return castHybridObject;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
jni::local_ref<JHybridNitroAudioRecordSpec::CxxPart::jhybriddata> JHybridNitroAudioRecordSpec::CxxPart::initHybrid(jni::alias_ref<jhybridobject> jThis) {
|
|
37
|
+
return makeCxxInstance(jThis);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
std::shared_ptr<JHybridObject> JHybridNitroAudioRecordSpec::CxxPart::createHybridObject(const jni::local_ref<JHybridObject::JavaPart>& javaPart) {
|
|
41
|
+
auto castJavaPart = jni::dynamic_ref_cast<JHybridNitroAudioRecordSpec::JavaPart>(javaPart);
|
|
42
|
+
if (castJavaPart == nullptr) [[unlikely]] {
|
|
43
|
+
throw std::runtime_error("Failed to cast JHybridObject::JavaPart to JHybridNitroAudioRecordSpec::JavaPart!");
|
|
44
|
+
}
|
|
45
|
+
return std::make_shared<JHybridNitroAudioRecordSpec>(castJavaPart);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
void JHybridNitroAudioRecordSpec::CxxPart::registerNatives() {
|
|
49
|
+
registerHybrid({
|
|
50
|
+
makeNativeMethod("initHybrid", JHybridNitroAudioRecordSpec::CxxPart::initHybrid),
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// Properties
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
// Methods
|
|
58
|
+
void JHybridNitroAudioRecordSpec::setup(const AudioRecordOptions& options) {
|
|
59
|
+
static const auto method = _javaPart->javaClassStatic()->getMethod<void(jni::alias_ref<JAudioRecordOptions> /* options */)>("setup");
|
|
60
|
+
method(_javaPart, JAudioRecordOptions::fromCpp(options));
|
|
61
|
+
}
|
|
62
|
+
void JHybridNitroAudioRecordSpec::start() {
|
|
63
|
+
static const auto method = _javaPart->javaClassStatic()->getMethod<void()>("start");
|
|
64
|
+
method(_javaPart);
|
|
65
|
+
}
|
|
66
|
+
std::shared_ptr<Promise<std::string>> JHybridNitroAudioRecordSpec::stop() {
|
|
67
|
+
static const auto method = _javaPart->javaClassStatic()->getMethod<jni::local_ref<JPromise::javaobject>()>("stop");
|
|
68
|
+
auto __result = method(_javaPart);
|
|
69
|
+
return [&]() {
|
|
70
|
+
auto __promise = Promise<std::string>::create();
|
|
71
|
+
__result->cthis()->addOnResolvedListener([=](const jni::alias_ref<jni::JObject>& __boxedResult) {
|
|
72
|
+
auto __result = jni::static_ref_cast<jni::JString>(__boxedResult);
|
|
73
|
+
__promise->resolve(__result->toStdString());
|
|
74
|
+
});
|
|
75
|
+
__result->cthis()->addOnRejectedListener([=](const jni::alias_ref<jni::JThrowable>& __throwable) {
|
|
76
|
+
jni::JniException __jniError(__throwable);
|
|
77
|
+
__promise->reject(std::make_exception_ptr(__jniError));
|
|
78
|
+
});
|
|
79
|
+
return __promise;
|
|
80
|
+
}();
|
|
81
|
+
}
|
|
82
|
+
void JHybridNitroAudioRecordSpec::onData(const std::function<void(const std::shared_ptr<ArrayBuffer>& /* data */)>& callback) {
|
|
83
|
+
static const auto method = _javaPart->javaClassStatic()->getMethod<void(jni::alias_ref<JFunc_void_std__shared_ptr_ArrayBuffer_::javaobject> /* callback */)>("onData_cxx");
|
|
84
|
+
method(_javaPart, JFunc_void_std__shared_ptr_ArrayBuffer__cxx::fromCpp(callback));
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
} // namespace margelo::nitro::nitroaudiorecord
|