聲臨其境,輕鬆幾步教你把音訊變成3D環繞音

HMS Core發表於2021-12-14

在音樂創作、音視訊剪輯和遊戲等領域中,給使用者帶來沉浸式音訊體驗越來越重要。開發者如何在應用內打造3D環繞聲效?華為音訊編輯服務6.2.0版本此次帶來了空間動態渲染功能,可以將人聲、樂器等音訊元素渲染到指定的三維空間方位,支援靜態和動態渲染兩種模式,進一步提升應用中的音效體驗。開發者可以點選檢視以下Demo演示,瞭解整合效果並上手實驗功能特性。

開發實戰

1. 開發準備

開發者提前準備音樂素材,MP3格式最佳。其他音訊格式請參考“2.4”步驟轉換,視訊格式請參考“2.5”步驟進行音訊提取。

1.1專案級build.gradle裡配置Maven倉地址

buildscript {
    repositories {
        google()
        jcenter()
        // 配置HMS Core SDK的Maven倉地址。
        maven {url 'https://developer.huawei.com/repo/'}
    }
    dependencies {
        ...
        // 增加agcp外掛配置。
        classpath 'com.huawei.agconnect:agcp:1.4.2.300'
    }
}
allprojects {
    repositories {
        google()
        jcenter()
        // 配置HMS Core SDK的Maven倉地址。
        maven {url 'https://developer.huawei.com/repo/'}
    }
} 

1.2 檔案頭增加配置

apply plugin: 'com.huawei.agconnect'

1.3 應用級build.gradle裡配置SDK依賴

dependencies{
    implementation 'com.huawei.hms:audio-editor-ui:{version}'
}

1.4在AndroidManifest.xml檔案中申請如下許可權

<!--震動許可權-->
<uses-permission android:name="android.permission.VIBRATE" />
<!--麥克風許可權-->
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<!--寫儲存許可權-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<!--讀儲存許可權-->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<!--網路許可權-->
<uses-permission android:name="android.permission.INTERNET" />
<!--網路狀態許可權-->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<!--網路狀態變化許可權-->
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />

2.程式碼開發

2.1建立應用自定義的activity介面,用於選擇音訊,並將該音訊檔案路徑返回給音訊編輯SDK

// 將音訊檔案路徑List返回到音訊編輯頁面
private void sendAudioToSdk() {
    // 獲取到的音訊檔案路徑 filePath
    String filePath = "/sdcard/AudioEdit/audio/music.aac";
    ArrayList<String> audioList = new ArrayList<>();
    audioList.add(filePath);
    // 將音訊檔案路徑返回到音訊編輯頁面
    Intent intent = new Intent();
    // 使用sdk提供的HAEConstant.AUDIO_PATH_LIST
    intent.putExtra(HAEConstant.AUDIO_PATH_LIST, audioList);
    // 使用sdk提供的HAEConstant.RESULT_CODE為結果CODE
    this.setResult(HAEConstant.RESULT_CODE, intent);
    finish();
}

2.2在UI介面匯入音訊時,SDK會傳送一個action值為com.huawei.hms.audioeditor.chooseaudio的intent以跳轉到該activity。因此,該activity“AndroidManifest.xml”中的註冊形式如下

<activity android:name="Activity "> 
<intent-filter> 
<action android:name="com.huawei.hms.audioeditor.chooseaudio"/> 
<category android:name="android.intent.category.DEFAULT"/> 
</intent-filter> 
</activity>

2.3啟動音訊編輯頁面,點選“新增音訊”,SDK會主動呼叫“2.1”步驟中定義的activity。新增好音訊,就可以進行音訊編輯、特效新增等操作,完成後匯出編輯音訊

HAEUIManager.getInstance().launchEditorActivity(this);

2.4.如果音訊素材不是MP3格式,此步驟可以完成音訊格式轉換

呼叫transformAudioUseDefaultPath介面進行音訊格式轉換,轉換後的音訊檔案匯出到預設路徑。

// 音訊格式轉換介面
HAEAudioExpansion.getInstance().transformAudioUseDefaultPath(context,inAudioPath, audioFormat, new OnTransformCallBack() {
    // 進度回撥(0-100)
    @Override
    public void onProgress(int progress) {
    }
    // 轉換失敗
    @Override
    public void onFail(int errorCode) {
    }
    // 轉換成功
    @Override
    public void onSuccess(String outPutPath) {
    }
    // 取消轉換
    @Override
    public void onCancel() {
    }
    });

// 取消轉換任務介面
HAEAudioExpansion.getInstance().cancelTransformAudio();

呼叫transformAudio介面進行音訊格式轉換,轉換後的音訊檔案匯出到目標路徑。

// 音訊格式轉換介面
HAEAudioExpansion.getInstance().transformAudio(context,inAudioPath, outAudioPath, new OnTransformCallBack(){
    // 進度回撥(0-100)
    @Override
    public void onProgress(int progress) {
    }
    // 轉換失敗
    @Override
    public void onFail(int errorCode) {
    }
    // 轉換成功
    @Override
    public void onSuccess(String outPutPath) {
    }
    // 取消轉換
    @Override
    public void onCancel() {
    }
    });
// 取消轉換任務介面
HAEAudioExpansion.getInstance().cancelTransformAudio();

2.5如果素材是視訊格式,可以呼叫extractAudio介面進行音訊提取,從視訊中提取音訊檔案再匯出到指定目錄

// outAudioDir提取出的音訊儲存的資料夾路徑,非必填
// outAudioName提取出的音訊名稱,不帶字尾,非必填
HAEAudioExpansion.getInstance().extractAudio(context,inVideoPath,outAudioDir, outAudioName,new AudioExtractCallBack() {
    @Override
    public void onSuccess(String audioPath) {
    Log.d(TAG, "ExtractAudio onSuccess : " + audioPath);
    }
    @Override
    public void onProgress(int progress) {
    Log.d(TAG, "ExtractAudio onProgress : " + progress);
    }
    @Override
    public void onFail(int errCode) {
    Log.i(TAG, "ExtractAudio onFail : " + errCode);
    }
    @Override
    public void onCancel() {
    Log.d(TAG, "ExtractAudio onCancel.");
    }
    });
// 取消音訊提取任務介面
HAEAudioExpansion.getInstance().cancelExtractAudio();

2.6呼叫getInstruments和startSeparationTasks介面進行伴奏提取

// 獲取提取伴奏型別ID,後面將此ID傳給介面
HAEAudioSeparationFile haeAudioSeparationFile = new HAEAudioSeparationFile();
haeAudioSeparationFile.getInstruments(new SeparationCloudCallBack<List<SeparationBean>>() {
    @Override
public void onFinish(List<SeparationBean> response) {
// 返回的資料,包括伴奏的型別ID
}
    @Override
    public void onError(int errorCode) {
        // 失敗返回
}
});
// 設定要提取的伴奏引數
List instruments = new ArrayList<>();
instruments.add(“伴奏id”);
haeAudioSeparationFile.setInstruments(instruments);
// 開始進行伴奏分離
haeAudioSeparationFile.startSeparationTasks(inAudioPath, outAudioDir, outAudioName, new AudioSeparationCallBack() {
    @Override
    public void onResult(SeparationBean separationBean) { }
    @Override
    public void onFinish(List<SeparationBean> separationBeans) {}
    @Override
    public void onFail(int errorCode) {}
    @Override
    public void onCancel() {}
});
// 取消分離任務
haeAudioSeparationFile.cancel();

2.7呼叫applyAudioFile介面進行空間方位渲染

// 空間方位渲染
// 固定擺位
HAESpaceRenderFile haeSpaceRenderFile = new HAESpaceRenderFile(SpaceRenderMode.POSITION);
haeSpaceRenderFile.setSpacePositionParams(
                            new SpaceRenderPositionParams(x, y, z));
// 動態渲染
HAESpaceRenderFile haeSpaceRenderFile = new HAESpaceRenderFile(SpaceRenderMode.ROTATION);
haeSpaceRenderFile.setRotationParams( new SpaceRenderRotationParams(
                                    x, y, z, surroundTime, surroundDirection));
// 擴充套件
HAESpaceRenderFile haeSpaceRenderFile = new HAESpaceRenderFile(SpaceRenderMode.EXTENSION);
haeSpaceRenderFile.setExtensionParams(new SpaceRenderExtensionParams(radiusVal, angledVal));
// 呼叫介面
haeSpaceRenderFile.applyAudioFile(inAudioPath, outAudioDir, outAudioName, callBack);
// 取消空間方位渲染
haeSpaceRenderFile.cancel();

完成以上步驟,就可以得到對應的空間動態渲染效果,在應用內輕鬆實現2D轉3D音效啦!這項功能還可以應用到企業會議以及運動康復領域,比如在展會上進行產品沉浸式展示、作為視障人群的方向感線索,為日常生活提供便利等。開發者們可以根據自己應用的實際需求選擇使用,如需瞭解更多詳情,請參考:
華為開發者聯盟音訊編輯服務官網; 獲取整合音訊編輯服務指導文件

瞭解更多詳情>>

訪問華為開發者聯盟官網
獲取開發指導文件
華為移動服務開源倉庫地址:GitHubGitee

關注我們,第一時間瞭解 HMS Core 最新技術資訊~

相關文章