2022-02-11 09:07:11 +11:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-or-later
|
2022-02-09 16:21:21 +11:00
|
|
|
* Copyright 2001-2002 NaN Holding BV. All rights reserved.
|
|
|
|
* 2003-2009 Blender Foundation.
|
2022-02-11 09:07:11 +11:00
|
|
|
* 2005-2006 Peter Schlaile <peter [at] schlaile [dot] de> */
|
2020-11-16 05:02:30 +01:00
|
|
|
|
|
|
|
/** \file
|
|
|
|
* \ingroup bke
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
#include "DNA_scene_types.h"
|
|
|
|
#include "DNA_sequence_types.h"
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
#include "BLI_ghash.h"
|
2020-11-16 05:02:30 +01:00
|
|
|
#include "BLI_listbase.h"
|
|
|
|
|
|
|
|
#include "BKE_scene.h"
|
|
|
|
|
2020-12-19 06:44:57 +01:00
|
|
|
#include "SEQ_iterator.h"
|
2022-04-21 01:57:55 +02:00
|
|
|
#include "SEQ_relations.h"
|
2022-04-04 12:52:48 +02:00
|
|
|
#include "SEQ_render.h"
|
2021-09-21 09:38:30 +02:00
|
|
|
#include "SEQ_time.h"
|
|
|
|
#include "render.h"
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \Iterator API
|
|
|
|
* \{ */
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
bool SEQ_iterator_ensure(SeqCollection *collection, SeqIterator *iterator, Sequence **r_seq)
|
2020-11-16 05:02:30 +01:00
|
|
|
{
|
2021-05-07 10:25:13 +02:00
|
|
|
if (iterator->iterator_initialized) {
|
|
|
|
return true;
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
if (BLI_gset_len(collection->set) == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
iterator->collection = collection;
|
|
|
|
BLI_gsetIterator_init(&iterator->gsi, iterator->collection->set);
|
|
|
|
iterator->iterator_initialized = true;
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
*r_seq = BLI_gsetIterator_getKey(&iterator->gsi);
|
|
|
|
BLI_gsetIterator_step(&iterator->gsi);
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
return true;
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
Sequence *SEQ_iterator_yield(SeqIterator *iterator)
|
2020-11-16 05:02:30 +01:00
|
|
|
{
|
2021-05-07 10:25:13 +02:00
|
|
|
Sequence *seq = BLI_gsetIterator_done(&iterator->gsi) ? NULL :
|
|
|
|
BLI_gsetIterator_getKey(&iterator->gsi);
|
|
|
|
BLI_gsetIterator_step(&iterator->gsi);
|
|
|
|
return seq;
|
|
|
|
}
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-08-20 16:30:34 +02:00
|
|
|
static bool seq_for_each_recursive(ListBase *seqbase, SeqForEachFunc callback, void *user_data)
|
|
|
|
{
|
|
|
|
LISTBASE_FOREACH (Sequence *, seq, seqbase) {
|
|
|
|
if (!callback(seq, user_data)) {
|
|
|
|
/* Callback signaled stop, return. */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (seq->type == SEQ_TYPE_META) {
|
|
|
|
if (!seq_for_each_recursive(&seq->seqbase, callback, user_data)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SEQ_for_each_callback(ListBase *seqbase, SeqForEachFunc callback, void *user_data)
|
|
|
|
{
|
|
|
|
seq_for_each_recursive(seqbase, callback, user_data);
|
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
void SEQ_collection_free(SeqCollection *collection)
|
|
|
|
{
|
|
|
|
BLI_gset_free(collection->set, NULL);
|
|
|
|
MEM_freeN(collection);
|
|
|
|
}
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-07-01 22:14:29 +02:00
|
|
|
SeqCollection *SEQ_collection_create(const char *name)
|
2021-05-07 10:25:13 +02:00
|
|
|
{
|
2021-07-01 22:14:29 +02:00
|
|
|
SeqCollection *collection = MEM_callocN(sizeof(SeqCollection), name);
|
2021-05-07 10:25:13 +02:00
|
|
|
collection->set = BLI_gset_new(
|
|
|
|
BLI_ghashutil_ptrhash, BLI_ghashutil_ptrcmp, "SeqCollection GSet");
|
|
|
|
return collection;
|
|
|
|
}
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-06-30 16:37:14 +10:00
|
|
|
uint SEQ_collection_len(const SeqCollection *collection)
|
2021-06-29 20:12:19 +02:00
|
|
|
{
|
|
|
|
return BLI_gset_len(collection->set);
|
|
|
|
}
|
|
|
|
|
2021-08-24 00:46:34 +02:00
|
|
|
bool SEQ_collection_has_strip(const Sequence *seq, const SeqCollection *collection)
|
|
|
|
{
|
|
|
|
return BLI_gset_haskey(collection->set, seq);
|
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
SeqCollection *SEQ_query_by_reference(Sequence *seq_reference,
|
|
|
|
ListBase *seqbase,
|
|
|
|
void seq_query_func(Sequence *seq_reference,
|
|
|
|
ListBase *seqbase,
|
|
|
|
SeqCollection *collection))
|
|
|
|
{
|
2021-07-01 22:14:29 +02:00
|
|
|
SeqCollection *collection = SEQ_collection_create(__func__);
|
2021-05-07 10:25:13 +02:00
|
|
|
seq_query_func(seq_reference, seqbase, collection);
|
|
|
|
return collection;
|
|
|
|
}
|
|
|
|
bool SEQ_collection_append_strip(Sequence *seq, SeqCollection *collection)
|
|
|
|
{
|
2021-10-07 00:35:57 +02:00
|
|
|
void **key;
|
|
|
|
if (BLI_gset_ensure_p_ex(collection->set, seq, &key)) {
|
2021-05-07 10:25:13 +02:00
|
|
|
return false;
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
2021-10-07 00:35:57 +02:00
|
|
|
|
|
|
|
*key = (void *)seq;
|
2021-05-07 10:25:13 +02:00
|
|
|
return true;
|
|
|
|
}
|
2020-11-16 05:02:30 +01:00
|
|
|
|
VSE: Fix rendering inconsistency
Fix issue described in T87678, which was partially a bug and partially
change in intended(at least as far as I can tell) behaior.
Function `evaluate_seq_frame_gen` that was partially responsible for
filtering strips in stack for rendering wasn't working correctly.
Intended functionality seems to be removing all effect inputs from stack
as it is unlikely that user would want these to be blended in. However
there was logic to exclude effects placed into same input, which because
of weak implementation caused, that any effect input, that is effect as
well will be considered to be part of stack to be blended in.
This bug was apparently used to produce effects like glow over original
image.
Even though this is originally unintended, I have kept this logic, but
I have made it explicit.
Another change is request made in T87678 to make it possible to keep
effect inputs as part of stack when they are placed above the effect,
which would imply that blending is intended. This change is again
explicitly defined.
Whole implementation has been refactored, so logic is consolidated
and code should be as explicit as possible and more readable.
`must_render_strip function` may be still quite hard to read, not sure
if I can make it nicer.
Last change is for remove gaps feature code - it used same rendering
code, which may be reason why its logic was split in first place.
Now it uses sequencer iterator, which will definitely be faster than
original code, but I could have used `LISTBASE_FOREACH` in this case.
Reviewed By: sergey
Differential Revision: https://developer.blender.org/D11301
2021-05-19 22:59:33 +02:00
|
|
|
bool SEQ_collection_remove_strip(Sequence *seq, SeqCollection *collection)
|
|
|
|
{
|
|
|
|
return BLI_gset_remove(collection->set, seq, NULL);
|
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
void SEQ_collection_merge(SeqCollection *collection_dst, SeqCollection *collection_src)
|
|
|
|
{
|
|
|
|
Sequence *seq;
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, collection_src) {
|
|
|
|
SEQ_collection_append_strip(seq, collection_dst);
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
2021-05-07 10:25:13 +02:00
|
|
|
SEQ_collection_free(collection_src);
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
|
|
|
|
2021-07-26 14:55:14 +02:00
|
|
|
void SEQ_collection_exclude(SeqCollection *collection, SeqCollection *exclude_elements)
|
|
|
|
{
|
|
|
|
Sequence *seq;
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, exclude_elements) {
|
|
|
|
SEQ_collection_remove_strip(seq, collection);
|
|
|
|
}
|
|
|
|
SEQ_collection_free(exclude_elements);
|
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
void SEQ_collection_expand(ListBase *seqbase,
|
|
|
|
SeqCollection *collection,
|
|
|
|
void seq_query_func(Sequence *seq_reference,
|
|
|
|
ListBase *seqbase,
|
|
|
|
SeqCollection *collection))
|
2020-11-16 05:02:30 +01:00
|
|
|
{
|
2021-05-07 10:25:13 +02:00
|
|
|
/* Collect expanded results for each sequence in provided SeqIteratorCollection. */
|
2021-08-20 16:30:34 +02:00
|
|
|
SeqCollection *query_matches = SEQ_collection_create(__func__);
|
2020-11-16 05:02:30 +01:00
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
Sequence *seq;
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, collection) {
|
2021-08-20 16:30:34 +02:00
|
|
|
SEQ_collection_merge(query_matches, SEQ_query_by_reference(seq, seqbase, seq_query_func));
|
2021-05-07 10:25:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Merge all expanded results in provided SeqIteratorCollection. */
|
2021-08-20 16:30:34 +02:00
|
|
|
SEQ_collection_merge(collection, query_matches);
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
|
|
|
|
2021-07-26 14:55:14 +02:00
|
|
|
SeqCollection *SEQ_collection_duplicate(SeqCollection *collection)
|
|
|
|
{
|
|
|
|
SeqCollection *duplicate = SEQ_collection_create(__func__);
|
|
|
|
Sequence *seq;
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, collection) {
|
|
|
|
SEQ_collection_append_strip(seq, duplicate);
|
|
|
|
}
|
|
|
|
return duplicate;
|
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
/** \} */
|
|
|
|
|
2021-08-20 16:30:34 +02:00
|
|
|
static void query_all_strips_recursive(ListBase *seqbase, SeqCollection *collection)
|
|
|
|
{
|
|
|
|
LISTBASE_FOREACH (Sequence *, seq, seqbase) {
|
|
|
|
if (seq->type == SEQ_TYPE_META) {
|
|
|
|
query_all_strips_recursive(&seq->seqbase, collection);
|
|
|
|
}
|
|
|
|
SEQ_collection_append_strip(seq, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
SeqCollection *SEQ_query_all_strips_recursive(ListBase *seqbase)
|
2020-11-16 05:02:30 +01:00
|
|
|
{
|
2021-07-01 22:14:29 +02:00
|
|
|
SeqCollection *collection = SEQ_collection_create(__func__);
|
2021-05-07 10:25:13 +02:00
|
|
|
LISTBASE_FOREACH (Sequence *, seq, seqbase) {
|
|
|
|
if (seq->type == SEQ_TYPE_META) {
|
2021-08-20 16:30:34 +02:00
|
|
|
query_all_strips_recursive(&seq->seqbase, collection);
|
2021-05-07 10:25:13 +02:00
|
|
|
}
|
|
|
|
SEQ_collection_append_strip(seq, collection);
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
2021-05-07 10:25:13 +02:00
|
|
|
return collection;
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
|
|
|
|
VSE: Fix rendering inconsistency
Fix issue described in T87678, which was partially a bug and partially
change in intended(at least as far as I can tell) behaior.
Function `evaluate_seq_frame_gen` that was partially responsible for
filtering strips in stack for rendering wasn't working correctly.
Intended functionality seems to be removing all effect inputs from stack
as it is unlikely that user would want these to be blended in. However
there was logic to exclude effects placed into same input, which because
of weak implementation caused, that any effect input, that is effect as
well will be considered to be part of stack to be blended in.
This bug was apparently used to produce effects like glow over original
image.
Even though this is originally unintended, I have kept this logic, but
I have made it explicit.
Another change is request made in T87678 to make it possible to keep
effect inputs as part of stack when they are placed above the effect,
which would imply that blending is intended. This change is again
explicitly defined.
Whole implementation has been refactored, so logic is consolidated
and code should be as explicit as possible and more readable.
`must_render_strip function` may be still quite hard to read, not sure
if I can make it nicer.
Last change is for remove gaps feature code - it used same rendering
code, which may be reason why its logic was split in first place.
Now it uses sequencer iterator, which will definitely be faster than
original code, but I could have used `LISTBASE_FOREACH` in this case.
Reviewed By: sergey
Differential Revision: https://developer.blender.org/D11301
2021-05-19 22:59:33 +02:00
|
|
|
SeqCollection *SEQ_query_all_strips(ListBase *seqbase)
|
|
|
|
{
|
2021-07-01 22:14:29 +02:00
|
|
|
SeqCollection *collection = SEQ_collection_create(__func__);
|
2021-10-08 09:29:15 +02:00
|
|
|
LISTBASE_FOREACH (Sequence *, seq, seqbase) {
|
|
|
|
SEQ_collection_append_strip(seq, collection);
|
|
|
|
}
|
VSE: Fix rendering inconsistency
Fix issue described in T87678, which was partially a bug and partially
change in intended(at least as far as I can tell) behaior.
Function `evaluate_seq_frame_gen` that was partially responsible for
filtering strips in stack for rendering wasn't working correctly.
Intended functionality seems to be removing all effect inputs from stack
as it is unlikely that user would want these to be blended in. However
there was logic to exclude effects placed into same input, which because
of weak implementation caused, that any effect input, that is effect as
well will be considered to be part of stack to be blended in.
This bug was apparently used to produce effects like glow over original
image.
Even though this is originally unintended, I have kept this logic, but
I have made it explicit.
Another change is request made in T87678 to make it possible to keep
effect inputs as part of stack when they are placed above the effect,
which would imply that blending is intended. This change is again
explicitly defined.
Whole implementation has been refactored, so logic is consolidated
and code should be as explicit as possible and more readable.
`must_render_strip function` may be still quite hard to read, not sure
if I can make it nicer.
Last change is for remove gaps feature code - it used same rendering
code, which may be reason why its logic was split in first place.
Now it uses sequencer iterator, which will definitely be faster than
original code, but I could have used `LISTBASE_FOREACH` in this case.
Reviewed By: sergey
Differential Revision: https://developer.blender.org/D11301
2021-05-19 22:59:33 +02:00
|
|
|
return collection;
|
|
|
|
}
|
|
|
|
|
2021-05-07 10:25:13 +02:00
|
|
|
SeqCollection *SEQ_query_selected_strips(ListBase *seqbase)
|
2020-11-16 05:02:30 +01:00
|
|
|
{
|
2021-07-01 22:14:29 +02:00
|
|
|
SeqCollection *collection = SEQ_collection_create(__func__);
|
2021-05-07 10:25:13 +02:00
|
|
|
LISTBASE_FOREACH (Sequence *, seq, seqbase) {
|
|
|
|
if ((seq->flag & SELECT) == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
SEQ_collection_append_strip(seq, collection);
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
2021-05-07 10:25:13 +02:00
|
|
|
return collection;
|
2020-11-16 05:02:30 +01:00
|
|
|
}
|
2021-05-18 23:28:00 +02:00
|
|
|
|
2021-09-21 09:38:30 +02:00
|
|
|
static SeqCollection *query_strips_at_frame(ListBase *seqbase, const int timeline_frame)
|
|
|
|
{
|
|
|
|
SeqCollection *collection = SEQ_collection_create(__func__);
|
|
|
|
|
|
|
|
LISTBASE_FOREACH (Sequence *, seq, seqbase) {
|
|
|
|
if (SEQ_time_strip_intersects_frame(seq, timeline_frame)) {
|
|
|
|
SEQ_collection_append_strip(seq, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return collection;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void collection_filter_channel_up_to_incl(SeqCollection *collection, const int channel)
|
|
|
|
{
|
|
|
|
Sequence *seq;
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, collection) {
|
|
|
|
if (seq->machine <= channel) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
SEQ_collection_remove_strip(seq, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if seq must be rendered. This depends on whole stack in some cases, not only seq itself.
|
|
|
|
* Order of applying these conditions is important. */
|
|
|
|
static bool must_render_strip(const Sequence *seq, SeqCollection *strips_at_timeline_frame)
|
|
|
|
{
|
|
|
|
bool seq_have_effect_in_stack = false;
|
|
|
|
Sequence *seq_iter;
|
|
|
|
SEQ_ITERATOR_FOREACH (seq_iter, strips_at_timeline_frame) {
|
|
|
|
/* Strips is below another strip with replace blending are not rendered. */
|
|
|
|
if (seq_iter->blend_mode == SEQ_BLEND_REPLACE && seq->machine < seq_iter->machine) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-04-21 01:57:55 +02:00
|
|
|
if ((seq_iter->type & SEQ_TYPE_EFFECT) != 0 &&
|
|
|
|
SEQ_relation_is_effect_of_strip(seq_iter, seq)) {
|
2021-09-21 09:38:30 +02:00
|
|
|
/* Strips in same channel or higher than its effect are rendered. */
|
|
|
|
if (seq->machine >= seq_iter->machine) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
/* Mark that this strip has effect in stack, that is above the strip. */
|
|
|
|
seq_have_effect_in_stack = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* All effects are rendered (with respect to conditions above). */
|
|
|
|
if ((seq->type & SEQ_TYPE_EFFECT) != 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If strip has effects in stack, and all effects are above this strip, it is not rendered. */
|
|
|
|
if (seq_have_effect_in_stack) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove strips we don't want to render from collection. */
|
2022-04-04 12:52:48 +02:00
|
|
|
static void collection_filter_rendered_strips(ListBase *channels, SeqCollection *collection)
|
2021-09-21 09:38:30 +02:00
|
|
|
{
|
|
|
|
Sequence *seq;
|
|
|
|
|
|
|
|
/* Remove sound strips and muted strips from collection, because these are not rendered.
|
|
|
|
* Function #must_render_strip() don't have to check for these strips anymore. */
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, collection) {
|
2022-04-04 12:52:48 +02:00
|
|
|
if (seq->type == SEQ_TYPE_SOUND_RAM || SEQ_render_is_muted(channels, seq)) {
|
2021-09-21 09:38:30 +02:00
|
|
|
SEQ_collection_remove_strip(seq, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, collection) {
|
|
|
|
if (must_render_strip(seq, collection)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
SEQ_collection_remove_strip(seq, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-04 12:52:48 +02:00
|
|
|
SeqCollection *SEQ_query_rendered_strips(ListBase *channels,
|
|
|
|
ListBase *seqbase,
|
2021-09-21 09:38:30 +02:00
|
|
|
const int timeline_frame,
|
|
|
|
const int displayed_channel)
|
|
|
|
{
|
|
|
|
SeqCollection *collection = query_strips_at_frame(seqbase, timeline_frame);
|
|
|
|
if (displayed_channel != 0) {
|
|
|
|
collection_filter_channel_up_to_incl(collection, displayed_channel);
|
|
|
|
}
|
2022-04-04 12:52:48 +02:00
|
|
|
collection_filter_rendered_strips(channels, collection);
|
2021-09-21 09:38:30 +02:00
|
|
|
return collection;
|
|
|
|
}
|
|
|
|
|
2021-08-27 12:59:46 +02:00
|
|
|
SeqCollection *SEQ_query_unselected_strips(ListBase *seqbase)
|
|
|
|
{
|
|
|
|
SeqCollection *collection = SEQ_collection_create(__func__);
|
|
|
|
LISTBASE_FOREACH (Sequence *, seq, seqbase) {
|
|
|
|
if ((seq->flag & SELECT) != 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
SEQ_collection_append_strip(seq, collection);
|
|
|
|
}
|
|
|
|
return collection;
|
|
|
|
}
|
|
|
|
|
2021-05-18 23:28:00 +02:00
|
|
|
void SEQ_query_strip_effect_chain(Sequence *seq_reference,
|
|
|
|
ListBase *seqbase,
|
|
|
|
SeqCollection *collection)
|
|
|
|
{
|
|
|
|
if (!SEQ_collection_append_strip(seq_reference, collection)) {
|
|
|
|
return; /* Strip is already in set, so all effects connected to it are as well. */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find all strips that seq_reference is connected to. */
|
|
|
|
if (seq_reference->type & SEQ_TYPE_EFFECT) {
|
|
|
|
if (seq_reference->seq1) {
|
|
|
|
SEQ_query_strip_effect_chain(seq_reference->seq1, seqbase, collection);
|
|
|
|
}
|
|
|
|
if (seq_reference->seq2) {
|
|
|
|
SEQ_query_strip_effect_chain(seq_reference->seq2, seqbase, collection);
|
|
|
|
}
|
|
|
|
if (seq_reference->seq3) {
|
|
|
|
SEQ_query_strip_effect_chain(seq_reference->seq3, seqbase, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find all strips connected to seq_reference. */
|
|
|
|
LISTBASE_FOREACH (Sequence *, seq_test, seqbase) {
|
|
|
|
if (seq_test->seq1 == seq_reference || seq_test->seq2 == seq_reference ||
|
|
|
|
seq_test->seq3 == seq_reference) {
|
|
|
|
SEQ_query_strip_effect_chain(seq_test, seqbase, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-09-21 09:38:30 +02:00
|
|
|
|
|
|
|
void SEQ_filter_selected_strips(SeqCollection *collection)
|
|
|
|
{
|
|
|
|
Sequence *seq;
|
|
|
|
SEQ_ITERATOR_FOREACH (seq, collection) {
|
|
|
|
if ((seq->flag & SELECT) == 0) {
|
|
|
|
SEQ_collection_remove_strip(seq, collection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|