Skip to content

Commit

Permalink
Merge branch 'main' into feature/websockets
Browse files Browse the repository at this point in the history
# Conflicts:
#	ElevenLabs/Packages/com.rest.elevenlabs/Runtime/Authentication/ElevenLabsSettingsInfo.cs
#	ElevenLabs/Packages/com.rest.elevenlabs/Runtime/TextToSpeech/TextToSpeechEndpoint.cs
#	ElevenLabs/Packages/com.rest.elevenlabs/package.json
#	ElevenLabs/Packages/manifest.json
  • Loading branch information
StephenHodgson committed Jan 13, 2025
2 parents 8d71d78 + 689ad11 commit bb65bc5
Show file tree
Hide file tree
Showing 33 changed files with 946 additions and 368 deletions.
16 changes: 8 additions & 8 deletions .github/workflows/unity.yml
Original file line number Diff line number Diff line change
Expand Up @@ -11,26 +11,24 @@ on:
workflow_dispatch:
concurrency:
group: ${{ github.workflow }}-${{ github.ref }}
cancel-in-progress: ${{ ( github.event_name == 'pull_request' || github.event.action == 'synchronize' ) }}
permissions:
checks: write
pull-requests: write
cancel-in-progress: ${{ (github.event_name == 'pull_request' || github.event.action == 'synchronize') }}
jobs:
build:
env:
UNITY_PROJECT_PATH: ''
permissions:
checks: write
pull-requests: write
runs-on: ${{ matrix.os }}
strategy:
fail-fast: false
matrix:
os: [ubuntu-latest, windows-latest, macos-13]
os: [ubuntu-latest, windows-latest, macos-15]
unity-versions: [2021.x, 2022.x, 6000.x]
include:
- os: ubuntu-latest
build-target: StandaloneLinux64
- os: windows-latest
build-target: StandaloneWindows64
- os: macos-13
- os: macos-15
build-target: StandaloneOSX
steps:
- uses: actions/checkout@v4
Expand All @@ -46,11 +44,13 @@ jobs:
- uses: RageAgainstThePixel/unity-action@v1
name: '${{ matrix.build-target }}-Validate'
with:
build-target: ${{ matrix.build-target }}
log-name: '${{ matrix.build-target }}-Validate'
args: '-quit -nographics -batchmode -executeMethod Utilities.Editor.BuildPipeline.UnityPlayerBuildTools.ValidateProject -importTMProEssentialsAsset'
- uses: RageAgainstThePixel/unity-action@v1
name: '${{ matrix.build-target }}-Build'
with:
build-target: ${{ matrix.build-target }}
log-name: '${{ matrix.build-target }}-Build'
args: '-quit -nographics -batchmode -executeMethod Utilities.Editor.BuildPipeline.UnityPlayerBuildTools.StartCommandLineBuild'
- uses: actions/upload-artifact@v4
Expand Down
3 changes: 3 additions & 0 deletions ElevenLabs/.editorconfig
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,9 @@ dotnet_style_predefined_type_for_locals_parameters_members = true
# Code Style
csharp_style_var_when_type_is_apparent = true

dotnet_diagnostic.IDE0051.severity = none
dotnet_diagnostic.CS0649.severity = none

#### Resharper/Rider Rules ####
# https://www.jetbrains.com/help/resharper/EditorConfig_Properties.html

Expand Down
47 changes: 22 additions & 25 deletions ElevenLabs/Packages/com.rest.elevenlabs/Documentation~/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@ The recommended installation method is though the unity package manager and [Ope
- [com.utilities.extensions](https://github.com/RageAgainstThePixel/com.utilities.extensions)
- [com.utilities.audio](https://github.com/RageAgainstThePixel/com.utilities.audio)
- [com.utilities.encoder.ogg](https://github.com/RageAgainstThePixel/com.utilities.encoder.ogg)
- [com.utilities.encoder.wav](https://github.com/RageAgainstThePixel/com.utilities.encoder.wav)
- [com.utilities.rest](https://github.com/RageAgainstThePixel/com.utilities.rest)

---
Expand All @@ -59,7 +60,7 @@ The recommended installation method is though the unity package manager and [Ope
- [Text to Speech](#text-to-speech)
- [Stream Text To Speech](#stream-text-to-speech)
- [Voices](#voices)
- [Get Shared Voices](#get-shared-voices) :new:
- [Get Shared Voices](#get-shared-voices)
- [Get All Voices](#get-all-voices)
- [Get Default Voice Settings](#get-default-voice-settings)
- [Get Voice](#get-voice)
Expand All @@ -70,13 +71,13 @@ The recommended installation method is though the unity package manager and [Ope
- [Samples](#samples)
- [Download Voice Sample](#download-voice-sample)
- [Delete Voice Sample](#delete-voice-sample)
- [Dubbing](#dubbing) :new:
- [Dub](#dub) :new:
- [Get Dubbing Metadata](#get-dubbing-metadata) :new:
- [Get Transcript for Dub](#get-transcript-for-dub) :new:
- [Get dubbed file](#get-dubbed-file) :new:
- [Delete Dubbing Project](#delete-dubbing-project) :new:
- [SFX Generation](#sfx-generation) :new:
- [Dubbing](#dubbing)
- [Dub](#dub)
- [Get Dubbing Metadata](#get-dubbing-metadata)
- [Get Transcript for Dub](#get-transcript-for-dub)
- [Get dubbed file](#get-dubbed-file)
- [Delete Dubbing Project](#delete-dubbing-project)
- [SFX Generation](#sfx-generation)
- [History](#history)
- [Get History](#get-history)
- [Get History Item](#get-history-item)
Expand Down Expand Up @@ -265,8 +266,8 @@ Convert text to speech.
var api = new ElevenLabsClient();
var text = "The quick brown fox jumps over the lazy dog.";
var voice = (await api.VoicesEndpoint.GetAllVoicesAsync()).FirstOrDefault();
var defaultVoiceSettings = await api.VoicesEndpoint.GetDefaultVoiceSettingsAsync();
var voiceClip = await api.TextToSpeechEndpoint.TextToSpeechAsync(text, voice, defaultVoiceSettings);
var request = new TextToSpeechRequest(voice, text);
var voiceClip = await api.TextToSpeechEndpoint.TextToSpeechAsync(request);
audioSource.PlayOneShot(voiceClip.AudioClip);
```

Expand All @@ -284,18 +285,14 @@ Stream text to speech.
var api = new ElevenLabsClient();
var text = "The quick brown fox jumps over the lazy dog.";
var voice = (await api.VoicesEndpoint.GetAllVoicesAsync()).FirstOrDefault();
var partialClips = new Queue<AudioClip>();
var voiceClip = await api.TextToSpeechEndpoint.StreamTextToSpeechAsync(
text,
voice,
partialClip =>
{
// Note: Best to queue them and play them in update loop!
// See TextToSpeech sample demo for details
partialClips.Enqueue(partialClip);
});
// The full completed clip:
audioSource.clip = voiceClip.AudioClip;
var partialClips = new Queue<VoiceClip>();
var request = new TextToSpeechRequest(voice, message, model: Model.EnglishTurboV2, outputFormat: OutputFormat.PCM_44100);
var voiceClip = await api.TextToSpeechEndpoint.StreamTextToSpeechAsync(request, partialClip =>
{
// Note: check demo scene for best practices
// on how to handle playback with OnAudioFilterRead
partialClips.Enqueue(partialClip);
});
```

### [Voices](https://docs.elevenlabs.io/api-reference/voices)
Expand All @@ -308,7 +305,7 @@ Gets a list of shared voices in the public voice library.

```csharp
var api = new ElevenLabsClient();
var results = await ElevenLabsClient.SharedVoicesEndpoint.GetSharedVoicesAsync();
var results = await api.SharedVoicesEndpoint.GetSharedVoicesAsync();
foreach (var voice in results.Voices)
{
Debug.Log($"{voice.OwnerId} | {voice.VoiceId} | {voice.Date} | {voice.Name}");
Expand Down Expand Up @@ -455,7 +452,7 @@ Returns downloaded dubbed file path.
> Videos will be returned in MP4 format and audio only dubs will be returned in MP3.
```csharp
var dubbedClipPath = await ElevenLabsClient.DubbingEndpoint.GetDubbedFileAsync(metadata.DubbingId, request.TargetLanguage);
var dubbedClipPath = await api.DubbingEndpoint.GetDubbedFileAsync(metadata.DubbingId, request.TargetLanguage);
var dubbedClip = await Rest.DownloadAudioClipAsync($"file://{dubbedClipPath}", AudioType.MPEG);
audioSource.PlayOneShot(dubbedClip);
```
Expand All @@ -467,7 +464,7 @@ Returns transcript for the dub in the desired format.
```csharp
var srcFile = new FileInfo(audioPath);
var transcriptPath = new FileInfo($"{srcFile.FullName}.dubbed.{request.TargetLanguage}.srt");
var transcriptFile = await ElevenLabsClient.DubbingEndpoint.GetTranscriptForDubAsync(metadata.DubbingId, request.TargetLanguage);
var transcriptFile = await api.DubbingEndpoint.GetTranscriptForDubAsync(metadata.DubbingId, request.TargetLanguage);
await File.WriteAllTextAsync(transcriptPath.FullName, transcriptFile);
```

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1106,7 +1106,7 @@ private async void GenerateSynthesizedText()
Directory.CreateDirectory(downloadDir);
}

voiceClip = await api.TextToSpeechEndpoint.TextToSpeechAsync(speechSynthesisTextInput, currentVoiceOption, currentVoiceSettings, currentModelOption);
voiceClip = await api.TextToSpeechEndpoint.TextToSpeechAsync(new(currentVoiceOption, speechSynthesisTextInput, voiceSettings: currentVoiceSettings, model: currentModelOption));
voiceClip.CopyIntoProject(editorDownloadDirectory);
}
catch (Exception e)
Expand Down Expand Up @@ -1225,7 +1225,7 @@ private void RenderVoiceLab()
EditorGUILayout.Space(EndWidth);
EditorGUILayout.EndHorizontal();
EditorGUI.indentLevel++;

EditorGUILayout.BeginHorizontal();
{
EditorGUILayout.LabelField(voice.Id, EditorStyles.boldLabel);
Expand All @@ -1242,7 +1242,7 @@ private void RenderVoiceLab()
EditorGUILayout.Space(EndWidth);
EditorGUILayout.EndHorizontal();
EditorGUI.indentLevel++;

if (!voiceLabels.TryGetValue(voice.Id, out var cachedLabels))
{
cachedLabels = new Dictionary<string, string>();
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
// Licensed under the MIT License. See LICENSE in the project root for license information.

namespace ElevenLabs
{
public enum CacheFormat
{
None,
Ogg,
Wav
}
}

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,12 @@

using ElevenLabs.Extensions;
using System;
using System.Threading;
using System.Threading.Tasks;
using UnityEngine;
using UnityEngine.Scripting;
using Utilities.Audio;
using Utilities.WebRequestRest;

namespace ElevenLabs
{
Expand All @@ -12,16 +16,30 @@ namespace ElevenLabs
public class GeneratedClip : ISerializationCallbackReceiver
{
[Preserve]
internal GeneratedClip(string id, string text, AudioClip audioClip, string cachedPath)
internal GeneratedClip(string id, string text, AudioClip audioClip, string cachedPath = null)
{
this.id = id;
this.text = text;
TextHash = $"{id}{text}".GenerateGuid();
textHash = TextHash.ToString();
this.audioClip = audioClip;
this.cachedPath = cachedPath;
SampleRate = audioClip.frequency;
}

[Preserve]
internal GeneratedClip(string id, string text, ReadOnlyMemory<byte> clipData, int sampleRate, string cachedPath = null)
{
this.id = id;
this.text = text;
TextHash = $"{id}{text}".GenerateGuid();
textHash = TextHash.ToString();
this.cachedPath = cachedPath;
ClipData = clipData;
SampleRate = sampleRate;
}

private readonly ReadOnlyMemory<byte> audioData;

[SerializeField]
private string id;

Expand All @@ -44,16 +62,73 @@ internal GeneratedClip(string id, string text, AudioClip audioClip, string cache
private AudioClip audioClip;

[Preserve]
public AudioClip AudioClip => audioClip;
public AudioClip AudioClip
{
get
{
if (audioClip == null && !audioData.IsEmpty)
{
var pcmData = PCMEncoder.Decode(audioData.ToArray());
audioClip = AudioClip.Create(Id, pcmData.Length, 1, SampleRate, false);
audioClip.SetData(pcmData, 0);
}

if (audioClip == null)
{
Debug.LogError($"{nameof(audioClip)} is null, try loading it with LoadCachedAudioClipAsync");
}

return audioClip;
}
}

[SerializeField]
private string cachedPath;

[Preserve]
public string CachedPath => cachedPath;

public ReadOnlyMemory<byte> ClipData { get; }

private float[] clipSamples;

public float[] ClipSamples
{
get
{
if (!ClipData.IsEmpty)
{
clipSamples ??= PCMEncoder.Decode(ClipData.ToArray());
}
else if (audioClip != null)
{
clipSamples = new float[audioClip.samples];
audioClip.GetData(clipSamples, 0);
}

return clipSamples;
}
}

public int SampleRate { get; }

public void OnBeforeSerialize() => textHash = TextHash.ToString();

public void OnAfterDeserialize() => TextHash = Guid.Parse(textHash);

public static implicit operator AudioClip(GeneratedClip clip) => clip?.AudioClip;

public async Task<AudioClip> LoadCachedAudioClipAsync(CancellationToken cancellationToken = default)
{
var audioType = cachedPath switch
{
var path when path.EndsWith(".ogg") => AudioType.OGGVORBIS,
var path when path.EndsWith(".wav") => AudioType.WAV,
var path when path.EndsWith(".mp3") => AudioType.MPEG,
_ => AudioType.UNKNOWN
};

return await Rest.DownloadAudioClipAsync($"file://{cachedPath}", audioType, cancellationToken: cancellationToken);
}
}
}

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
// Licensed under the MIT License. See LICENSE in the project root for license information.

using Newtonsoft.Json;
using UnityEngine.Scripting;

namespace ElevenLabs
{
/// <summary>
/// Represents timing information for a single character in the transcript
/// </summary>
[Preserve]
public class TimestampedTranscriptCharacter
{
[Preserve]
[JsonConstructor]
internal TimestampedTranscriptCharacter(string character, double startTime, double endTime)
{
Character = character;
StartTime = startTime;
EndTime = endTime;
}

/// <summary>
/// The character being spoken
/// </summary>
[Preserve]
[JsonProperty("character")]
public string Character { get; }

/// <summary>
/// The time in seconds when this character starts being spoken
/// </summary>
[Preserve]
[JsonProperty("character_start_times_seconds")]
public double StartTime { get; }

/// <summary>
/// The time in seconds when this character finishes being spoken
/// </summary>
[Preserve]
[JsonProperty("character_end_times_seconds")]
public double EndTime { get; }
}
}
Loading

0 comments on commit bb65bc5

Please sign in to comment.