1
0
mirror of https://github.com/Artemis-RGB/Artemis synced 2025-12-13 05:48:35 +00:00

Timeline segments - Move keyframes on segment changes

This commit is contained in:
SpoinkyNL 2020-07-26 17:39:40 +02:00
parent 2045b230c2
commit 8cd9c8a5f8
8 changed files with 121 additions and 48 deletions

View File

@ -1,12 +1,13 @@
using System; using System;
using Artemis.Core.Utilities; using Artemis.Core.Utilities;
using Stylet;
namespace Artemis.Core.Models.Profile.LayerProperties namespace Artemis.Core.Models.Profile.LayerProperties
{ {
/// <summary> /// <summary>
/// For internal use only, use <see cref="LayerPropertyKeyframe{T}" /> instead. /// For internal use only, use <see cref="LayerPropertyKeyframe{T}" /> instead.
/// </summary> /// </summary>
public abstract class BaseLayerPropertyKeyframe public abstract class BaseLayerPropertyKeyframe : PropertyChangedBase
{ {
internal BaseLayerPropertyKeyframe(BaseLayerProperty baseLayerProperty) internal BaseLayerPropertyKeyframe(BaseLayerProperty baseLayerProperty)
{ {
@ -27,5 +28,10 @@ namespace Artemis.Core.Models.Profile.LayerProperties
/// The easing function applied on the value of the keyframe /// The easing function applied on the value of the keyframe
/// </summary> /// </summary>
public Easings.Functions EasingFunction { get; set; } public Easings.Functions EasingFunction { get; set; }
/// <summary>
/// Removes the keyframe from the layer property
/// </summary>
public abstract void Remove();
} }
} }

View File

@ -6,6 +6,8 @@ namespace Artemis.Core.Models.Profile.LayerProperties
public class LayerPropertyKeyframe<T> : BaseLayerPropertyKeyframe public class LayerPropertyKeyframe<T> : BaseLayerPropertyKeyframe
{ {
private TimeSpan _position; private TimeSpan _position;
private T _value;
private LayerProperty<T> _layerProperty;
public LayerPropertyKeyframe(T value, TimeSpan position, Easings.Functions easingFunction, LayerProperty<T> layerProperty) : base(layerProperty) public LayerPropertyKeyframe(T value, TimeSpan position, Easings.Functions easingFunction, LayerProperty<T> layerProperty) : base(layerProperty)
{ {
@ -18,12 +20,20 @@ namespace Artemis.Core.Models.Profile.LayerProperties
/// <summary> /// <summary>
/// The layer property this keyframe is applied to /// The layer property this keyframe is applied to
/// </summary> /// </summary>
public LayerProperty<T> LayerProperty { get; internal set; } public LayerProperty<T> LayerProperty
{
get => _layerProperty;
internal set => SetAndNotify(ref _layerProperty, value);
}
/// <summary> /// <summary>
/// The value of this keyframe /// The value of this keyframe
/// </summary> /// </summary>
public T Value { get; set; } public T Value
{
get => _value;
set => SetAndNotify(ref _value, value);
}
/// <inheritdoc /> /// <inheritdoc />
public override TimeSpan Position public override TimeSpan Position
@ -31,9 +41,15 @@ namespace Artemis.Core.Models.Profile.LayerProperties
get => _position; get => _position;
set set
{ {
_position = value; SetAndNotify(ref _position, value);
LayerProperty.SortKeyframes(); LayerProperty.SortKeyframes();
} }
} }
/// <inheritdoc />
public override void Remove()
{
LayerProperty.RemoveKeyframe(this);
}
} }
} }

View File

@ -19,7 +19,7 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.DisplayConditions.Abstract
public abstract void Update(); public abstract void Update();
public void Delete() public virtual void Delete()
{ {
Model.Parent.RemoveChild(Model); Model.Parent.RemoveChild(Model);
Parent.Update(); Parent.Update();

View File

@ -23,6 +23,7 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.DisplayConditions
private readonly IDataModelVisualizationService _dataModelVisualizationService; private readonly IDataModelVisualizationService _dataModelVisualizationService;
private readonly IEventAggregator _eventAggregator; private readonly IEventAggregator _eventAggregator;
private readonly IProfileEditorService _profileEditorService; private readonly IProfileEditorService _profileEditorService;
private bool _isInitialized;
private DataModelPropertiesViewModel _leftSideDataModel; private DataModelPropertiesViewModel _leftSideDataModel;
private List<DisplayConditionOperator> _operators; private List<DisplayConditionOperator> _operators;
private DataModelPropertiesViewModel _rightSideDataModel; private DataModelPropertiesViewModel _rightSideDataModel;
@ -34,7 +35,6 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.DisplayConditions
private DataModelVisualizationViewModel _selectedRightSideProperty; private DataModelVisualizationViewModel _selectedRightSideProperty;
private List<Type> _supportedInputTypes; private List<Type> _supportedInputTypes;
private bool _isInitialized;
public DisplayConditionPredicateViewModel(DisplayConditionPredicate displayConditionPredicate, DisplayConditionViewModel parent, IProfileEditorService profileEditorService, public DisplayConditionPredicateViewModel(DisplayConditionPredicate displayConditionPredicate, DisplayConditionViewModel parent, IProfileEditorService profileEditorService,
IDataModelVisualizationService dataModelVisualizationService, IDataModelService dataModelService, IEventAggregator eventAggregator) IDataModelVisualizationService dataModelVisualizationService, IDataModelService dataModelService, IEventAggregator eventAggregator)
@ -145,6 +145,12 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.DisplayConditions
RightSideInputViewModel.Submit(); RightSideInputViewModel.Submit();
} }
public override void Delete()
{
base.Delete();
_profileEditorService.UpdateSelectedProfileElement();
}
public void Initialize() public void Initialize()
{ {
// Get the data models // Get the data models

View File

@ -1,4 +1,5 @@
using System; using System;
using System.ComponentModel;
using System.Linq; using System.Linq;
using Artemis.Core.Models.Profile.LayerProperties; using Artemis.Core.Models.Profile.LayerProperties;
using Artemis.Core.Utilities; using Artemis.Core.Utilities;
@ -43,7 +44,7 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
#endregion #endregion
} }
public abstract class TimelineKeyframeViewModel : PropertyChangedBase public abstract class TimelineKeyframeViewModel : PropertyChangedBase, IDisposable
{ {
private readonly IProfileEditorService _profileEditorService; private readonly IProfileEditorService _profileEditorService;
private BindableCollection<TimelineEasingViewModel> _easingViewModels; private BindableCollection<TimelineEasingViewModel> _easingViewModels;
@ -57,6 +58,8 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
_profileEditorService = profileEditorService; _profileEditorService = profileEditorService;
BaseLayerPropertyKeyframe = baseLayerPropertyKeyframe; BaseLayerPropertyKeyframe = baseLayerPropertyKeyframe;
EasingViewModels = new BindableCollection<TimelineEasingViewModel>(); EasingViewModels = new BindableCollection<TimelineEasingViewModel>();
BaseLayerPropertyKeyframe.PropertyChanged += BaseLayerPropertyKeyframeOnPropertyChanged;
} }
public BaseLayerPropertyKeyframe BaseLayerPropertyKeyframe { get; } public BaseLayerPropertyKeyframe BaseLayerPropertyKeyframe { get; }
@ -97,6 +100,12 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
public abstract void Delete(); public abstract void Delete();
private void BaseLayerPropertyKeyframeOnPropertyChanged(object sender, PropertyChangedEventArgs e)
{
if (e.PropertyName == nameof(BaseLayerPropertyKeyframe.Position))
Update(_pixelsPerSecond);
}
#region Easing #region Easing
public void CreateEasingViewModels() public void CreateEasingViewModels()
@ -121,12 +130,6 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
private TimeSpan? _offset; private TimeSpan? _offset;
public void ApplyMovement(TimeSpan cursorTime)
{
UpdatePosition(cursorTime);
Update(_pixelsPerSecond);
}
public void ReleaseMovement() public void ReleaseMovement()
{ {
_offset = null; _offset = null;
@ -154,7 +157,7 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
UpdatePosition(keyframeViewModel.BaseLayerPropertyKeyframe.Position + _offset.Value); UpdatePosition(keyframeViewModel.BaseLayerPropertyKeyframe.Position + _offset.Value);
} }
private void UpdatePosition(TimeSpan position) public void UpdatePosition(TimeSpan position)
{ {
if (position < TimeSpan.Zero) if (position < TimeSpan.Zero)
BaseLayerPropertyKeyframe.Position = TimeSpan.Zero; BaseLayerPropertyKeyframe.Position = TimeSpan.Zero;
@ -167,5 +170,10 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
} }
#endregion #endregion
public void Dispose()
{
BaseLayerPropertyKeyframe.PropertyChanged -= BaseLayerPropertyKeyframeOnPropertyChanged;
}
} }
} }

View File

@ -30,14 +30,17 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
{ {
var keyframes = LayerPropertyViewModel.LayerProperty.Keyframes.ToList(); var keyframes = LayerPropertyViewModel.LayerProperty.Keyframes.ToList();
var toRemove = TimelineKeyframeViewModels.Where(t => !keyframes.Contains(t.BaseLayerPropertyKeyframe)).ToList(); var toRemove = TimelineKeyframeViewModels.Where(t => !keyframes.Contains(t.BaseLayerPropertyKeyframe)).ToList();
foreach (var timelineKeyframeViewModel in toRemove)
timelineKeyframeViewModel.Dispose();
TimelineKeyframeViewModels.RemoveRange(toRemove); TimelineKeyframeViewModels.RemoveRange(toRemove);
TimelineKeyframeViewModels.AddRange( TimelineKeyframeViewModels.AddRange(keyframes
keyframes.Where(k => TimelineKeyframeViewModels.All(t => t.BaseLayerPropertyKeyframe != k)) .Where(k => TimelineKeyframeViewModels.All(t => t.BaseLayerPropertyKeyframe != k))
.Select(k => new TimelineKeyframeViewModel<T>(_profileEditorService, k)) .Select(k => new TimelineKeyframeViewModel<T>(_profileEditorService, k))
); );
} }
else else
TimelineKeyframeViewModels.Clear(); DisposeKeyframeViewModels();
foreach (var timelineKeyframeViewModel in TimelineKeyframeViewModels) foreach (var timelineKeyframeViewModel in TimelineKeyframeViewModels)
timelineKeyframeViewModel.Update(_profileEditorService.PixelsPerSecond); timelineKeyframeViewModel.Update(_profileEditorService.PixelsPerSecond);
@ -49,6 +52,14 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
LayerPropertyViewModel.LayerProperty.KeyframeAdded -= LayerPropertyOnKeyframeModified; LayerPropertyViewModel.LayerProperty.KeyframeAdded -= LayerPropertyOnKeyframeModified;
LayerPropertyViewModel.LayerProperty.KeyframeRemoved -= LayerPropertyOnKeyframeModified; LayerPropertyViewModel.LayerProperty.KeyframeRemoved -= LayerPropertyOnKeyframeModified;
LayerPropertyViewModel.LayerProperty.KeyframesToggled -= LayerPropertyOnKeyframeModified; LayerPropertyViewModel.LayerProperty.KeyframesToggled -= LayerPropertyOnKeyframeModified;
DisposeKeyframeViewModels();
}
private void DisposeKeyframeViewModels()
{
foreach (var timelineKeyframeViewModel in TimelineKeyframeViewModels)
timelineKeyframeViewModel.Dispose();
TimelineKeyframeViewModels.Clear();
} }
private void LayerPropertyOnKeyframeModified(object sender, EventArgs e) private void LayerPropertyOnKeyframeModified(object sender, EventArgs e)

View File

@ -1,5 +1,6 @@
using System; using System;
using System.ComponentModel; using System.ComponentModel;
using System.Linq;
using System.Windows; using System.Windows;
using System.Windows.Controls; using System.Windows.Controls;
using System.Windows.Input; using System.Windows.Input;
@ -130,44 +131,66 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
public void DisableSegment() public void DisableSegment()
{ {
switch (Segment) var keyframes = SelectedProfileElement.GetAllKeyframes();
var startSegmentEnd = SelectedProfileElement.StartSegmentLength;
var mainSegmentEnd = SelectedProfileElement.StartSegmentLength + SelectedProfileElement.MainSegmentLength;
var oldSegmentLength = SegmentLength;
if (Segment == SegmentViewModelType.Start)
{ {
case SegmentViewModelType.Start: // Remove keyframes that fall in this segment
SelectedProfileElement.StartSegmentLength = TimeSpan.Zero; foreach (var baseLayerPropertyKeyframe in keyframes.Where(k => k.Position < startSegmentEnd))
break; baseLayerPropertyKeyframe.Remove();
case SegmentViewModelType.Main: SelectedProfileElement.StartSegmentLength = TimeSpan.Zero;
SelectedProfileElement.MainSegmentLength = TimeSpan.Zero; }
break; else if (Segment == SegmentViewModelType.Main)
case SegmentViewModelType.End: {
SelectedProfileElement.EndSegmentLength = TimeSpan.Zero; // Remove keyframes that fall in this segment
break; foreach (var baseLayerPropertyKeyframe in keyframes.Where(k => k.Position > startSegmentEnd && k.Position < mainSegmentEnd))
default: baseLayerPropertyKeyframe.Remove();
throw new ArgumentOutOfRangeException(); SelectedProfileElement.MainSegmentLength = TimeSpan.Zero;
}
else if (Segment == SegmentViewModelType.End)
{
// Remove keyframes that fall in this segment
foreach (var baseLayerPropertyKeyframe in keyframes.Where(k => k.Position > mainSegmentEnd))
baseLayerPropertyKeyframe.Remove();
SelectedProfileElement.EndSegmentLength = TimeSpan.Zero;
} }
NotifyOfPropertyChange(nameof(SegmentEnabled)); NotifyOfPropertyChange(nameof(SegmentEnabled));
ShiftNextSegment(SegmentLength - oldSegmentLength);
ProfileEditorService.UpdateSelectedProfileElement(); ProfileEditorService.UpdateSelectedProfileElement();
} }
public void EnableSegment() public void EnableSegment()
{ {
switch (Segment) ShiftNextSegment(TimeSpan.FromSeconds(1));
{ if (Segment == SegmentViewModelType.Start)
case SegmentViewModelType.Start: SelectedProfileElement.StartSegmentLength = TimeSpan.FromSeconds(1);
SelectedProfileElement.StartSegmentLength = TimeSpan.FromSeconds(1); else if (Segment == SegmentViewModelType.Main)
break; SelectedProfileElement.MainSegmentLength = TimeSpan.FromSeconds(1);
case SegmentViewModelType.Main: else if (Segment == SegmentViewModelType.End)
SelectedProfileElement.MainSegmentLength = TimeSpan.FromSeconds(1); SelectedProfileElement.EndSegmentLength = TimeSpan.FromSeconds(1);
break;
case SegmentViewModelType.End:
SelectedProfileElement.EndSegmentLength = TimeSpan.FromSeconds(1);
break;
default:
throw new ArgumentOutOfRangeException();
}
NotifyOfPropertyChange(nameof(SegmentEnabled)); NotifyOfPropertyChange(nameof(SegmentEnabled));
ProfileEditorService.UpdateSelectedProfileElement(); ProfileEditorService.UpdateSelectedProfileElement();
UpdateDisplay();
}
public void ShiftNextSegment(TimeSpan amount)
{
var segmentEnd = TimeSpan.Zero;
if (Segment == SegmentViewModelType.Start)
segmentEnd = SelectedProfileElement.StartSegmentLength;
else if (Segment == SegmentViewModelType.Main)
segmentEnd = SelectedProfileElement.StartSegmentLength + SelectedProfileElement.MainSegmentLength;
else if (Segment == SegmentViewModelType.End)
segmentEnd = SelectedProfileElement.TimelineLength;
foreach (var baseLayerPropertyKeyframe in SelectedProfileElement.GetAllKeyframes().Where(k => k.Position >= segmentEnd))
baseLayerPropertyKeyframe.Position += amount;
} }
public void SegmentMouseDown(object sender, MouseButtonEventArgs e) public void SegmentMouseDown(object sender, MouseButtonEventArgs e)
@ -180,6 +203,8 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
{ {
((IInputElement) sender).ReleaseMouseCapture(); ((IInputElement) sender).ReleaseMouseCapture();
_draggingSegment = false; _draggingSegment = false;
ProfileEditorService.UpdateSelectedProfileElement();
} }
public void SegmentMouseMove(object sender, MouseEventArgs e) public void SegmentMouseMove(object sender, MouseEventArgs e)
@ -189,7 +214,7 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
// Get the parent scroll viewer, need that for our position // Get the parent scroll viewer, need that for our position
var parent = VisualTreeUtilities.FindParent<ScrollViewer>((DependencyObject) sender, "TimelineHeaderScrollViewer"); var parent = VisualTreeUtilities.FindParent<ScrollViewer>((DependencyObject) sender, "TimelineHeaderScrollViewer");
var x = Math.Max(0, e.GetPosition(parent).X); var x = Math.Max(0, e.GetPosition(parent).X);
var newTime = TimeSpan.FromSeconds(x / ProfileEditorService.PixelsPerSecond); var newTime = TimeSpan.FromSeconds(x / ProfileEditorService.PixelsPerSecond);
@ -208,7 +233,7 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
else if (Keyboard.IsKeyDown(Key.LeftCtrl) || Keyboard.IsKeyDown(Key.RightCtrl)) else if (Keyboard.IsKeyDown(Key.LeftCtrl) || Keyboard.IsKeyDown(Key.RightCtrl))
newTime = TimeSpan.FromMilliseconds(Math.Round(newTime.TotalMilliseconds / 50.0) * 50.0); newTime = TimeSpan.FromMilliseconds(Math.Round(newTime.TotalMilliseconds / 50.0) * 50.0);
var oldSegmentLength = SegmentLength;
if (Segment == SegmentViewModelType.Start) if (Segment == SegmentViewModelType.Start)
{ {
if (newTime < TimeSpan.FromMilliseconds(100)) if (newTime < TimeSpan.FromMilliseconds(100))
@ -231,12 +256,13 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
NotifyOfPropertyChange(nameof(SegmentLength)); NotifyOfPropertyChange(nameof(SegmentLength));
NotifyOfPropertyChange(nameof(SegmentWidth)); NotifyOfPropertyChange(nameof(SegmentWidth));
ShiftNextSegment(SegmentLength - oldSegmentLength);
UpdateDisplay(); UpdateDisplay();
} }
private void SelectedProfileElementOnPropertyChanged(object sender, PropertyChangedEventArgs e) private void SelectedProfileElementOnPropertyChanged(object sender, PropertyChangedEventArgs e)
{ {
if (e.PropertyName == nameof(RenderProfileElement.StartSegmentLength) || if (e.PropertyName == nameof(RenderProfileElement.StartSegmentLength) ||
e.PropertyName == nameof(RenderProfileElement.MainSegmentLength) || e.PropertyName == nameof(RenderProfileElement.MainSegmentLength) ||
e.PropertyName == nameof(RenderProfileElement.EndSegmentLength)) e.PropertyName == nameof(RenderProfileElement.EndSegmentLength))
{ {

View File

@ -220,7 +220,7 @@ namespace Artemis.UI.Screens.Module.ProfileEditor.LayerProperties.Timeline
foreach (var keyframeViewModel in keyframeViewModels.Where(k => k.IsSelected)) foreach (var keyframeViewModel in keyframeViewModels.Where(k => k.IsSelected))
keyframeViewModel.SaveOffsetToKeyframe(sourceKeyframeViewModel); keyframeViewModel.SaveOffsetToKeyframe(sourceKeyframeViewModel);
sourceKeyframeViewModel.ApplyMovement(cursorTime); sourceKeyframeViewModel.UpdatePosition(cursorTime);
foreach (var keyframeViewModel in keyframeViewModels.Where(k => k.IsSelected)) foreach (var keyframeViewModel in keyframeViewModels.Where(k => k.IsSelected))
keyframeViewModel.ApplyOffsetToKeyframe(sourceKeyframeViewModel); keyframeViewModel.ApplyOffsetToKeyframe(sourceKeyframeViewModel);