| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * - Blender Foundation, 2003-2009 | 
					
						
							|  |  |  |  * - Peter Schlaile <peter [at] schlaile [dot] de> 2005/2006 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bke | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <math.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | #include "DNA_mask_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | #include "DNA_scene_types.h"
 | 
					
						
							|  |  |  | #include "DNA_sequence_types.h"
 | 
					
						
							|  |  |  | #include "DNA_sound_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_listbase.h"
 | 
					
						
							|  |  |  | #include "BLI_path_util.h"
 | 
					
						
							|  |  |  | #include "BLI_string.h"
 | 
					
						
							|  |  |  | #include "BLI_string_utf8.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BKE_context.h"
 | 
					
						
							|  |  |  | #include "BKE_image.h"
 | 
					
						
							|  |  |  | #include "BKE_lib_id.h"
 | 
					
						
							|  |  |  | #include "BKE_main.h"
 | 
					
						
							|  |  |  | #include "BKE_mask.h"
 | 
					
						
							|  |  |  | #include "BKE_movieclip.h"
 | 
					
						
							|  |  |  | #include "BKE_scene.h"
 | 
					
						
							|  |  |  | #include "BKE_sound.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "DEG_depsgraph_query.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 07:19:30 +02:00
										 |  |  | #include "IMB_colormanagement.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | #include "IMB_imbuf.h"
 | 
					
						
							|  |  |  | #include "IMB_imbuf_types.h"
 | 
					
						
							|  |  |  | #include "IMB_metadata.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 06:44:57 +01:00
										 |  |  | #include "SEQ_add.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | #include "SEQ_effects.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-19 06:44:57 +01:00
										 |  |  | #include "SEQ_relations.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | #include "SEQ_render.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-19 06:44:57 +01:00
										 |  |  | #include "SEQ_select.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | #include "SEQ_sequencer.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-19 06:44:57 +01:00
										 |  |  | #include "SEQ_time.h"
 | 
					
						
							|  |  |  | #include "SEQ_transform.h"
 | 
					
						
							|  |  |  | #include "SEQ_utils.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "multiview.h"
 | 
					
						
							|  |  |  | #include "proxy.h"
 | 
					
						
							|  |  |  | #include "utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Initialize common SeqLoadData members | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData to be initialized | 
					
						
							|  |  |  |  * \param name: strip name (can be NULL) | 
					
						
							|  |  |  |  * \param path: path to file that is used as strip input (can be NULL) | 
					
						
							|  |  |  |  * \param start_frame: timeline frame where strip will be created | 
					
						
							|  |  |  |  * \param channel: timeline channel where strip will be created | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void SEQ_add_load_data_init(SeqLoadData *load_data, | 
					
						
							|  |  |  |                             const char *name, | 
					
						
							|  |  |  |                             const char *path, | 
					
						
							|  |  |  |                             const int start_frame, | 
					
						
							|  |  |  |                             const int channel) | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   memset(load_data, 0, sizeof(SeqLoadData)); | 
					
						
							|  |  |  |   if (name != NULL) { | 
					
						
							|  |  |  |     BLI_strncpy(load_data->name, name, sizeof(load_data->name)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (path != NULL) { | 
					
						
							|  |  |  |     BLI_strncpy(load_data->path, path, sizeof(load_data->path)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   load_data->start_frame = start_frame; | 
					
						
							|  |  |  |   load_data->channel = channel; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | static void seq_add_generic_update(Scene *scene, ListBase *seqbase, Sequence *seq) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   SEQ_sequence_base_unique_name_recursive(seqbase, seq); | 
					
						
							|  |  |  |   SEQ_time_update_sequence_bounds(scene, seq); | 
					
						
							| 
									
										
										
										
											2021-05-19 22:45:43 +02:00
										 |  |  |   SEQ_sort(seqbase); | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   SEQ_relations_invalidate_cache_composite(scene, seq); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | static void seq_add_set_name(Sequence *seq, SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-03-21 14:00:21 +11:00
										 |  |  |   if (load_data->name[0] != '\0') { | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     BLI_strncpy(seq->name + 2, load_data->name, sizeof(seq->name) - 2); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     if (seq->type == SEQ_TYPE_SCENE) { | 
					
						
							|  |  |  |       BLI_strncpy(seq->name + 2, load_data->scene->id.name + 2, sizeof(seq->name) - 2); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     else if (seq->type == SEQ_TYPE_MOVIECLIP) { | 
					
						
							|  |  |  |       BLI_strncpy(seq->name + 2, load_data->clip->id.name + 2, sizeof(seq->name) - 2); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     else if (seq->type == SEQ_TYPE_MASK) { | 
					
						
							|  |  |  |       BLI_strncpy(seq->name + 2, load_data->mask->id.name + 2, sizeof(seq->name) - 2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if ((seq->type & SEQ_TYPE_EFFECT) != 0) { | 
					
						
							|  |  |  |       BLI_strncpy(seq->name + 2, SEQ_sequence_give_name(seq), sizeof(seq->name) - 2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { /* Image, sound and movie. */ | 
					
						
							|  |  |  |       BLI_strncpy_utf8(seq->name + 2, load_data->name, sizeof(seq->name) - 2); | 
					
						
							|  |  |  |       BLI_utf8_invalid_strip(seq->name + 2, strlen(seq->name + 2)); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 07:19:30 +02:00
										 |  |  | static void seq_add_set_view_transform(Scene *scene, Sequence *seq, SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const char *strip_colorspace = seq->strip->colorspace_settings.name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (load_data->flags & SEQ_LOAD_SET_VIEW_TRANSFORM) { | 
					
						
							|  |  |  |     const char *role_colorspace_byte; | 
					
						
							|  |  |  |     role_colorspace_byte = IMB_colormanagement_role_colorspace_name_get(COLOR_ROLE_DEFAULT_BYTE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (STREQ(strip_colorspace, role_colorspace_byte)) { | 
					
						
							|  |  |  |       struct ColorManagedDisplay *display = IMB_colormanagement_display_get_named( | 
					
						
							|  |  |  |           scene->display_settings.display_device); | 
					
						
							|  |  |  |       const char *default_view_transform = | 
					
						
							|  |  |  |           IMB_colormanagement_display_get_default_view_transform_name(display); | 
					
						
							|  |  |  |       STRNCPY(scene->view_settings.view_transform, default_view_transform); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Add scene strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Sequence *SEQ_add_scene_strip(Scene *scene, ListBase *seqbase, struct SeqLoadData *load_data) | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   Sequence *seq = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, SEQ_TYPE_SCENE); | 
					
						
							|  |  |  |   seq->blend_mode = SEQ_TYPE_CROSS; | 
					
						
							|  |  |  |   seq->scene = load_data->scene; | 
					
						
							|  |  |  |   seq->len = load_data->scene->r.efra - load_data->scene->r.sfra + 1; | 
					
						
							|  |  |  |   id_us_ensure_real((ID *)load_data->scene); | 
					
						
							|  |  |  |   seq_add_set_name(seq, load_data); | 
					
						
							|  |  |  |   seq_add_generic_update(scene, seqbase, seq); | 
					
						
							|  |  |  |   return seq; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Add movieclip strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Sequence *SEQ_add_movieclip_strip(Scene *scene, ListBase *seqbase, struct SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Sequence *seq = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, SEQ_TYPE_MOVIECLIP); | 
					
						
							|  |  |  |   seq->blend_mode = SEQ_TYPE_CROSS; | 
					
						
							|  |  |  |   seq->clip = load_data->clip; | 
					
						
							|  |  |  |   seq->len = BKE_movieclip_get_duration(load_data->clip); | 
					
						
							|  |  |  |   id_us_ensure_real((ID *)load_data->clip); | 
					
						
							|  |  |  |   seq_add_set_name(seq, load_data); | 
					
						
							|  |  |  |   seq_add_generic_update(scene, seqbase, seq); | 
					
						
							|  |  |  |   return seq; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Add mask strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Sequence *SEQ_add_mask_strip(Scene *scene, ListBase *seqbase, struct SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Sequence *seq = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, SEQ_TYPE_MASK); | 
					
						
							|  |  |  |   seq->blend_mode = SEQ_TYPE_CROSS; | 
					
						
							|  |  |  |   seq->mask = load_data->mask; | 
					
						
							|  |  |  |   seq->len = BKE_mask_get_duration(load_data->mask); | 
					
						
							|  |  |  |   id_us_ensure_real((ID *)load_data->mask); | 
					
						
							|  |  |  |   seq_add_set_name(seq, load_data); | 
					
						
							|  |  |  |   seq_add_generic_update(scene, seqbase, seq); | 
					
						
							|  |  |  |   return seq; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Add effect strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Sequence *SEQ_add_effect_strip(Scene *scene, ListBase *seqbase, struct SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Sequence *seq = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, load_data->effect.type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   seq->flag |= SEQ_USE_EFFECT_DEFAULT_FADE; | 
					
						
							|  |  |  |   struct SeqEffectHandle sh = SEQ_effect_handle_get(seq); | 
					
						
							|  |  |  |   sh.init(seq); | 
					
						
							|  |  |  |   seq->seq1 = load_data->effect.seq1; | 
					
						
							|  |  |  |   seq->seq2 = load_data->effect.seq2; | 
					
						
							|  |  |  |   seq->seq3 = load_data->effect.seq3; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (seq->type == SEQ_TYPE_COLOR) { | 
					
						
							|  |  |  |     seq->blend_mode = SEQ_TYPE_CROSS; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (seq->type == SEQ_TYPE_ADJUSTMENT) { | 
					
						
							|  |  |  |     seq->blend_mode = SEQ_TYPE_CROSS; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (seq->type == SEQ_TYPE_TEXT) { | 
					
						
							|  |  |  |     seq->blend_mode = SEQ_TYPE_ALPHAOVER; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (SEQ_effect_get_num_inputs(seq->type) == 1) { | 
					
						
							|  |  |  |     seq->blend_mode = seq->seq1->blend_mode; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   if (!load_data->effect.seq1) { | 
					
						
							|  |  |  |     seq->len = 1; /* Effect is generator, set non zero length. */ | 
					
						
							| 
									
										
										
										
											2021-03-20 01:34:12 +01:00
										 |  |  |     SEQ_transform_set_right_handle_frame(seq, load_data->effect.end_frame); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-03-20 01:34:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   SEQ_relations_update_changed_seq_and_deps(scene, seq, 1, 1); /* Runs SEQ_time_update_sequence. */ | 
					
						
							|  |  |  |   seq_add_set_name(seq, load_data); | 
					
						
							|  |  |  |   seq_add_generic_update(scene, seqbase, seq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return seq; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Set directory used by image strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param seq: image strip to be changed | 
					
						
							|  |  |  |  * \param path: directory path | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void SEQ_add_image_set_directory(Sequence *seq, char *path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLI_strncpy(seq->strip->dir, path, sizeof(seq->strip->dir)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Set directory used by image strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param seq: image strip to be changed | 
					
						
							|  |  |  |  * \param strip_frame: frame index of strip to be changed | 
					
						
							|  |  |  |  * \param filename: image filename (only filename, not complete path) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void SEQ_add_image_load_file(Sequence *seq, size_t strip_frame, char *filename) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   StripElem *se = SEQ_render_give_stripelem(seq, seq->start + strip_frame); | 
					
						
							|  |  |  |   BLI_strncpy(se->name, filename, sizeof(se->name)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Set image strip alpha mode | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param seq: image strip to be changed | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void SEQ_add_image_init_alpha_mode(Sequence *seq) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (seq->strip && seq->strip->stripdata) { | 
					
						
							|  |  |  |     char name[FILE_MAX]; | 
					
						
							|  |  |  |     ImBuf *ibuf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     BLI_join_dirfile(name, sizeof(name), seq->strip->dir, seq->strip->stripdata->name); | 
					
						
							|  |  |  |     BLI_path_abs(name, BKE_main_blendfile_path_from_global()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Initialize input color space. */ | 
					
						
							|  |  |  |     if (seq->type == SEQ_TYPE_IMAGE) { | 
					
						
							|  |  |  |       ibuf = IMB_loadiffname( | 
					
						
							|  |  |  |           name, IB_test | IB_alphamode_detect, seq->strip->colorspace_settings.name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Byte images are default to straight alpha, however sequencer
 | 
					
						
							|  |  |  |        * works in premul space, so mark strip to be premultiplied first. | 
					
						
							|  |  |  |        */ | 
					
						
							|  |  |  |       seq->alpha_mode = SEQ_ALPHA_STRAIGHT; | 
					
						
							|  |  |  |       if (ibuf) { | 
					
						
							|  |  |  |         if (ibuf->flags & IB_alphamode_premul) { | 
					
						
							|  |  |  |           seq->alpha_mode = IMA_ALPHA_PREMUL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-16 20:34:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |         IMB_freeImBuf(ibuf); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Add image strip. | 
					
						
							|  |  |  |  * NOTE: Use SEQ_add_image_set_directory() and SEQ_add_image_load_file() to load image sequences | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param main: Main reference | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Sequence *SEQ_add_image_strip(Main *bmain, Scene *scene, ListBase *seqbase, SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Sequence *seq = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, SEQ_TYPE_IMAGE); | 
					
						
							|  |  |  |   seq->blend_mode = SEQ_TYPE_CROSS; /* so alpha adjustment fade to the strip below */ | 
					
						
							|  |  |  |   seq->len = load_data->image.len; | 
					
						
							|  |  |  |   Strip *strip = seq->strip; | 
					
						
							|  |  |  |   strip->stripdata = MEM_callocN(load_data->image.len * sizeof(StripElem), "stripelem"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Multiview settings. */ | 
					
						
							|  |  |  |   if (load_data->use_multiview) { | 
					
						
							|  |  |  |     seq->flag |= SEQ_USE_VIEWS; | 
					
						
							|  |  |  |     seq->views_format = load_data->views_format; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (load_data->stereo3d_format) { | 
					
						
							|  |  |  |     seq->stereo3d_format = load_data->stereo3d_format; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Set initial scale based on load_data->fit_method. */ | 
					
						
							| 
									
										
										
										
											2020-12-16 20:34:26 +01:00
										 |  |  |   char file_path[FILE_MAX]; | 
					
						
							| 
									
										
										
										
											2021-03-30 02:41:12 +02:00
										 |  |  |   BLI_strncpy(file_path, load_data->path, sizeof(file_path)); | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   BLI_path_abs(file_path, BKE_main_blendfile_path(bmain)); | 
					
						
							| 
									
										
										
										
											2020-12-16 20:34:26 +01:00
										 |  |  |   ImBuf *ibuf = IMB_loadiffname(file_path, IB_rect, seq->strip->colorspace_settings.name); | 
					
						
							|  |  |  |   if (ibuf != NULL) { | 
					
						
							| 
									
										
										
										
											2021-03-20 00:08:32 +01:00
										 |  |  |     /* Set image resolution. Assume that all images in sequence are same size. This fields are only
 | 
					
						
							|  |  |  |      * informative. */ | 
					
						
							|  |  |  |     StripElem *strip_elem = strip->stripdata; | 
					
						
							|  |  |  |     for (int i = 0; i < load_data->image.len; i++) { | 
					
						
							|  |  |  |       strip_elem->orig_width = ibuf->x; | 
					
						
							|  |  |  |       strip_elem->orig_height = ibuf->y; | 
					
						
							|  |  |  |       strip_elem++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  |     SEQ_set_scale_to_fit( | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |         seq, ibuf->x, ibuf->y, scene->r.xsch, scene->r.ysch, load_data->fit_method); | 
					
						
							| 
									
										
										
										
											2020-12-16 20:34:26 +01:00
										 |  |  |     IMB_freeImBuf(ibuf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   /* Set Last active directory. */ | 
					
						
							|  |  |  |   BLI_strncpy(scene->ed->act_imagedir, seq->strip->dir, sizeof(scene->ed->act_imagedir)); | 
					
						
							| 
									
										
										
										
											2021-05-05 07:19:30 +02:00
										 |  |  |   seq_add_set_view_transform(scene, seq, load_data); | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   seq_add_set_name(seq, load_data); | 
					
						
							|  |  |  |   seq_add_generic_update(scene, seqbase, seq); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return seq; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef WITH_AUDASPACE
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Add sound strip. | 
					
						
							|  |  |  |  * NOTE: Use SEQ_add_image_set_directory() and SEQ_add_image_load_file() to load image sequences | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param main: Main reference | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | Sequence *SEQ_add_sound_strip(Main *bmain, Scene *scene, ListBase *seqbase, SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   bSound *sound = BKE_sound_new_file(bmain, load_data->path); /* Handles relative paths. */ | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   SoundInfo info; | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   bool sound_loaded = BKE_sound_info_get(bmain, sound, &info); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!sound_loaded && !load_data->allow_invalid_file) { | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |     BKE_id_free(bmain, sound); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   if (info.specs.channels == SOUND_CHANNELS_INVALID && !load_data->allow_invalid_file) { | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |     BKE_id_free(bmain, sound); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   Sequence *seq = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, SEQ_TYPE_SOUND_RAM); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   seq->sound = sound; | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   seq->scene_sound = NULL; | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* We add a very small negative offset here, because
 | 
					
						
							|  |  |  |    * ceil(132.0) == 133.0, not nice with videos, see T47135. */ | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   seq->len = MAX2(1, (int)ceil((double)info.length * FPS - 1e-4)); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   Strip *strip = seq->strip; | 
					
						
							|  |  |  |   /* We only need 1 element to store the filename. */ | 
					
						
							|  |  |  |   StripElem *se = strip->stripdata = se = MEM_callocN(sizeof(StripElem), "stripelem"); | 
					
						
							|  |  |  |   BLI_split_dirfile(load_data->path, strip->dir, se->name, sizeof(strip->dir), sizeof(se->name)); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   if (seq != NULL && seq->sound != NULL) { | 
					
						
							|  |  |  |     if (load_data->flags & SEQ_LOAD_SOUND_MONO) { | 
					
						
							|  |  |  |       seq->sound->flags |= SOUND_FLAGS_MONO; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     if (load_data->flags & SEQ_LOAD_SOUND_CACHE) { | 
					
						
							|  |  |  |       if (seq->sound) { | 
					
						
							|  |  |  |         seq->sound->flags |= SOUND_FLAGS_CACHING; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   /* Set Last active directory. */ | 
					
						
							|  |  |  |   BLI_strncpy(scene->ed->act_sounddir, strip->dir, FILE_MAXDIR); | 
					
						
							|  |  |  |   seq_add_set_name(seq, load_data); | 
					
						
							|  |  |  |   seq_add_generic_update(scene, seqbase, seq); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return seq; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | #else   // WITH_AUDASPACE
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | Sequence *SEQ_add_sound_strip(Main *UNUSED(bmain), | 
					
						
							|  |  |  |                               Scene *UNUSED(scene), | 
					
						
							|  |  |  |                               ListBase *UNUSED(seqbase), | 
					
						
							| 
									
										
										
										
											2021-03-02 17:47:21 +01:00
										 |  |  |                               SeqLoadData *UNUSED(load_data)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | #endif  // WITH_AUDASPACE
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 23:55:39 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Add meta strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Sequence *SEQ_add_meta_strip(Scene *scene, ListBase *seqbase, SeqLoadData *load_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Allocate sequence. */ | 
					
						
							|  |  |  |   Sequence *seqm = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, SEQ_TYPE_META); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Set name. */ | 
					
						
							|  |  |  |   seq_add_set_name(seqm, load_data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Set frames start and length. */ | 
					
						
							|  |  |  |   seqm->start = load_data->start_frame; | 
					
						
							|  |  |  |   seqm->len = 1; | 
					
						
							|  |  |  |   SEQ_time_update_sequence(scene, seqm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return seqm; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Add movie strip. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param main: Main reference | 
					
						
							|  |  |  |  * \param scene: Scene where strips will be added | 
					
						
							|  |  |  |  * \param seqbase: ListBase where strips will be added | 
					
						
							|  |  |  |  * \param load_data: SeqLoadData with information necessary to create strip | 
					
						
							|  |  |  |  * \return created strip | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | Sequence *SEQ_add_movie_strip(Main *bmain, Scene *scene, ListBase *seqbase, SeqLoadData *load_data) | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   char path[sizeof(load_data->path)]; | 
					
						
							|  |  |  |   BLI_strncpy(path, load_data->path, sizeof(path)); | 
					
						
							|  |  |  |   BLI_path_abs(path, BKE_main_blendfile_path(bmain)); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   char colorspace[64] = "\0"; /* MAX_COLORSPACE_NAME */ | 
					
						
							|  |  |  |   bool is_multiview_loaded = false; | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   const int totfiles = seq_num_files(scene, load_data->views_format, load_data->use_multiview); | 
					
						
							|  |  |  |   struct anim **anim_arr = MEM_callocN(sizeof(struct anim *) * totfiles, "Video files"); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   int i; | 
					
						
							| 
									
										
										
										
											2021-03-21 14:08:32 +11:00
										 |  |  |   int orig_width = 0; | 
					
						
							|  |  |  |   int orig_height = 0; | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   if (load_data->use_multiview && (load_data->views_format == R_IMF_VIEWS_INDIVIDUAL)) { | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |     char prefix[FILE_MAX]; | 
					
						
							|  |  |  |     const char *ext = NULL; | 
					
						
							|  |  |  |     size_t j = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     BKE_scene_multiview_view_prefix_get(scene, path, prefix, &ext); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (prefix[0] != '\0') { | 
					
						
							|  |  |  |       for (i = 0; i < totfiles; i++) { | 
					
						
							|  |  |  |         char str[FILE_MAX]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         seq_multiview_name(scene, i, prefix, ext, str, FILE_MAX); | 
					
						
							|  |  |  |         anim_arr[j] = openanim(str, IB_rect, 0, colorspace); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (anim_arr[j]) { | 
					
						
							|  |  |  |           seq_anim_add_suffix(scene, anim_arr[j], i); | 
					
						
							|  |  |  |           j++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       is_multiview_loaded = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (is_multiview_loaded == false) { | 
					
						
							|  |  |  |     anim_arr[0] = openanim(path, IB_rect, 0, colorspace); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   if (anim_arr[0] == NULL && !load_data->allow_invalid_file) { | 
					
						
							|  |  |  |     MEM_freeN(anim_arr); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   Sequence *seq = SEQ_sequence_alloc( | 
					
						
							|  |  |  |       seqbase, load_data->start_frame, load_data->channel, SEQ_TYPE_MOVIE); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   /* Multiview settings. */ | 
					
						
							|  |  |  |   if (load_data->use_multiview) { | 
					
						
							|  |  |  |     seq->flag |= SEQ_USE_VIEWS; | 
					
						
							|  |  |  |     seq->views_format = load_data->views_format; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (load_data->stereo3d_format) { | 
					
						
							|  |  |  |     seq->stereo3d_format = load_data->stereo3d_format; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < totfiles; i++) { | 
					
						
							|  |  |  |     if (anim_arr[i]) { | 
					
						
							|  |  |  |       StripAnim *sanim = MEM_mallocN(sizeof(StripAnim), "Strip Anim"); | 
					
						
							|  |  |  |       BLI_addtail(&seq->anims, sanim); | 
					
						
							|  |  |  |       sanim->anim = anim_arr[i]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   seq->blend_mode = SEQ_TYPE_CROSS; /* so alpha adjustment fade to the strip below */ | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-31 18:12:44 +02:00
										 |  |  |   float video_fps = 0.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   if (anim_arr[0] != NULL) { | 
					
						
							|  |  |  |     seq->len = IMB_anim_get_duration(anim_arr[0], IMB_TC_RECORD_RUN); | 
					
						
							| 
									
										
										
										
											2020-12-16 20:34:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     IMB_anim_load_metadata(anim_arr[0]); | 
					
						
							| 
									
										
										
										
											2020-12-16 20:34:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-31 18:12:44 +02:00
										 |  |  |     short fps_denom; | 
					
						
							|  |  |  |     float fps_num; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IMB_anim_get_fps(anim_arr[0], &fps_denom, &fps_num, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     video_fps = fps_denom / fps_num; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     /* Adjust scene's frame rate settings to match. */ | 
					
						
							|  |  |  |     if (load_data->flags & SEQ_LOAD_MOVIE_SYNC_FPS) { | 
					
						
							| 
									
										
										
										
											2021-05-31 18:12:44 +02:00
										 |  |  |       scene->r.frs_sec = fps_denom; | 
					
						
							|  |  |  |       scene->r.frs_sec_base = fps_num; | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |     /* Set initial scale based on load_data->fit_method. */ | 
					
						
							| 
									
										
										
										
											2021-03-21 14:08:32 +11:00
										 |  |  |     orig_width = IMB_anim_get_image_width(anim_arr[0]); | 
					
						
							|  |  |  |     orig_height = IMB_anim_get_image_height(anim_arr[0]); | 
					
						
							|  |  |  |     SEQ_set_scale_to_fit( | 
					
						
							|  |  |  |         seq, orig_width, orig_height, scene->r.xsch, scene->r.ysch, load_data->fit_method); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   seq->len = MAX2(1, seq->len); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   BLI_strncpy(seq->strip->colorspace_settings.name, | 
					
						
							|  |  |  |               colorspace, | 
					
						
							|  |  |  |               sizeof(seq->strip->colorspace_settings.name)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   Strip *strip = seq->strip; | 
					
						
							|  |  |  |   /* We only need 1 element for MOVIE strips. */ | 
					
						
							|  |  |  |   StripElem *se; | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   strip->stripdata = se = MEM_callocN(sizeof(StripElem), "stripelem"); | 
					
						
							| 
									
										
										
										
											2021-03-21 14:08:32 +11:00
										 |  |  |   strip->stripdata->orig_width = orig_width; | 
					
						
							|  |  |  |   strip->stripdata->orig_height = orig_height; | 
					
						
							| 
									
										
										
										
											2021-05-31 18:12:44 +02:00
										 |  |  |   strip->stripdata->orig_fps = video_fps; | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   BLI_split_dirfile(load_data->path, strip->dir, se->name, sizeof(strip->dir), sizeof(se->name)); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 07:19:30 +02:00
										 |  |  |   seq_add_set_view_transform(scene, seq, load_data); | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   seq_add_set_name(seq, load_data); | 
					
						
							|  |  |  |   seq_add_generic_update(scene, seqbase, seq); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   MEM_freeN(anim_arr); | 
					
						
							|  |  |  |   return seq; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  | /* note: caller should run SEQ_time_update_sequence(scene, seq) after */ | 
					
						
							|  |  |  | void SEQ_add_reload_new_file(Main *bmain, Scene *scene, Sequence *seq, const bool lock_range) | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   char path[FILE_MAX]; | 
					
						
							|  |  |  |   int prev_startdisp = 0, prev_enddisp = 0; | 
					
						
							|  |  |  |   /* note: don't rename the strip, will break animation curves */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ELEM(seq->type, | 
					
						
							|  |  |  |            SEQ_TYPE_MOVIE, | 
					
						
							|  |  |  |            SEQ_TYPE_IMAGE, | 
					
						
							|  |  |  |            SEQ_TYPE_SOUND_RAM, | 
					
						
							|  |  |  |            SEQ_TYPE_SCENE, | 
					
						
							|  |  |  |            SEQ_TYPE_META, | 
					
						
							|  |  |  |            SEQ_TYPE_MOVIECLIP, | 
					
						
							|  |  |  |            SEQ_TYPE_MASK) == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (lock_range) { | 
					
						
							|  |  |  |     /* keep so we don't have to move the actual start and end points (only the data) */ | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  |     SEQ_time_update_sequence_bounds(scene, seq); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |     prev_startdisp = seq->startdisp; | 
					
						
							|  |  |  |     prev_enddisp = seq->enddisp; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (seq->type) { | 
					
						
							|  |  |  |     case SEQ_TYPE_IMAGE: { | 
					
						
							|  |  |  |       /* Hack? */ | 
					
						
							|  |  |  |       size_t olen = MEM_allocN_len(seq->strip->stripdata) / sizeof(StripElem); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       seq->len = olen; | 
					
						
							|  |  |  |       seq->len -= seq->anim_startofs; | 
					
						
							|  |  |  |       seq->len -= seq->anim_endofs; | 
					
						
							|  |  |  |       if (seq->len < 0) { | 
					
						
							|  |  |  |         seq->len = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case SEQ_TYPE_MOVIE: { | 
					
						
							|  |  |  |       StripAnim *sanim; | 
					
						
							|  |  |  |       bool is_multiview_loaded = false; | 
					
						
							|  |  |  |       const bool is_multiview = (seq->flag & SEQ_USE_VIEWS) != 0 && | 
					
						
							|  |  |  |                                 (scene->r.scemode & R_MULTIVIEW) != 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       BLI_join_dirfile(path, sizeof(path), seq->strip->dir, seq->strip->stripdata->name); | 
					
						
							|  |  |  |       BLI_path_abs(path, BKE_main_blendfile_path_from_global()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  |       SEQ_relations_sequence_free_anim(seq); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (is_multiview && (seq->views_format == R_IMF_VIEWS_INDIVIDUAL)) { | 
					
						
							|  |  |  |         char prefix[FILE_MAX]; | 
					
						
							|  |  |  |         const char *ext = NULL; | 
					
						
							|  |  |  |         const int totfiles = seq_num_files(scene, seq->views_format, true); | 
					
						
							|  |  |  |         int i = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         BKE_scene_multiview_view_prefix_get(scene, path, prefix, &ext); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (prefix[0] != '\0') { | 
					
						
							|  |  |  |           for (i = 0; i < totfiles; i++) { | 
					
						
							|  |  |  |             struct anim *anim; | 
					
						
							|  |  |  |             char str[FILE_MAX]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             seq_multiview_name(scene, i, prefix, ext, str, FILE_MAX); | 
					
						
							|  |  |  |             anim = openanim(str, | 
					
						
							|  |  |  |                             IB_rect | ((seq->flag & SEQ_FILTERY) ? IB_animdeinterlace : 0), | 
					
						
							|  |  |  |                             seq->streamindex, | 
					
						
							|  |  |  |                             seq->strip->colorspace_settings.name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (anim) { | 
					
						
							|  |  |  |               seq_anim_add_suffix(scene, anim, i); | 
					
						
							|  |  |  |               sanim = MEM_mallocN(sizeof(StripAnim), "Strip Anim"); | 
					
						
							|  |  |  |               BLI_addtail(&seq->anims, sanim); | 
					
						
							|  |  |  |               sanim->anim = anim; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           is_multiview_loaded = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (is_multiview_loaded == false) { | 
					
						
							|  |  |  |         struct anim *anim; | 
					
						
							|  |  |  |         anim = openanim(path, | 
					
						
							|  |  |  |                         IB_rect | ((seq->flag & SEQ_FILTERY) ? IB_animdeinterlace : 0), | 
					
						
							|  |  |  |                         seq->streamindex, | 
					
						
							|  |  |  |                         seq->strip->colorspace_settings.name); | 
					
						
							|  |  |  |         if (anim) { | 
					
						
							|  |  |  |           sanim = MEM_mallocN(sizeof(StripAnim), "Strip Anim"); | 
					
						
							|  |  |  |           BLI_addtail(&seq->anims, sanim); | 
					
						
							|  |  |  |           sanim->anim = anim; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* use the first video as reference for everything */ | 
					
						
							|  |  |  |       sanim = seq->anims.first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if ((!sanim) || (!sanim->anim)) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       IMB_anim_load_metadata(sanim->anim); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       seq->len = IMB_anim_get_duration( | 
					
						
							|  |  |  |           sanim->anim, seq->strip->proxy ? seq->strip->proxy->tc : IMB_TC_RECORD_RUN); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       seq->len -= seq->anim_startofs; | 
					
						
							|  |  |  |       seq->len -= seq->anim_endofs; | 
					
						
							|  |  |  |       if (seq->len < 0) { | 
					
						
							|  |  |  |         seq->len = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case SEQ_TYPE_MOVIECLIP: | 
					
						
							|  |  |  |       if (seq->clip == NULL) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       seq->len = BKE_movieclip_get_duration(seq->clip); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       seq->len -= seq->anim_startofs; | 
					
						
							|  |  |  |       seq->len -= seq->anim_endofs; | 
					
						
							|  |  |  |       if (seq->len < 0) { | 
					
						
							|  |  |  |         seq->len = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SEQ_TYPE_MASK: | 
					
						
							|  |  |  |       if (seq->mask == NULL) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       seq->len = BKE_mask_get_duration(seq->mask); | 
					
						
							|  |  |  |       seq->len -= seq->anim_startofs; | 
					
						
							|  |  |  |       seq->len -= seq->anim_endofs; | 
					
						
							|  |  |  |       if (seq->len < 0) { | 
					
						
							|  |  |  |         seq->len = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SEQ_TYPE_SOUND_RAM: | 
					
						
							|  |  |  | #ifdef WITH_AUDASPACE
 | 
					
						
							|  |  |  |       if (!seq->sound) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       seq->len = ceil((double)BKE_sound_get_length(bmain, seq->sound) * FPS); | 
					
						
							|  |  |  |       seq->len -= seq->anim_startofs; | 
					
						
							|  |  |  |       seq->len -= seq->anim_endofs; | 
					
						
							|  |  |  |       if (seq->len < 0) { | 
					
						
							|  |  |  |         seq->len = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |       UNUSED_VARS(bmain); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case SEQ_TYPE_SCENE: { | 
					
						
							|  |  |  |       seq->len = (seq->scene) ? seq->scene->r.efra - seq->scene->r.sfra + 1 : 0; | 
					
						
							|  |  |  |       seq->len -= seq->anim_startofs; | 
					
						
							|  |  |  |       seq->len -= seq->anim_endofs; | 
					
						
							|  |  |  |       if (seq->len < 0) { | 
					
						
							|  |  |  |         seq->len = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   free_proxy_seq(seq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (lock_range) { | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  |     SEQ_transform_set_left_handle_frame(seq, prev_startdisp); | 
					
						
							|  |  |  |     SEQ_transform_set_right_handle_frame(seq, prev_enddisp); | 
					
						
							|  |  |  |     SEQ_transform_fix_single_image_seq_offsets(seq); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  |   SEQ_time_update_sequence(scene, seq); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  | void SEQ_add_movie_reload_if_needed(struct Main *bmain, | 
					
						
							|  |  |  |                                     struct Scene *scene, | 
					
						
							|  |  |  |                                     struct Sequence *seq, | 
					
						
							|  |  |  |                                     bool *r_was_reloaded, | 
					
						
							|  |  |  |                                     bool *r_can_produce_frames) | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   BLI_assert(seq->type == SEQ_TYPE_MOVIE || | 
					
						
							|  |  |  |              !"This function is only implemented for movie strips."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool must_reload = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-02 12:08:16 +01:00
										 |  |  |   /* The Sequence struct allows for multiple anim structs to be associated with one strip.
 | 
					
						
							|  |  |  |    * This function will return true only if there is at least one 'anim' AND all anims can | 
					
						
							|  |  |  |    * produce frames. */ | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (BLI_listbase_is_empty(&seq->anims)) { | 
					
						
							|  |  |  |     /* No anim present, so reloading is always necessary. */ | 
					
						
							|  |  |  |     must_reload = true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     LISTBASE_FOREACH (StripAnim *, sanim, &seq->anims) { | 
					
						
							|  |  |  |       if (!IMB_anim_can_produce_frames(sanim->anim)) { | 
					
						
							|  |  |  |         /* Anim cannot produce frames, try reloading. */ | 
					
						
							|  |  |  |         must_reload = true; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!must_reload) { | 
					
						
							|  |  |  |     /* There are one or more anims, and all can produce frames. */ | 
					
						
							|  |  |  |     *r_was_reloaded = false; | 
					
						
							|  |  |  |     *r_can_produce_frames = true; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 05:57:27 +01:00
										 |  |  |   SEQ_add_reload_new_file(bmain, scene, seq, true); | 
					
						
							| 
									
										
										
										
											2020-11-16 05:02:30 +01:00
										 |  |  |   *r_was_reloaded = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (BLI_listbase_is_empty(&seq->anims)) { | 
					
						
							|  |  |  |     /* No anims present after reloading => no frames can be produced. */ | 
					
						
							|  |  |  |     *r_can_produce_frames = false; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Check if there are still anims that cannot produce frames. */ | 
					
						
							|  |  |  |   LISTBASE_FOREACH (StripAnim *, sanim, &seq->anims) { | 
					
						
							|  |  |  |     if (!IMB_anim_can_produce_frames(sanim->anim)) { | 
					
						
							|  |  |  |       /* There still is an anim that cannot produce frames. */ | 
					
						
							|  |  |  |       *r_can_produce_frames = false; | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* There are one or more anims, and all can produce frames. */ | 
					
						
							|  |  |  |   *r_can_produce_frames = true; | 
					
						
							|  |  |  | } |