--- license: apache-2.0 base_model: - THUDM/CogVideoX-5b language: - en tags: - video-generation - paddlemix --- English | [็ฎ€ไฝ“ไธญๆ–‡](README_zh.md) # VCtrl

๐Ÿค— Huggingface Space | ๐ŸŒ Github | ๐Ÿ“œ arxiv | ๐Ÿ“ท Project

## Model Introduction **VCtrl** is a versatile video generation control model that introduces an auxiliary conditional encoder to flexibly connect with various control modules while avoiding large-scale retraining of the original generator. The model efficiently transmits control signals through sparse residual connections and standardizes diverse control inputs into a unified representation via a consistent encoding process. Task-specific masks are further incorporated to enhance adaptability. Thanks to this unified and flexible design, VCtrl can be widely applied in โ€‹**character animation**, โ€‹**scene transition**, โ€‹**video editing**, and other video generation scenarios. The table below provides detailed information about the video generation models we offer:
Model Name VCtrl-Canny VCtrl-Mask VCtrl-Pose
Video Resolution 720 * 480 720 * 480 720 * 480 & 480 * 720
Inference Precision FP16(Recommended)
Single GPU VRAM Usage V100: 32GB minimum*
Inference Speed
(Step = 25, FP16)
Single A100: ~300s(49 frames)
Single V100: ~400s(49 frames)
Prompt Language English*
Prompt Length Limit 224 Tokens
Video Length T2V model supports only 49 frames, I2V model can extend to any frame count
Frame Rate 30 FPS
## Quick Start ๐Ÿค— This model is now supported for deployment using the ppdiffusers library from paddlemix. Follow the steps below to get started. **We recommend visiting our [github](https://github.com/PaddlePaddle/PaddleMIX/tree/develop/ppdiffusers/examples/ppvctrl) for a better experience.** 1. Install dependencies ```shell # Clone the PaddleMIX repository git clone https://github.com/PaddlePaddle/PaddleMIX.git # Install paddlemix cd PaddleMIX pip install -e . # Install ppdiffusers pip install -e ppdiffusers # Install paddlenlp pip install paddlenlp==v3.0.0-beta2 # Navigate to the vctrl directory cd ppdiffusers/examples/ppvctrl # Install other required dependencies pip install -r requirements.txt # Install paddlex pip install paddlex==3.0.0b2 ``` 2. Run the code ```python import os import paddle import numpy as np from decord import VideoReader from moviepy.editor import ImageSequenceClip from PIL import Image from ppdiffusers import ( CogVideoXDDIMScheduler, CogVideoXTransformer3DVCtrlModel, CogVideoXVCtrlPipeline, VCtrlModel, ) def write_mp4(video_path, samples, fps=8): clip = ImageSequenceClip(samples, fps=fps) clip.write_videofile(video_path, audio_codec="aac") def save_vid_side_by_side(batch_output, validation_control_images, output_folder, fps): flattened_batch_output = [img for sublist in batch_output for img in sublist] ori_video_path = output_folder + "/origin_predict.mp4" video_path = output_folder + "/test_1.mp4" ori_final_images = [] final_images = [] outputs = [] def get_concat_h(im1, im2): dst = Image.new("RGB", (im1.width + im2.width, max(im1.height, im2.height))) dst.paste(im1, (0, 0)) dst.paste(im2, (im1.width, 0)) return dst for image_list in zip(validation_control_images, flattened_batch_output): predict_img = image_list[1].resize(image_list[0].size) result = get_concat_h(image_list[0], predict_img) ori_final_images.append(np.array(image_list[1])) final_images.append(np.array(result)) outputs.append(np.array(predict_img)) write_mp4(ori_video_path, ori_final_images, fps=fps) write_mp4(video_path, final_images, fps=fps) output_path = output_folder + "/output.mp4" write_mp4(output_path, outputs, fps=fps) def load_images_from_folder_to_pil(folder): images = [] valid_extensions = {".jpg", ".jpeg", ".png", ".bmp", ".gif", ".tiff"} def frame_number(filename): new_pattern_match = re.search("frame_(\\d+)_7fps", filename) if new_pattern_match: return int(new_pattern_match.group(1)) matches = re.findall("\\d+", filename) if matches: if matches[-1] == "0000" and len(matches) > 1: return int(matches[-2]) return int(matches[-1]) return float("inf") sorted_files = sorted(os.listdir(folder), key=frame_number) for filename in sorted_files: ext = os.path.splitext(filename)[1].lower() if ext in valid_extensions: img = Image.open(os.path.join(folder, filename)).convert("RGB") images.append(img) return images def load_images_from_video_to_pil(video_path): images = [] vr = VideoReader(video_path) length = len(vr) for idx in range(length): frame = vr[idx].asnumpy() images.append(Image.fromarray(frame)) return images validation_control_images = load_images_from_video_to_pil('your_path') prompt = 'Group of fishes swimming in aquarium.' vctrl = VCtrlModel.from_pretrained( paddlemix/vctrl-5b-t2v-canny, low_cpu_mem_usage=True, paddle_dtype=paddle.float16 ) pipeline = CogVideoXVCtrlPipeline.from_pretrained( paddlemix/cogvideox-5b-vctrl, vctrl=vctrl, paddle_dtype=paddle.float16, low_cpu_mem_usage=True, map_location="cpu", ) pipeline.scheduler = CogVideoXDDIMScheduler.from_config(pipeline.scheduler.config, timestep_spacing="trailing") pipeline.vae.enable_tiling() pipeline.vae.enable_slicing() task='canny' final_result=[] video = pipeline( prompt=prompt, num_inference_steps=25, num_frames=49, guidance_scale=35, generator=paddle.Generator().manual_seed(42), conditioning_frames=validation_control_images[:num_frames], conditioning_frame_indices=list(range(num_frames)), conditioning_scale=1.0, width=720, height=480, task='canny', conditioning_masks=validation_mask_images[:num_frames] if task == "mask" else None, vctrl_layout_type='spacing', ).frames[0] final_result.append(video) save_vid_side_by_side(final_result, validation_control_images[:num_frames], 'save.mp4', fps=30) ``` ## In-Depth Exploration Welcome to our [github]("https://github.com/PaddlePaddle/PaddleMIX/tree/develop/ppdiffusers/examples/ppvctrl"), where you will find๏ผš 1. More detailed technical explanations and code walkthroughs. 2. Algorithm details for extracting control conditions. 3. Detailed code for model inference. 4. Project update logs and more interactive opportunities. 5. PaddleMix toolchain to help you better utilize the model.