C
C#•2d ago
IAmMaddieAtYou

WPF Video Editor program | Help with the (Timeline/TimeMarks)

Hi there! I am in need of help making the timeline and time marks for my program. I personally tend to struggle when it comes to math(dyscalculia hates me) so the concept of this is really not clicking for me!
11 Replies
Buddy
Buddy•2d ago
We do not help in DMs Please use help channels for help, not a redirection to DMs. Plus, more people can help you. If say, that person is asleep, someone else can step in.
IAmMaddieAtYou
IAmMaddieAtYouOP•2d ago
Oh my bad I didn't see that anywhere!
Buddy
Buddy•2d ago
If you are unable to seek help in a public chat, then I'm afraid we can't help.
IAmMaddieAtYou
IAmMaddieAtYouOP•2d ago
I can!
Buddy
Buddy•2d ago
Then ask away.
IAmMaddieAtYou
IAmMaddieAtYouOP•2d ago
😭 first I need to sleep cause it's 3am but I will be back in the morning :ChibiSalute:
IAmMaddieAtYou
IAmMaddieAtYouOP•2d ago
:Wave: Hi there! So my current issue around my video editor like program is the Timeline and Timemarker system. I have a basic one written out but when I scroll it doesn't show the times correctly as well as when I zoom out far it doesn't go farther then showing like 2 seconds in the view.
No description
IAmMaddieAtYou
IAmMaddieAtYouOP•2d ago
Ontop of that when I move the slider it's delayed initially and if I move my time enough no more time markers will be shown.
No description
IAmMaddieAtYou
IAmMaddieAtYouOP•2d ago
Code for majority of it: Scrolling mechanism
timer = new DispatcherTimer();
timer.Interval = TimeSpan.FromMilliseconds(1); // Reduced interval
timer.Tick += Timer_Tick;
timer = new DispatcherTimer();
timer.Interval = TimeSpan.FromMilliseconds(1); // Reduced interval
timer.Tick += Timer_Tick;
I create a dispatcherTimer that updates my timeline
private void Timer_Tick(object sender, EventArgs e)
{
if (mediaPlayer.Source != null && mediaPlayer.NaturalDuration.HasTimeSpan)
{
Wahoo();
SliderS.Value = mediaPlayer.Position.TotalMilliseconds; // Update Slider ONLY
UpdateTimelineScroll();
UpdateTimeline();
}
}
private void Timer_Tick(object sender, EventArgs e)
{
if (mediaPlayer.Source != null && mediaPlayer.NaturalDuration.HasTimeSpan)
{
Wahoo();
SliderS.Value = mediaPlayer.Position.TotalMilliseconds; // Update Slider ONLY
UpdateTimelineScroll();
UpdateTimeline();
}
}
This updates my Timelines Scroll
private void UpdateTimelineScroll()
{
TLines.UpdateLayout();
TRLines.UpdateLayout();
TrackLines.UpdateLayout();
double currentPosition = SliderS.Value;
double viewportWidth = TLines.ViewportWidth;
double totalWidth = TrackLines.Width;

Console.WriteLine($"currentPosition: {currentPosition}, viewportWidth: {viewportWidth}, totalWidth: {totalWidth}, zoomFactor: {zoomFactor}");

if (viewportWidth > 0 && totalWidth > 0 && !double.IsNaN(currentPosition) && !double.IsNaN(zoomFactor))
{

// Calculate desired offset to keep current position centered
double targetOffset = Math.Round((currentPosition * zoomFactor) - (viewportWidth / 2), 2);

// Keep offset within bounds
targetOffset = Math.Max(0, Math.Min(targetOffset, totalWidth - viewportWidth));

if (!double.IsNaN(targetOffset))
{
Dispatcher.BeginInvoke(() =>
{
TLines.ScrollToHorizontalOffset(targetOffset);
TRLines.ScrollToHorizontalOffset(targetOffset);
});
}
else
{
Console.WriteLine("Target offset is NaN");
}
}
else
{
Console.WriteLine("One or more values are invalid");
}
}
private void UpdateTimelineScroll()
{
TLines.UpdateLayout();
TRLines.UpdateLayout();
TrackLines.UpdateLayout();
double currentPosition = SliderS.Value;
double viewportWidth = TLines.ViewportWidth;
double totalWidth = TrackLines.Width;

Console.WriteLine($"currentPosition: {currentPosition}, viewportWidth: {viewportWidth}, totalWidth: {totalWidth}, zoomFactor: {zoomFactor}");

if (viewportWidth > 0 && totalWidth > 0 && !double.IsNaN(currentPosition) && !double.IsNaN(zoomFactor))
{

// Calculate desired offset to keep current position centered
double targetOffset = Math.Round((currentPosition * zoomFactor) - (viewportWidth / 2), 2);

// Keep offset within bounds
targetOffset = Math.Max(0, Math.Min(targetOffset, totalWidth - viewportWidth));

if (!double.IsNaN(targetOffset))
{
Dispatcher.BeginInvoke(() =>
{
TLines.ScrollToHorizontalOffset(targetOffset);
TRLines.ScrollToHorizontalOffset(targetOffset);
});
}
else
{
Console.WriteLine("Target offset is NaN");
}
}
else
{
Console.WriteLine("One or more values are invalid");
}
}
This updates my Timeline and creates the Timemarks code:
public void UpdateTimeline() // Time ruler update
{
if (SliderS != null && mediaPlayer != null)
{
double totalMilliseconds = SliderS.Maximum;
double currentMilliseconds = SliderS.Value;

TrackLines.Children.Clear();

// Set TrackLines width
TrackLines.Width = totalMilliseconds * zoomFactor;

int intervalDivisor = (int)IntervalSlider.Value;
TimeSpan interval;
string timeFormat;

if (zoomFactor >= 3)
{
interval = TimeSpan.FromMilliseconds(totalMilliseconds / intervalDivisor);
timeFormat = @"fff\ ms";
}
else if (zoomFactor >= 2)
{
interval = TimeSpan.FromSeconds(totalMilliseconds / 1000 / intervalDivisor);
timeFormat = @"ss\ s";
}
else if (zoomFactor >= 1)
{
interval = TimeSpan.FromMinutes(totalMilliseconds / 1000 / 60 / intervalDivisor);
timeFormat = @"mm\:ss";
}
else
{
interval = TimeSpan.FromHours(totalMilliseconds / 1000 / 60 / 60 / intervalDivisor);
timeFormat = @"hh\:mm\:ss";
}

double pixelsPerMillisecond = zoomFactor;

// Calculate visible time window
double viewportStartPixels = TLines.HorizontalOffset;
double viewportEndPixels = viewportStartPixels + TLines.ViewportWidth;
double visibleStartTimeMilliseconds = viewportStartPixels / pixelsPerMillisecond;
double visibleEndTimeMilliseconds = viewportEndPixels / pixelsPerMillisecond;

// Ensure visible time window is within media bounds
visibleStartTimeMilliseconds = Math.Max(0, visibleStartTimeMilliseconds);
visibleEndTimeMilliseconds = Math.Min(totalMilliseconds, visibleEndTimeMilliseconds);

TimeSpan currentTime = TimeSpan.FromMilliseconds(visibleStartTimeMilliseconds);
string lasttick = "";

while (currentTime.TotalMilliseconds <= visibleEndTimeMilliseconds)
{
double position = currentTime.TotalMilliseconds * pixelsPerMillisecond;

Line tick = new Line
{
X1 = position,
X2 = position,
Y1 = TrackLines.ActualHeight - ((currentTime.TotalMilliseconds % (interval.TotalMilliseconds * 5) == 0) ? 10 : 5),
Y2 = TrackLines.ActualHeight,
Stroke = System.Windows.Media.Brushes.Black,
StrokeThickness = 1
};
TrackLines.Children.Add(tick);

if (currentTime.TotalMilliseconds % interval.TotalMilliseconds == 0)
{
string timeLabel = currentTime.ToString(timeFormat);
if (lasttick != timeLabel)
{
lasttick = timeLabel;
TextBlock label = new TextBlock
{
Text = timeLabel,
Foreground = System.Windows.Media.Brushes.Black,
FontSize = 10
};
Canvas.SetLeft(label, position - (label.ActualWidth / 2));
Canvas.SetTop(label, TrackLines.ActualHeight - 20);
TrackLines.Children.Add(label);
}
}
currentTime = currentTime.Add(interval);
}
}
}
public void UpdateTimeline() // Time ruler update
{
if (SliderS != null && mediaPlayer != null)
{
double totalMilliseconds = SliderS.Maximum;
double currentMilliseconds = SliderS.Value;

TrackLines.Children.Clear();

// Set TrackLines width
TrackLines.Width = totalMilliseconds * zoomFactor;

int intervalDivisor = (int)IntervalSlider.Value;
TimeSpan interval;
string timeFormat;

if (zoomFactor >= 3)
{
interval = TimeSpan.FromMilliseconds(totalMilliseconds / intervalDivisor);
timeFormat = @"fff\ ms";
}
else if (zoomFactor >= 2)
{
interval = TimeSpan.FromSeconds(totalMilliseconds / 1000 / intervalDivisor);
timeFormat = @"ss\ s";
}
else if (zoomFactor >= 1)
{
interval = TimeSpan.FromMinutes(totalMilliseconds / 1000 / 60 / intervalDivisor);
timeFormat = @"mm\:ss";
}
else
{
interval = TimeSpan.FromHours(totalMilliseconds / 1000 / 60 / 60 / intervalDivisor);
timeFormat = @"hh\:mm\:ss";
}

double pixelsPerMillisecond = zoomFactor;

// Calculate visible time window
double viewportStartPixels = TLines.HorizontalOffset;
double viewportEndPixels = viewportStartPixels + TLines.ViewportWidth;
double visibleStartTimeMilliseconds = viewportStartPixels / pixelsPerMillisecond;
double visibleEndTimeMilliseconds = viewportEndPixels / pixelsPerMillisecond;

// Ensure visible time window is within media bounds
visibleStartTimeMilliseconds = Math.Max(0, visibleStartTimeMilliseconds);
visibleEndTimeMilliseconds = Math.Min(totalMilliseconds, visibleEndTimeMilliseconds);

TimeSpan currentTime = TimeSpan.FromMilliseconds(visibleStartTimeMilliseconds);
string lasttick = "";

while (currentTime.TotalMilliseconds <= visibleEndTimeMilliseconds)
{
double position = currentTime.TotalMilliseconds * pixelsPerMillisecond;

Line tick = new Line
{
X1 = position,
X2 = position,
Y1 = TrackLines.ActualHeight - ((currentTime.TotalMilliseconds % (interval.TotalMilliseconds * 5) == 0) ? 10 : 5),
Y2 = TrackLines.ActualHeight,
Stroke = System.Windows.Media.Brushes.Black,
StrokeThickness = 1
};
TrackLines.Children.Add(tick);

if (currentTime.TotalMilliseconds % interval.TotalMilliseconds == 0)
{
string timeLabel = currentTime.ToString(timeFormat);
if (lasttick != timeLabel)
{
lasttick = timeLabel;
TextBlock label = new TextBlock
{
Text = timeLabel,
Foreground = System.Windows.Media.Brushes.Black,
FontSize = 10
};
Canvas.SetLeft(label, position - (label.ActualWidth / 2));
Canvas.SetTop(label, TrackLines.ActualHeight - 20);
TrackLines.Children.Add(label);
}
}
currentTime = currentTime.Add(interval);
}
}
}
private void PlaybackSlider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
{

if (mediaPlayer.NaturalDuration.HasTimeSpan)
{
mediaPlayer.Position = TimeSpan.FromMilliseconds(SliderS.Value); // Update Media ONLY
UpdateTimelineScroll();
UpdateTimeline();
}

}
private void PlaybackSlider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
{

if (mediaPlayer.NaturalDuration.HasTimeSpan)
{
mediaPlayer.Position = TimeSpan.FromMilliseconds(SliderS.Value); // Update Media ONLY
UpdateTimelineScroll();
UpdateTimeline();
}

}
private void UpdateTimelineWidth()
{
double newTimelineWidth = totalDurationMilliseconds * zoomFactor;
TrackBX.Width = newTimelineWidth;
TrackLines.Width = newTimelineWidth;
}
private void UpdateTimelineWidth()
{
double newTimelineWidth = totalDurationMilliseconds * zoomFactor;
TrackBX.Width = newTimelineWidth;
TrackLines.Width = newTimelineWidth;
}
If anyone could help me improve this system and fix the issues that would mean alot!
bighugemassive3
bighugemassive3•16h ago
I made a video editor called FramePFX so maybe that will help you. See TimelineControl.cs
IAmMaddieAtYou
IAmMaddieAtYouOP•13h ago
I was just looking at your github post trying to find it thank you!

Did you find this page helpful?