Remove direct access of Player from UI code

This commit is contained in:
Matt Nadareski
2021-07-03 15:37:56 -07:00
parent faba0ffada
commit 0d7681543a
3 changed files with 234 additions and 160 deletions

View File

@@ -12,16 +12,15 @@ using Avalonia.Markup.Xaml;
using Avalonia.Media.Imaging; using Avalonia.Media.Imaging;
using Avalonia.Platform; using Avalonia.Platform;
using Avalonia.Threading; using Avalonia.Threading;
using RedBookPlayer.Hardware;
namespace RedBookPlayer.GUI namespace RedBookPlayer.GUI
{ {
public class PlayerView : UserControl public class PlayerView : UserControl
{ {
/// <summary> /// <summary>
/// Player representing the internal state /// Read-only access to the view model
/// </summary> /// </summary>
public static Player Player = new Player(); public PlayerViewModel PlayerViewModel => DataContext as PlayerViewModel;
/// <summary> /// <summary>
/// Set of images representing the digits for the UI /// Set of images representing the digits for the UI
@@ -71,13 +70,12 @@ namespace RedBookPlayer.GUI
public async Task<bool> LoadImage(string path) public async Task<bool> LoadImage(string path)
{ {
// If the player is currently running, stop it // If the player is currently running, stop it
if((DataContext as PlayerViewModel).Playing != true) if(PlayerViewModel.Playing != true) PlayerViewModel.Playing = null;
(DataContext as PlayerViewModel).Playing = null;
bool result = await Task.Run(() => bool result = await Dispatcher.UIThread.InvokeAsync(() =>
{ {
Player.Init(path, App.Settings.AutoPlay); PlayerViewModel.Init(path, App.Settings.AutoPlay);
return Player.Initialized; return PlayerViewModel.Initialized;
}); });
if(result) if(result)
@@ -122,7 +120,7 @@ namespace RedBookPlayer.GUI
private void InitializeComponent(string xaml) private void InitializeComponent(string xaml)
{ {
DataContext = new PlayerViewModel(); DataContext = new PlayerViewModel();
(DataContext as PlayerViewModel).PropertyChanged += UpdateModel; PlayerViewModel.PropertyChanged += UpdateModel;
// Load the theme // Load the theme
try try
@@ -199,7 +197,7 @@ namespace RedBookPlayer.GUI
{ {
Dispatcher.UIThread.InvokeAsync(() => Dispatcher.UIThread.InvokeAsync(() =>
{ {
Player.UpdateModel(DataContext as PlayerViewModel); PlayerViewModel.UpdateModel();
}); });
} }
@@ -210,14 +208,14 @@ namespace RedBookPlayer.GUI
{ {
Dispatcher.UIThread.InvokeAsync(() => Dispatcher.UIThread.InvokeAsync(() =>
{ {
string digitString = Player.GenerateDigitString(); string digitString = PlayerViewModel.GenerateDigitString();
for (int i = 0; i < _digits.Length; i++) for (int i = 0; i < _digits.Length; i++)
{ {
if (_digits[i] != null) if (_digits[i] != null)
_digits[i].Source = GetBitmap(digitString[i]); _digits[i].Source = GetBitmap(digitString[i]);
} }
Player.UpdateDataContext(DataContext as PlayerViewModel); PlayerViewModel?.UpdateView();
}); });
} }
@@ -234,49 +232,55 @@ namespace RedBookPlayer.GUI
await LoadImage(path); await LoadImage(path);
} }
public void PlayButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).Playing = true; public void PlayButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.Playing = true;
public void PauseButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).Playing = false; public void PauseButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.Playing = false;
public void PlayPauseButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).Playing = !(DataContext as PlayerViewModel).Playing; public void PlayPauseButton_Click(object sender, RoutedEventArgs e)
{
if(PlayerViewModel.Playing == true)
PlayerViewModel.Playing = false;
else
PlayerViewModel.Playing = true;
}
public void StopButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).Playing = null; public void StopButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.Playing = null;
public void NextTrackButton_Click(object sender, RoutedEventArgs e) => Player.NextTrack(); public void NextTrackButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.NextTrack();
public void PreviousTrackButton_Click(object sender, RoutedEventArgs e) => Player.PreviousTrack(); public void PreviousTrackButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.PreviousTrack();
public void NextIndexButton_Click(object sender, RoutedEventArgs e) => Player.NextIndex(App.Settings.IndexButtonChangeTrack); public void NextIndexButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.NextIndex(App.Settings.IndexButtonChangeTrack);
public void PreviousIndexButton_Click(object sender, RoutedEventArgs e) => Player.PreviousIndex(App.Settings.IndexButtonChangeTrack); public void PreviousIndexButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.PreviousIndex(App.Settings.IndexButtonChangeTrack);
public void FastForwardButton_Click(object sender, RoutedEventArgs e) => Player.FastForward(); public void FastForwardButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.FastForward();
public void RewindButton_Click(object sender, RoutedEventArgs e) => Player.Rewind(); public void RewindButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.Rewind();
public void VolumeUpButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).Volume++; public void VolumeUpButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.Volume++;
public void VolumeDownButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).Volume--; public void VolumeDownButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.Volume--;
public void MuteToggleButton_Click(object sender, RoutedEventArgs e) public void MuteToggleButton_Click(object sender, RoutedEventArgs e)
{ {
if (_lastVolume == null) if (_lastVolume == null)
{ {
_lastVolume = (DataContext as PlayerViewModel).Volume; _lastVolume = PlayerViewModel.Volume;
(DataContext as PlayerViewModel).Volume = 0; PlayerViewModel.Volume = 0;
} }
else else
{ {
(DataContext as PlayerViewModel).Volume = _lastVolume.Value; PlayerViewModel.Volume = _lastVolume.Value;
_lastVolume = null; _lastVolume = null;
} }
} }
public void EnableDeEmphasisButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).ApplyDeEmphasis = true; public void EnableDeEmphasisButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.ApplyDeEmphasis = true;
public void DisableDeEmphasisButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).ApplyDeEmphasis = false; public void DisableDeEmphasisButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.ApplyDeEmphasis = false;
public void EnableDisableDeEmphasisButton_Click(object sender, RoutedEventArgs e) => (DataContext as PlayerViewModel).ApplyDeEmphasis = !(DataContext as PlayerViewModel).ApplyDeEmphasis; public void EnableDisableDeEmphasisButton_Click(object sender, RoutedEventArgs e) => PlayerViewModel.ApplyDeEmphasis = !PlayerViewModel.ApplyDeEmphasis;
#endregion #endregion
} }

View File

@@ -1,11 +1,22 @@
using System.Linq;
using Aaru.CommonTypes.Enums;
using ReactiveUI; using ReactiveUI;
using RedBookPlayer.Discs;
using RedBookPlayer.Hardware;
namespace RedBookPlayer.GUI namespace RedBookPlayer.GUI
{ {
public class PlayerViewModel : ReactiveObject public class PlayerViewModel : ReactiveObject
{ {
/// <summary>
/// Player representing the internal state
/// </summary>
private Player _player;
#region Player Status #region Player Status
public bool Initialized => _player?.Initialized ?? false;
private bool? _playing; private bool? _playing;
public bool? Playing public bool? Playing
{ {
@@ -93,5 +104,137 @@ namespace RedBookPlayer.GUI
} }
#endregion #endregion
/// <summary>
/// Initialize the view model with a given image path
/// </summary>
/// <param name="path">Path to the disc image</param>
/// <param name="autoPlay">True if playback should begin immediately, false otherwise</param>
public void Init(string path, bool autoPlay)
{
_player = new Player();
_player.Init(path, autoPlay);
if(Initialized)
UpdateModel();
}
#region Playback
/// <summary>
/// Move to the next playable track
/// </summary>
public void NextTrack() => _player?.NextTrack();
/// <summary>
/// Move to the previous playable track
/// </summary>
public void PreviousTrack() => _player?.PreviousTrack();
/// <summary>
/// Move to the next index
/// </summary>
/// <param name="changeTrack">True if index changes can trigger a track change, false otherwise</param>
public void NextIndex(bool changeTrack) => _player?.NextIndex(changeTrack);
/// <summary>
/// Move to the previous index
/// </summary>
/// <param name="changeTrack">True if index changes can trigger a track change, false otherwise</param>
public void PreviousIndex(bool changeTrack) => _player?.PreviousIndex(changeTrack);
/// <summary>
/// Fast-forward playback by 75 sectors, if possible
/// </summary>
public void FastForward() => _player?.FastForward();
/// <summary>
/// Rewind playback by 75 sectors, if possible
/// </summary>
public void Rewind() => _player?.Rewind();
#endregion
#region Helpers
/// <summary>
/// Generate the digit string to be interpreted by the frontend
/// </summary>
/// <returns>String representing the digits for the frontend</returns>
public string GenerateDigitString()
{
// If the disc isn't initialized, return all '-' characters
if(_player?.OpticalDisc == null || !_player.OpticalDisc.Initialized)
return string.Empty.PadLeft(20, '-');
// Otherwise, take the current time into account
ulong sectorTime = _player.GetCurrentSectorTime();
int[] numbers = new int[]
{
_player.OpticalDisc.CurrentTrackNumber + 1,
_player.OpticalDisc.CurrentTrackIndex,
(int)(sectorTime / (75 * 60)),
(int)(sectorTime / 75 % 60),
(int)(sectorTime % 75),
_player.OpticalDisc.TotalTracks,
_player.OpticalDisc.TotalIndexes,
(int)(_player.OpticalDisc.TotalTime / (75 * 60)),
(int)(_player.OpticalDisc.TotalTime / 75 % 60),
(int)(_player.OpticalDisc.TotalTime % 75),
};
return string.Join("", numbers.Select(i => i.ToString().PadLeft(2, '0').Substring(0, 2)));
}
/// <summary>
/// Update the UI from the internal player
/// </summary>
public void UpdateView()
{
if(_player?.Initialized != true)
return;
Playing = _player.Playing;
CurrentSector = _player.GetCurrentSectorTime();
TotalSectors = _player.OpticalDisc.TotalTime;
Volume = App.Settings.Volume;
ApplyDeEmphasis = _player.ApplyDeEmphasis;
HiddenTrack = _player.OpticalDisc.TimeOffset > 150;
if(_player.OpticalDisc is CompactDisc compactDisc)
{
QuadChannel = compactDisc.QuadChannel;
IsDataTrack = compactDisc.IsDataTrack;
CopyAllowed = compactDisc.CopyAllowed;
TrackHasEmphasis = compactDisc.TrackHasEmphasis;
}
else
{
QuadChannel = false;
IsDataTrack = _player.OpticalDisc.TrackType != TrackType.Audio;
CopyAllowed = false;
TrackHasEmphasis = false;
}
}
/// <summary>
/// Update the internal player from the UI
/// </summary>
public void UpdateModel()
{
if(_player?.Initialized != true)
return;
_player.SetPlayingState(Playing);
App.Settings.Volume = Volume;
_player.SetDeEmphasis(ApplyDeEmphasis);
}
#endregion
} }
} }

View File

@@ -1,11 +1,8 @@
using System; using System;
using System.IO; using System.IO;
using System.Linq;
using Aaru.CommonTypes.Enums;
using Aaru.DiscImages; using Aaru.DiscImages;
using Aaru.Filters; using Aaru.Filters;
using RedBookPlayer.Discs; using RedBookPlayer.Discs;
using RedBookPlayer.GUI;
namespace RedBookPlayer.Hardware namespace RedBookPlayer.Hardware
{ {
@@ -18,10 +15,15 @@ namespace RedBookPlayer.Hardware
/// </summary> /// </summary>
public bool Initialized { get; private set; } = false; public bool Initialized { get; private set; } = false;
/// <summary>
/// OpticalDisc object
/// </summary>
public OpticalDisc OpticalDisc { get; private set; }
/// <summary> /// <summary>
/// Indicate if the disc is playing /// Indicate if the disc is playing
/// </summary> /// </summary>
public bool Playing => _soundOutput?.Playing ?? false; public bool? Playing => _soundOutput?.Playing;
/// <summary> /// <summary>
/// Indicates if de-emphasis should be applied /// Indicates if de-emphasis should be applied
@@ -32,11 +34,6 @@ namespace RedBookPlayer.Hardware
#region Private State Variables #region Private State Variables
/// <summary>
/// OpticalDisc object
/// </summary>
private OpticalDisc _opticalDisc;
/// <summary> /// <summary>
/// Sound output handling class /// Sound output handling class
/// </summary> /// </summary>
@@ -55,7 +52,7 @@ namespace RedBookPlayer.Hardware
Initialized = false; Initialized = false;
_soundOutput = new SoundOutput(); _soundOutput = new SoundOutput();
_soundOutput.ApplyDeEmphasis = false; _soundOutput.ApplyDeEmphasis = false;
_opticalDisc = null; OpticalDisc = null;
try try
{ {
@@ -70,7 +67,7 @@ namespace RedBookPlayer.Hardware
image.Open(filter); image.Open(filter);
// Generate and instantiate the disc // Generate and instantiate the disc
_opticalDisc = OpticalDiscFactory.GenerateFromImage(image, App.Settings.AutoPlay); OpticalDisc = OpticalDiscFactory.GenerateFromImage(image, App.Settings.AutoPlay);
} }
catch catch
{ {
@@ -79,7 +76,7 @@ namespace RedBookPlayer.Hardware
} }
// Initialize the sound output // Initialize the sound output
_soundOutput.Init(_opticalDisc, autoPlay); _soundOutput.Init(OpticalDisc, autoPlay);
if(_soundOutput == null || !_soundOutput.Initialized) if(_soundOutput == null || !_soundOutput.Initialized)
return; return;
@@ -93,15 +90,19 @@ namespace RedBookPlayer.Hardware
/// Set the current audio playback state /// Set the current audio playback state
/// </summary> /// </summary>
/// <param name="start">True to start playback, false to pause, null to stop</param> /// <param name="start">True to start playback, false to pause, null to stop</param>
private void SetPlayingState(bool? start) public void SetPlayingState(bool? start)
{ {
if(_opticalDisc == null || !_opticalDisc.Initialized) if(OpticalDisc == null || !OpticalDisc.Initialized)
return;
// If the playing state has not changed, do nothing
if(start == Playing)
return; return;
if(start == true) if(start == true)
{ {
_soundOutput.Play(); _soundOutput.Play();
_opticalDisc.SetTotalIndexes(); OpticalDisc.SetTotalIndexes();
} }
else if(start == false) else if(start == false)
{ {
@@ -110,7 +111,7 @@ namespace RedBookPlayer.Hardware
else else
{ {
_soundOutput.Stop(); _soundOutput.Stop();
_opticalDisc.LoadFirstTrack(); OpticalDisc.LoadFirstTrack();
} }
} }
@@ -119,17 +120,17 @@ namespace RedBookPlayer.Hardware
/// </summary> /// </summary>
public void NextTrack() public void NextTrack()
{ {
if(_opticalDisc == null || !_opticalDisc.Initialized) if(OpticalDisc == null || !OpticalDisc.Initialized)
return; return;
bool wasPlaying = Playing; bool? wasPlaying = Playing;
if(wasPlaying) SetPlayingState(false); if(wasPlaying == true) SetPlayingState(false);
_opticalDisc.NextTrack(); OpticalDisc.NextTrack();
if(_opticalDisc is CompactDisc compactDisc) if(OpticalDisc is CompactDisc compactDisc)
_soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis; _soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis;
if(wasPlaying) SetPlayingState(true); if(wasPlaying == true) SetPlayingState(true);
} }
/// <summary> /// <summary>
@@ -137,17 +138,17 @@ namespace RedBookPlayer.Hardware
/// </summary> /// </summary>
public void PreviousTrack() public void PreviousTrack()
{ {
if(_opticalDisc == null || !_opticalDisc.Initialized) if(OpticalDisc == null || !OpticalDisc.Initialized)
return; return;
bool wasPlaying = Playing; bool? wasPlaying = Playing;
if(wasPlaying) SetPlayingState(false); if(wasPlaying == true) SetPlayingState(false);
_opticalDisc.PreviousTrack(); OpticalDisc.PreviousTrack();
if(_opticalDisc is CompactDisc compactDisc) if(OpticalDisc is CompactDisc compactDisc)
_soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis; _soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis;
if(wasPlaying) SetPlayingState(true); if(wasPlaying == true) SetPlayingState(true);
} }
/// <summary> /// <summary>
@@ -156,17 +157,17 @@ namespace RedBookPlayer.Hardware
/// <param name="changeTrack">True if index changes can trigger a track change, false otherwise</param> /// <param name="changeTrack">True if index changes can trigger a track change, false otherwise</param>
public void NextIndex(bool changeTrack) public void NextIndex(bool changeTrack)
{ {
if(_opticalDisc == null || !_opticalDisc.Initialized) if(OpticalDisc == null || !OpticalDisc.Initialized)
return; return;
bool wasPlaying = Playing; bool? wasPlaying = Playing;
if(wasPlaying) SetPlayingState(false); if(wasPlaying == true) SetPlayingState(false);
_opticalDisc.NextIndex(changeTrack); OpticalDisc.NextIndex(changeTrack);
if(_opticalDisc is CompactDisc compactDisc) if(OpticalDisc is CompactDisc compactDisc)
_soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis; _soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis;
if(wasPlaying) SetPlayingState(true); if(wasPlaying == true) SetPlayingState(true);
} }
/// <summary> /// <summary>
@@ -175,17 +176,17 @@ namespace RedBookPlayer.Hardware
/// <param name="changeTrack">True if index changes can trigger a track change, false otherwise</param> /// <param name="changeTrack">True if index changes can trigger a track change, false otherwise</param>
public void PreviousIndex(bool changeTrack) public void PreviousIndex(bool changeTrack)
{ {
if(_opticalDisc == null || !_opticalDisc.Initialized) if(OpticalDisc == null || !OpticalDisc.Initialized)
return; return;
bool wasPlaying = Playing; bool? wasPlaying = Playing;
if(wasPlaying) SetPlayingState(false); if(wasPlaying == true) SetPlayingState(false);
_opticalDisc.PreviousIndex(changeTrack); OpticalDisc.PreviousIndex(changeTrack);
if(_opticalDisc is CompactDisc compactDisc) if(OpticalDisc is CompactDisc compactDisc)
_soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis; _soundOutput.ApplyDeEmphasis = compactDisc.TrackHasEmphasis;
if(wasPlaying) SetPlayingState(true); if(wasPlaying == true) SetPlayingState(true);
} }
/// <summary> /// <summary>
@@ -193,10 +194,10 @@ namespace RedBookPlayer.Hardware
/// </summary> /// </summary>
public void FastForward() public void FastForward()
{ {
if(_opticalDisc == null || !_opticalDisc.Initialized) if(OpticalDisc == null || !OpticalDisc.Initialized)
return; return;
_opticalDisc.CurrentSector = Math.Min(_opticalDisc.TotalSectors, _opticalDisc.CurrentSector + 75); OpticalDisc.CurrentSector = Math.Min(OpticalDisc.TotalSectors, OpticalDisc.CurrentSector + 75);
} }
/// <summary> /// <summary>
@@ -204,112 +205,38 @@ namespace RedBookPlayer.Hardware
/// </summary> /// </summary>
public void Rewind() public void Rewind()
{ {
if(_opticalDisc == null || !_opticalDisc.Initialized) if(OpticalDisc == null || !OpticalDisc.Initialized)
return; return;
if(_opticalDisc.CurrentSector >= 75) if(OpticalDisc.CurrentSector >= 75)
_opticalDisc.CurrentSector -= 75; OpticalDisc.CurrentSector -= 75;
} }
#endregion #endregion
#region Helpers #region Helpers
/// <summary>
/// Generate the digit string to be interpreted by the frontend
/// </summary>
/// <returns>String representing the digits for the frontend</returns>
public string GenerateDigitString()
{
// If the disc isn't initialized, return all '-' characters
if(_opticalDisc == null || !_opticalDisc.Initialized)
return string.Empty.PadLeft(20, '-');
// Otherwise, take the current time into account
ulong sectorTime = GetCurrentSectorTime();
int[] numbers = new int[]
{
_opticalDisc.CurrentTrackNumber + 1,
_opticalDisc.CurrentTrackIndex,
(int)(sectorTime / (75 * 60)),
(int)(sectorTime / 75 % 60),
(int)(sectorTime % 75),
_opticalDisc.TotalTracks,
_opticalDisc.TotalIndexes,
(int)(_opticalDisc.TotalTime / (75 * 60)),
(int)(_opticalDisc.TotalTime / 75 % 60),
(int)(_opticalDisc.TotalTime % 75),
};
return string.Join("", numbers.Select(i => i.ToString().PadLeft(2, '0').Substring(0, 2)));
}
/// <summary>
/// Update the data context for the frontend
/// </summary>
/// <param name="dataContext">Data context to be updated</param>
public void UpdateDataContext(PlayerViewModel dataContext)
{
if(!Initialized || dataContext == null)
return;
dataContext.Playing = Playing;
dataContext.CurrentSector = GetCurrentSectorTime();
dataContext.TotalSectors = _opticalDisc.TotalTime;
dataContext.Volume = App.Settings.Volume;
dataContext.ApplyDeEmphasis = ApplyDeEmphasis;
dataContext.HiddenTrack = _opticalDisc.TimeOffset > 150;
if(_opticalDisc is CompactDisc compactDisc)
{
dataContext.QuadChannel = compactDisc.QuadChannel;
dataContext.IsDataTrack = compactDisc.IsDataTrack;
dataContext.CopyAllowed = compactDisc.CopyAllowed;
dataContext.TrackHasEmphasis = compactDisc.TrackHasEmphasis;
}
else
{
dataContext.QuadChannel = false;
dataContext.IsDataTrack = _opticalDisc.TrackType != TrackType.Audio;
dataContext.CopyAllowed = false;
dataContext.TrackHasEmphasis = false;
}
}
/// <summary>
/// Update the internal values from the frontend
/// </summary>
/// <param name="dataContext">Data context to update from</param>
public void UpdateModel(PlayerViewModel dataContext)
{
if(!Initialized || dataContext == null)
return;
SetPlayingState(dataContext.Playing);
App.Settings.Volume = dataContext.Volume;
_soundOutput?.ToggleDeEmphasis(dataContext.ApplyDeEmphasis);
}
/// <summary> /// <summary>
/// Get current sector time, accounting for offsets /// Get current sector time, accounting for offsets
/// </summary> /// </summary>
/// <returns>ulong representing the current sector time</returns> /// <returns>ulong representing the current sector time</returns>
private ulong GetCurrentSectorTime() public ulong GetCurrentSectorTime()
{ {
ulong sectorTime = _opticalDisc.CurrentSector; ulong sectorTime = OpticalDisc.CurrentSector;
if(_opticalDisc.SectionStartSector != 0) if (OpticalDisc.SectionStartSector != 0)
sectorTime -= _opticalDisc.SectionStartSector; sectorTime -= OpticalDisc.SectionStartSector;
else else
sectorTime += _opticalDisc.TimeOffset; sectorTime += OpticalDisc.TimeOffset;
return sectorTime; return sectorTime;
} }
/// <summary>
/// Set if de-emphasis should be applied
/// </summary>
/// <param name="apply">True to enable, false to disable</param>
public void SetDeEmphasis(bool apply) => _soundOutput?.ToggleDeEmphasis(apply);
#endregion #endregion
} }
} }