2019-08-15 10:46:47 +02:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using NAudio.CoreAudioApi;
|
2019-09-06 08:32:29 +02:00
|
|
|
|
using SafeExamBrowser.Configuration.Contracts.Settings.SystemComponents;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-08-30 17:33:28 +02:00
|
|
|
|
using SafeExamBrowser.SystemComponents.Contracts.Audio;
|
|
|
|
|
using SafeExamBrowser.SystemComponents.Contracts.Audio.Events;
|
2019-08-15 10:46:47 +02:00
|
|
|
|
|
2019-08-30 17:33:28 +02:00
|
|
|
|
namespace SafeExamBrowser.SystemComponents.Audio
|
2019-08-15 10:46:47 +02:00
|
|
|
|
{
|
2019-08-30 17:33:28 +02:00
|
|
|
|
public class Audio : IAudio
|
2019-08-15 10:46:47 +02:00
|
|
|
|
{
|
2019-08-16 08:26:11 +02:00
|
|
|
|
private AudioSettings settings;
|
2019-08-15 10:46:47 +02:00
|
|
|
|
private MMDevice audioDevice;
|
2019-08-30 17:33:28 +02:00
|
|
|
|
private string audioDeviceFullName;
|
2019-08-15 10:46:47 +02:00
|
|
|
|
private string audioDeviceShortName;
|
2019-08-16 08:26:11 +02:00
|
|
|
|
private float originalVolume;
|
2019-08-15 10:46:47 +02:00
|
|
|
|
private ILogger logger;
|
|
|
|
|
|
2019-08-30 17:33:28 +02:00
|
|
|
|
public string DeviceFullName => audioDeviceFullName ?? string.Empty;
|
|
|
|
|
public string DeviceShortName => audioDeviceShortName ?? string.Empty;
|
|
|
|
|
public bool HasOutputDevice => audioDevice != default(MMDevice);
|
|
|
|
|
public bool OutputMuted => audioDevice?.AudioEndpointVolume.Mute == true;
|
|
|
|
|
public double OutputVolume => audioDevice?.AudioEndpointVolume.MasterVolumeLevelScalar ?? 0;
|
|
|
|
|
|
2019-09-04 11:46:30 +02:00
|
|
|
|
public event VolumeChangedEventHandler VolumeChanged;
|
2019-08-30 17:33:28 +02:00
|
|
|
|
|
2019-09-03 11:46:36 +02:00
|
|
|
|
public Audio(AudioSettings settings, ILogger logger)
|
2019-08-15 10:46:47 +02:00
|
|
|
|
{
|
2019-08-16 08:26:11 +02:00
|
|
|
|
this.settings = settings;
|
2019-08-15 10:46:47 +02:00
|
|
|
|
this.logger = logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Initialize()
|
|
|
|
|
{
|
|
|
|
|
if (TryLoadAudioDevice())
|
|
|
|
|
{
|
|
|
|
|
InitializeAudioDevice();
|
|
|
|
|
InitializeSettings();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Warn("Could not find an active audio device!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-30 17:33:28 +02:00
|
|
|
|
public void Mute()
|
|
|
|
|
{
|
|
|
|
|
if (audioDevice != default(MMDevice))
|
|
|
|
|
{
|
|
|
|
|
audioDevice.AudioEndpointVolume.Mute = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-08-15 16:16:51 +02:00
|
|
|
|
|
2019-08-30 17:33:28 +02:00
|
|
|
|
public void Unmute()
|
|
|
|
|
{
|
|
|
|
|
if (audioDevice != default(MMDevice))
|
|
|
|
|
{
|
|
|
|
|
audioDevice.AudioEndpointVolume.Mute = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-08-15 10:46:47 +02:00
|
|
|
|
|
2019-08-30 17:33:28 +02:00
|
|
|
|
public void SetVolume(double value)
|
|
|
|
|
{
|
|
|
|
|
if (audioDevice != default(MMDevice))
|
|
|
|
|
{
|
|
|
|
|
audioDevice.AudioEndpointVolume.MasterVolumeLevelScalar = (float) value;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-08-15 10:46:47 +02:00
|
|
|
|
|
|
|
|
|
public void Terminate()
|
|
|
|
|
{
|
|
|
|
|
if (audioDevice != default(MMDevice))
|
|
|
|
|
{
|
2019-08-16 08:26:11 +02:00
|
|
|
|
RevertSettings();
|
|
|
|
|
FinalizeAudioDevice();
|
2019-08-15 10:46:47 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool TryLoadAudioDevice()
|
|
|
|
|
{
|
|
|
|
|
using (var enumerator = new MMDeviceEnumerator())
|
|
|
|
|
{
|
|
|
|
|
if (enumerator.HasDefaultAudioEndpoint(DataFlow.Render, Role.Console))
|
|
|
|
|
{
|
|
|
|
|
audioDevice = enumerator.GetDefaultAudioEndpoint(DataFlow.Render, Role.Console);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
audioDevice = enumerator.EnumerateAudioEndPoints(DataFlow.Render, DeviceState.Active).FirstOrDefault();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return audioDevice != default(MMDevice);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeAudioDevice()
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Found '{audioDevice}' to be the active audio device.");
|
|
|
|
|
audioDevice.AudioEndpointVolume.OnVolumeNotification += AudioEndpointVolume_OnVolumeNotification;
|
2019-08-30 17:33:28 +02:00
|
|
|
|
audioDeviceFullName = audioDevice.FriendlyName;
|
2019-08-15 10:46:47 +02:00
|
|
|
|
audioDeviceShortName = audioDevice.FriendlyName.Length > 25 ? audioDevice.FriendlyName.Split(' ').First() : audioDevice.FriendlyName;
|
|
|
|
|
logger.Info("Started monitoring the audio device.");
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-16 08:26:11 +02:00
|
|
|
|
private void FinalizeAudioDevice()
|
|
|
|
|
{
|
|
|
|
|
audioDevice.AudioEndpointVolume.OnVolumeNotification -= AudioEndpointVolume_OnVolumeNotification;
|
|
|
|
|
audioDevice.Dispose();
|
|
|
|
|
logger.Info("Stopped monitoring the audio device.");
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-15 10:46:47 +02:00
|
|
|
|
private void InitializeSettings()
|
|
|
|
|
{
|
2019-08-16 08:26:11 +02:00
|
|
|
|
if (settings.InitializeVolume)
|
|
|
|
|
{
|
|
|
|
|
originalVolume = audioDevice.AudioEndpointVolume.MasterVolumeLevelScalar;
|
|
|
|
|
logger.Info($"Saved original volume of {Math.Round(originalVolume * 100)}%.");
|
|
|
|
|
audioDevice.AudioEndpointVolume.MasterVolumeLevelScalar = settings.InitialVolume / 100f;
|
|
|
|
|
logger.Info($"Set initial volume to {settings.InitialVolume}%.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (settings.MuteAudio)
|
|
|
|
|
{
|
|
|
|
|
audioDevice.AudioEndpointVolume.Mute = true;
|
|
|
|
|
logger.Info("Muted audio device.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void RevertSettings()
|
|
|
|
|
{
|
|
|
|
|
if (settings.InitializeVolume)
|
|
|
|
|
{
|
|
|
|
|
audioDevice.AudioEndpointVolume.MasterVolumeLevelScalar = originalVolume;
|
|
|
|
|
logger.Info($"Reverted volume to original value of {Math.Round(originalVolume * 100)}%.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (settings.MuteAudio)
|
|
|
|
|
{
|
|
|
|
|
audioDevice.AudioEndpointVolume.Mute = false;
|
|
|
|
|
logger.Info("Unmuted audio device.");
|
|
|
|
|
}
|
2019-08-15 10:46:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void AudioEndpointVolume_OnVolumeNotification(AudioVolumeNotificationData data)
|
|
|
|
|
{
|
2019-09-03 11:46:36 +02:00
|
|
|
|
logger.Debug($"Volume is set to {data.MasterVolume * 100}%, audio device is {(data.Muted ? "muted" : "not muted")}.");
|
|
|
|
|
VolumeChanged?.Invoke(data.MasterVolume, data.Muted);
|
2019-08-15 10:46:47 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|