/* SPDX-License-Identifier: GPL-2.0-or-later * Copyright 2022 Blender Foundation. All rights reserved. */ /** \file * \ingroup sequencer */ #include #include "DNA_anim_types.h" #include "DNA_scene_types.h" #include "DNA_sequence_types.h" #include "BKE_animsys.h" #include "BKE_fcurve.h" #include "BLI_ghash.h" #include "BLI_listbase.h" #include "BLI_string.h" #include "DEG_depsgraph.h" #include "SEQ_animation.h" bool SEQ_animation_curves_exist(Scene *scene) { return scene->adt != NULL && scene->adt->action != NULL && !BLI_listbase_is_empty(&scene->adt->action->curves); } bool SEQ_animation_drivers_exist(Scene *scene) { return scene->adt != NULL && !BLI_listbase_is_empty(&scene->adt->drivers); } /* r_prefix + [" + escaped_name + "] + \0 */ #define SEQ_RNAPATH_MAXSTR ((30 + 2 + (SEQ_NAME_MAXSTR * 2) + 2) + 1) static size_t sequencer_rna_path_prefix(char str[SEQ_RNAPATH_MAXSTR], const char *name) { char name_esc[SEQ_NAME_MAXSTR * 2]; BLI_str_escape(name_esc, name, sizeof(name_esc)); return BLI_snprintf_rlen( str, SEQ_RNAPATH_MAXSTR, "sequence_editor.sequences_all[\"%s\"]", name_esc); } GSet *SEQ_fcurves_by_strip_get(const Sequence *seq, ListBase *fcurve_base) { char rna_path[SEQ_RNAPATH_MAXSTR]; size_t rna_path_len = sequencer_rna_path_prefix(rna_path, seq->name + 2); /* Only allocate `fcurves` if it's needed as it's possible there is no animation for `seq`. */ GSet *fcurves = NULL; LISTBASE_FOREACH (FCurve *, fcurve, fcurve_base) { if (STREQLEN(fcurve->rna_path, rna_path, rna_path_len)) { if (fcurves == NULL) { fcurves = BLI_gset_ptr_new(__func__); } BLI_gset_add(fcurves, fcurve); } } return fcurves; } #undef SEQ_RNAPATH_MAXSTR void SEQ_offset_animdata(Scene *scene, Sequence *seq, int ofs) { if (!SEQ_animation_curves_exist(scene) || ofs == 0) { return; } GSet *fcurves = SEQ_fcurves_by_strip_get(seq, &scene->adt->action->curves); if (fcurves == NULL) { return; } GSET_FOREACH_BEGIN (FCurve *, fcu, fcurves) { uint i; if (fcu->bezt) { for (i = 0; i < fcu->totvert; i++) { BezTriple *bezt = &fcu->bezt[i]; bezt->vec[0][0] += ofs; bezt->vec[1][0] += ofs; bezt->vec[2][0] += ofs; } } if (fcu->fpt) { for (i = 0; i < fcu->totvert; i++) { FPoint *fpt = &fcu->fpt[i]; fpt->vec[0] += ofs; } } } GSET_FOREACH_END(); BLI_gset_free(fcurves, NULL); DEG_id_tag_update(&scene->adt->action->id, ID_RECALC_ANIMATION); } void SEQ_free_animdata(Scene *scene, Sequence *seq) { if (!SEQ_animation_curves_exist(scene)) { return; } GSet *fcurves = SEQ_fcurves_by_strip_get(seq, &scene->adt->action->curves); if (fcurves == NULL) { return; } GSET_FOREACH_BEGIN (FCurve *, fcu, fcurves) { BLI_remlink(&scene->adt->action->curves, fcu); BKE_fcurve_free(fcu); } GSET_FOREACH_END(); BLI_gset_free(fcurves, NULL); } void SEQ_animation_backup_original(Scene *scene, SeqAnimationBackup *backup) { if (SEQ_animation_curves_exist(scene)) { BLI_movelisttolist(&backup->curves, &scene->adt->action->curves); } if (SEQ_animation_drivers_exist(scene)) { BLI_movelisttolist(&backup->drivers, &scene->adt->drivers); } } void SEQ_animation_restore_original(Scene *scene, SeqAnimationBackup *backup) { if (!BLI_listbase_is_empty(&backup->curves)) { BLI_movelisttolist(&scene->adt->action->curves, &backup->curves); } if (!BLI_listbase_is_empty(&backup->drivers)) { BLI_movelisttolist(&scene->adt->drivers, &backup->drivers); } } static void seq_animation_duplicate(Scene *scene, Sequence *seq, ListBase *dst, ListBase *src) { if (seq->type == SEQ_TYPE_META) { LISTBASE_FOREACH (Sequence *, meta_child, &seq->seqbase) { seq_animation_duplicate(scene, meta_child, dst, src); } } GSet *fcurves = SEQ_fcurves_by_strip_get(seq, src); if (fcurves == NULL) { return; } GSET_FOREACH_BEGIN (FCurve *, fcu, fcurves) { FCurve *fcu_cpy = BKE_fcurve_copy(fcu); BLI_addtail(dst, fcu_cpy); } GSET_FOREACH_END(); BLI_gset_free(fcurves, NULL); } void SEQ_animation_duplicate_backup_to_scene(Scene *scene, Sequence *seq, SeqAnimationBackup *backup) { if (!BLI_listbase_is_empty(&backup->curves)) { seq_animation_duplicate(scene, seq, &scene->adt->action->curves, &backup->curves); } if (!BLI_listbase_is_empty(&backup->drivers)) { seq_animation_duplicate(scene, seq, &scene->adt->drivers, &backup->drivers); } }