Skip to content

Commit

Permalink
feat: Rendering fix for mismatch camera/display files
Browse files Browse the repository at this point in the history
  • Loading branch information
richiemcilroy committed Jan 6, 2025
1 parent 173abf9 commit 1e91c64
Show file tree
Hide file tree
Showing 10 changed files with 485 additions and 128 deletions.
51 changes: 39 additions & 12 deletions apps/desktop/src-tauri/src/export.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,15 +16,33 @@ pub async fn export_video(
force: bool,
use_custom_muxer: bool,
) -> Result<PathBuf, String> {
let metadata = match get_video_metadata(app.clone(), video_id.clone(), Some(VideoType::Screen)).await {
Ok(meta) => meta,
Err(e) => {
sentry::capture_message(&format!("Failed to get video metadata: {}", e), sentry::Level::Error);
return Err("Failed to read video metadata. The recording may be from an incompatible version.".to_string());
}
};
let screen_metadata =
match get_video_metadata(app.clone(), video_id.clone(), Some(VideoType::Screen)).await {
Ok(meta) => meta,
Err(e) => {
sentry::capture_message(
&format!("Failed to get video metadata: {}", e),
sentry::Level::Error,
);
return Err(
"Failed to read video metadata. The recording may be from an incompatible version."
.to_string(),
);
}
};

// Get camera metadata if it exists
let camera_metadata =
get_video_metadata(app.clone(), video_id.clone(), Some(VideoType::Camera))
.await
.ok();

let VideoRecordingMetadata { duration, .. } = metadata;
// Use the longer duration between screen and camera
let duration = screen_metadata.duration.max(
camera_metadata
.map(|m| m.duration)
.unwrap_or(screen_metadata.duration),
);

// Calculate total frames with ceiling to ensure we don't exceed 100%
let total_frames = ((duration * 30.0).ceil() as u32).max(1);
Expand All @@ -42,16 +60,25 @@ pub async fn export_video(
.send(RenderProgress::EstimatedTotalFrames { total_frames })
.ok();

// Create a modified project configuration that accounts for different video lengths
let mut modified_project = project.clone();
if let Some(timeline) = &mut modified_project.timeline {
// Ensure timeline duration matches the longest video
for segment in timeline.segments.iter_mut() {
if segment.end > duration {
segment.end = duration;
}
}
}

let exporter = cap_export::Exporter::new(
project,
modified_project,
output_path.clone(),
move |frame_index| {
// Ensure progress never exceeds total frames
let current_frame = (frame_index + 1).min(total_frames);
progress
.send(RenderProgress::FrameRendered {
current_frame,
})
.send(RenderProgress::FrameRendered { current_frame })
.ok();
},
editor_instance.project_path.clone(),
Expand Down
14 changes: 13 additions & 1 deletion apps/desktop/src-tauri/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@ pub struct App {
pub enum VideoType {
Screen,
Output,
Camera,
}

#[derive(Serialize, Deserialize, specta::Type)]
Expand Down Expand Up @@ -952,6 +953,17 @@ async fn get_video_metadata(

let paths = match video_type {
Some(VideoType::Screen) => content_paths(&project_path, &meta),
Some(VideoType::Camera) => match &meta.content {
Content::SingleSegment { segment } => segment
.camera
.as_ref()
.map_or(vec![], |c| vec![segment.path(&meta, &c.path)]),
Content::MultipleSegments { inner } => inner
.segments
.iter()
.filter_map(|s| s.camera.as_ref().map(|c| inner.path(&meta, &c.path)))
.collect(),
},
Some(VideoType::Output) | None => {
let output_video_path = project_path.join("output").join("result.mp4");
println!("Using output video path: {:?}", output_video_path);
Expand Down Expand Up @@ -1039,7 +1051,7 @@ fn focus_captures_panel(app: AppHandle) {

#[derive(Serialize, Deserialize, specta::Type, Clone)]
#[serde(tag = "type")]
enum RenderProgress {
pub enum RenderProgress {
Starting { total_frames: u32 },
EstimatedTotalFrames { total_frames: u32 },
FrameRendered { current_frame: u32 },
Expand Down
11 changes: 8 additions & 3 deletions apps/desktop/src/routes/editor/Header.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -262,10 +262,12 @@ function ExportButton() {
progress.onmessage = (p) => {
if (p.type === "FrameRendered" && progressState.type === "saving") {
const percentComplete = Math.min(
Math.round((p.current_frame / (progressState.totalFrames || 1)) * 100),
Math.round(
(p.current_frame / (progressState.totalFrames || 1)) * 100
),
100
);

setProgressState({
...progressState,
renderProgress: p.current_frame,
Expand All @@ -280,7 +282,10 @@ function ExportButton() {
});
}
}
if (p.type === "EstimatedTotalFrames" && progressState.type === "saving") {
if (
p.type === "EstimatedTotalFrames" &&
progressState.type === "saving"
) {
setProgressState({
...progressState,
totalFrames: p.total_frames,
Expand Down
2 changes: 1 addition & 1 deletion apps/desktop/src/utils/tauri.ts
Original file line number Diff line number Diff line change
Expand Up @@ -284,7 +284,7 @@ export type UploadProgress = { stage: string; progress: number; message: string
export type UploadResult = { Success: string } | "NotAuthenticated" | "PlanCheckFailed" | "UpgradeRequired"
export type Video = { duration: number; width: number; height: number }
export type VideoRecordingMetadata = { duration: number; size: number }
export type VideoType = "screen" | "output"
export type VideoType = "screen" | "output" | "camera"
export type XY<T> = { x: T; y: T }
export type ZoomMode = "auto" | { manual: { x: number; y: number } }
export type ZoomSegment = { start: number; end: number; amount: number; mode: ZoomMode }
Expand Down
26 changes: 23 additions & 3 deletions crates/editor/src/editor.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
use std::{sync::Arc, time::Instant};

use cap_media::frame_ws::WSFrame;
use cap_project::{BackgroundSource, ProjectConfiguration};
use cap_rendering::{decoder::DecodedFrame, produce_frame, ProjectUniforms, RenderVideoConstants};
use cap_project::{BackgroundSource, ProjectConfiguration, RecordingMeta};
use cap_rendering::{
decoder::DecodedFrame, produce_frame, ProjectRecordings, ProjectUniforms, RenderVideoConstants,
};
use tokio::{
sync::{mpsc, oneshot},
task::JoinHandle,
Expand All @@ -26,6 +28,7 @@ pub struct Renderer {
rx: mpsc::Receiver<RendererMessage>,
frame_tx: flume::Sender<WSFrame>,
render_constants: Arc<RenderVideoConstants>,
total_frames: u32,
}

pub struct RendererHandle {
Expand All @@ -36,13 +39,28 @@ impl Renderer {
pub fn spawn(
render_constants: Arc<RenderVideoConstants>,
frame_tx: flume::Sender<WSFrame>,
meta: &RecordingMeta,
) -> RendererHandle {
let recordings = ProjectRecordings::new(meta);
let mut max_duration = recordings.duration();

// Check camera duration if it exists
if let Some(camera_path) = meta.content.camera_path() {
if let Ok(camera_duration) = recordings.get_source_duration(&camera_path) {
max_duration = max_duration.max(camera_duration);
}
}

let total_frames = (30_f64 * max_duration).ceil() as u32;
println!("Editor total frames: {total_frames}");

let (tx, rx) = mpsc::channel(4);

let this = Self {
rx,
frame_tx,
render_constants,
total_frames,
};

tokio::spawn(this.run());
Expand All @@ -61,7 +79,7 @@ impl Renderer {
camera_frame,
background,
uniforms,
time, // Add this
time,
finished,
} => {
if let Some(task) = frame_task.as_ref() {
Expand All @@ -74,6 +92,7 @@ impl Renderer {

let render_constants = self.render_constants.clone();
let frame_tx = self.frame_tx.clone();
let total_frames = self.total_frames;

frame_task = Some(tokio::spawn(async move {
let frame = produce_frame(
Expand All @@ -83,6 +102,7 @@ impl Renderer {
cap_rendering::Background::from(background),
&uniforms,
time,
total_frames,
)
.await
.unwrap();
Expand Down
6 changes: 5 additions & 1 deletion crates/editor/src/editor_instance.rs
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,11 @@ impl EditorInstance {
.unwrap(),
);

let renderer = Arc::new(editor::Renderer::spawn(render_constants.clone(), frame_tx));
let renderer = Arc::new(editor::Renderer::spawn(
render_constants.clone(),
frame_tx,
&meta,
));

let (preview_tx, preview_rx) = watch::channel(None);

Expand Down
12 changes: 12 additions & 0 deletions crates/project/src/meta.rs
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,18 @@ pub enum Content {
},
}

impl Content {
pub fn camera_path(&self) -> Option<PathBuf> {
match self {
Content::SingleSegment { segment } => segment.camera.as_ref().map(|c| c.path.clone()),
Content::MultipleSegments { inner } => inner
.segments
.first()
.and_then(|s| s.camera.as_ref().map(|c| c.path.clone())),
}
}
}

#[derive(Debug, Clone, Serialize, Deserialize, Type)]
pub struct SingleSegment {
pub display: Display,
Expand Down
Loading

0 comments on commit 1e91c64

Please sign in to comment.