Files
test2/source/blender/sequencer/intern/sound.cc
Falk David 63d3573c2b Cleanup: VSE: Rename occurences of sequence in lookup code
Rename `sequence` -> `strip` for all the lookup code for strips.
2025-01-07 17:04:00 +01:00

349 lines
10 KiB
C++

/* SPDX-FileCopyrightText: 2001-2002 NaN Holding BV. All rights reserved.
* SPDX-FileCopyrightText: 2003-2009 Blender Authors
* SPDX-FileCopyrightText: 2005-2006 Peter Schlaile <peter [at] schlaile [dot] de>
*
* SPDX-License-Identifier: GPL-2.0-or-later */
/** \file
* \ingroup bke
*/
#include <algorithm>
#include <cmath>
#include <cstring>
#include "MEM_guardedalloc.h"
#include "DNA_curve_types.h"
#include "DNA_scene_types.h"
#include "DNA_sequence_types.h"
#include "DNA_sound_types.h"
#include "BLI_listbase.h"
#include "BLI_utildefines.h"
#include "BKE_colortools.hh"
#include "BKE_sound.h"
#ifdef WITH_CONVOLUTION
# include "AUD_Sound.h"
#endif
#include "SEQ_sound.hh"
#include "SEQ_time.hh"
#include "sequencer.hh"
#include "strip_time.hh"
/* Unlike _update_sound_ functions,
* these ones take info from audaspace to update sequence length! */
const SoundModifierWorkerInfo workersSoundModifiers[] = {
{seqModifierType_SoundEqualizer, SEQ_sound_equalizermodifier_recreator}, {0, nullptr}};
#ifdef WITH_CONVOLUTION
static bool sequencer_refresh_sound_length_recursive(Main *bmain, Scene *scene, ListBase *seqbase)
{
bool changed = false;
LISTBASE_FOREACH (Strip *, strip, seqbase) {
if (strip->type == STRIP_TYPE_META) {
if (sequencer_refresh_sound_length_recursive(bmain, scene, &strip->seqbase)) {
changed = true;
}
}
else if (strip->type == STRIP_TYPE_SOUND_RAM && strip->sound) {
SoundInfo info;
if (!BKE_sound_info_get(bmain, strip->sound, &info)) {
continue;
}
int old = strip->len;
float fac;
strip->len = std::max(1, int(round((info.length - strip->sound->offset_time) * FPS)));
fac = float(strip->len) / float(old);
old = strip->startofs;
strip->startofs *= fac;
strip->endofs *= fac;
strip->start += (old -
strip->startofs); /* So that visual/"real" start frame does not change! */
changed = true;
}
}
return changed;
}
#endif
void SEQ_sound_update_length(Main *bmain, Scene *scene)
{
#ifdef WITH_CONVOLUTION
if (scene->ed) {
sequencer_refresh_sound_length_recursive(bmain, scene, &scene->ed->seqbase);
}
#else
UNUSED_VARS(bmain, scene);
#endif
}
void SEQ_sound_update_bounds_all(Scene *scene)
{
Editing *ed = scene->ed;
if (ed) {
LISTBASE_FOREACH (Strip *, strip, &ed->seqbase) {
if (strip->type == STRIP_TYPE_META) {
strip_update_sound_bounds_recursive(scene, strip);
}
else if (ELEM(strip->type, STRIP_TYPE_SOUND_RAM, STRIP_TYPE_SCENE)) {
SEQ_sound_update_bounds(scene, strip);
}
}
}
}
void SEQ_sound_update_bounds(Scene *scene, Strip *strip)
{
if (strip->type == STRIP_TYPE_SCENE) {
if (strip->scene && strip->scene_sound) {
/* We have to take into account start frame of the sequence's scene! */
int startofs = strip->startofs + strip->anim_startofs + strip->scene->r.sfra;
BKE_sound_move_scene_sound(scene,
strip->scene_sound,
SEQ_time_left_handle_frame_get(scene, strip),
SEQ_time_right_handle_frame_get(scene, strip),
startofs,
0.0);
}
}
else {
BKE_sound_move_scene_sound_defaults(scene, strip);
}
/* mute is set in strip_update_muting_recursive */
}
static void strip_update_sound_recursive(Scene *scene, ListBase *seqbasep, bSound *sound)
{
LISTBASE_FOREACH (Strip *, strip, seqbasep) {
if (strip->type == STRIP_TYPE_META) {
strip_update_sound_recursive(scene, &strip->seqbase, sound);
}
else if (strip->type == STRIP_TYPE_SOUND_RAM) {
if (strip->scene_sound && sound == strip->sound) {
BKE_sound_update_scene_sound(strip->scene_sound, sound);
}
}
}
}
void SEQ_sound_update(Scene *scene, bSound *sound)
{
if (scene->ed) {
strip_update_sound_recursive(scene, &scene->ed->seqbase, sound);
}
}
float SEQ_sound_pitch_get(const Scene *scene, const Strip *strip)
{
const Strip *meta_parent = SEQ_lookup_meta_by_strip(scene, strip);
if (meta_parent != nullptr) {
return strip->speed_factor * SEQ_sound_pitch_get(scene, meta_parent);
}
return strip->speed_factor;
}
EQCurveMappingData *SEQ_sound_equalizer_add(SoundEqualizerModifierData *semd,
float minX,
float maxX)
{
EQCurveMappingData *eqcmd;
if (maxX < 0) {
maxX = SOUND_EQUALIZER_DEFAULT_MAX_FREQ;
}
if (minX < 0) {
minX = 0.0;
}
/* It's the same as #BKE_curvemapping_add, but changing the name. */
eqcmd = MEM_cnew<EQCurveMappingData>("Equalizer");
BKE_curvemapping_set_defaults(&eqcmd->curve_mapping,
1, /* Total. */
minX,
-SOUND_EQUALIZER_DEFAULT_MAX_DB, /* Min x, y */
maxX,
SOUND_EQUALIZER_DEFAULT_MAX_DB, /* Max x, y */
HD_AUTO_ANIM);
eqcmd->curve_mapping.preset = CURVE_PRESET_CONSTANT_MEDIAN;
rctf clipr;
clipr.xmin = minX;
clipr.xmax = maxX;
clipr.ymin = 0.0;
clipr.ymax = 0.0;
BKE_curvemap_reset(&eqcmd->curve_mapping.cm[0], &clipr, CURVE_PRESET_CONSTANT_MEDIAN, 0);
BLI_addtail(&semd->graphics, eqcmd);
return eqcmd;
}
void SEQ_sound_equalizermodifier_set_graphs(SoundEqualizerModifierData *semd, int number)
{
SEQ_sound_equalizermodifier_free((SequenceModifierData *)semd);
if (number == 1) {
SEQ_sound_equalizer_add(
semd, SOUND_EQUALIZER_DEFAULT_MIN_FREQ, SOUND_EQUALIZER_DEFAULT_MAX_FREQ);
}
else if (number == 2) {
SEQ_sound_equalizer_add(semd, 30.0, 2000.0);
SEQ_sound_equalizer_add(semd, 2000.1, 20000.0);
}
else if (number == 3) {
SEQ_sound_equalizer_add(semd, 30.0, 1000.0);
SEQ_sound_equalizer_add(semd, 1000.1, 5000.0);
SEQ_sound_equalizer_add(semd, 5000.1, 20000.0);
}
}
EQCurveMappingData *SEQ_sound_equalizermodifier_add_graph(SoundEqualizerModifierData *semd,
float min_freq,
float max_freq)
{
if (min_freq < 0.0) {
return nullptr;
}
if (max_freq < 0.0) {
return nullptr;
}
if (max_freq <= min_freq) {
return nullptr;
}
return SEQ_sound_equalizer_add(semd, min_freq, max_freq);
}
void SEQ_sound_equalizermodifier_remove_graph(SoundEqualizerModifierData *semd,
EQCurveMappingData *eqcmd)
{
BLI_remlink_safe(&semd->graphics, eqcmd);
MEM_freeN(eqcmd);
}
void SEQ_sound_equalizermodifier_init_data(SequenceModifierData *smd)
{
SoundEqualizerModifierData *semd = (SoundEqualizerModifierData *)smd;
SEQ_sound_equalizer_add(
semd, SOUND_EQUALIZER_DEFAULT_MIN_FREQ, SOUND_EQUALIZER_DEFAULT_MAX_FREQ);
}
void SEQ_sound_equalizermodifier_free(SequenceModifierData *smd)
{
SoundEqualizerModifierData *semd = (SoundEqualizerModifierData *)smd;
LISTBASE_FOREACH_MUTABLE (EQCurveMappingData *, eqcmd, &semd->graphics) {
BKE_curvemapping_free_data(&eqcmd->curve_mapping);
MEM_freeN(eqcmd);
}
BLI_listbase_clear(&semd->graphics);
}
void SEQ_sound_equalizermodifier_copy_data(SequenceModifierData *target, SequenceModifierData *smd)
{
SoundEqualizerModifierData *semd = (SoundEqualizerModifierData *)smd;
SoundEqualizerModifierData *semd_target = (SoundEqualizerModifierData *)target;
EQCurveMappingData *eqcmd_n;
BLI_listbase_clear(&semd_target->graphics);
LISTBASE_FOREACH (EQCurveMappingData *, eqcmd, &semd->graphics) {
eqcmd_n = static_cast<EQCurveMappingData *>(MEM_dupallocN(eqcmd));
BKE_curvemapping_copy_data(&eqcmd_n->curve_mapping, &eqcmd->curve_mapping);
eqcmd_n->next = eqcmd_n->prev = nullptr;
BLI_addtail(&semd_target->graphics, eqcmd_n);
}
}
void *SEQ_sound_equalizermodifier_recreator(Strip *strip, SequenceModifierData *smd, void *sound)
{
#ifdef WITH_CONVOLUTION
UNUSED_VARS(strip);
SoundEqualizerModifierData *semd = (SoundEqualizerModifierData *)smd;
/* No equalizer definition. */
if (BLI_listbase_is_empty(&semd->graphics)) {
return sound;
}
float *buf = (float *)MEM_callocN(sizeof(float) * SOUND_EQUALIZER_SIZE_DEFINITION,
"eqrecreator");
CurveMapping *eq_mapping;
CurveMap *cm;
float minX;
float maxX;
float interval = SOUND_EQUALIZER_DEFAULT_MAX_FREQ / float(SOUND_EQUALIZER_SIZE_DEFINITION);
/* Visit all equalizer definitions. */
LISTBASE_FOREACH (EQCurveMappingData *, mapping, &semd->graphics) {
eq_mapping = &mapping->curve_mapping;
BKE_curvemapping_init(eq_mapping);
cm = eq_mapping->cm;
minX = eq_mapping->curr.xmin;
maxX = eq_mapping->curr.xmax;
int idx = int(ceil(minX / interval));
int i = idx;
for (; i * interval <= maxX && i < SOUND_EQUALIZER_SIZE_DEFINITION; i++) {
float freq = i * interval;
float val = BKE_curvemap_evaluateF(eq_mapping, cm, freq);
if (fabs(val) > SOUND_EQUALIZER_DEFAULT_MAX_DB)
val = (val / fabs(val)) * SOUND_EQUALIZER_DEFAULT_MAX_DB;
buf[i] = val;
/* To soften lower limit, but not the first position which is the constant value */
if (i == idx && i > 2) {
buf[i - 1] = 0.5 * (buf[i] + buf[i - 1]);
}
}
/* To soften higher limit */
if (i < SOUND_EQUALIZER_SIZE_DEFINITION)
buf[i] = 0.5 * (buf[i] + buf[i - 1]);
}
AUD_Sound *equ = AUD_Sound_equalize(sound,
buf,
SOUND_EQUALIZER_SIZE_DEFINITION,
SOUND_EQUALIZER_DEFAULT_MAX_FREQ,
SOUND_EQUALIZER_SIZE_CONVERSION);
MEM_freeN(buf);
return equ;
#else
UNUSED_VARS(strip, smd, sound);
return nullptr;
#endif
}
const SoundModifierWorkerInfo *SEQ_sound_modifier_worker_info_get(int type)
{
for (int i = 0; workersSoundModifiers[i].type > 0; i++) {
if (workersSoundModifiers[i].type == type) {
return &workersSoundModifiers[i];
}
}
return nullptr;
}
void *SEQ_sound_modifier_recreator(Strip *strip, SequenceModifierData *smd, void *sound)
{
if (!(smd->flag & SEQUENCE_MODIFIER_MUTE)) {
const SoundModifierWorkerInfo *smwi = SEQ_sound_modifier_worker_info_get(smd->type);
return smwi->recreator(strip, smd, sound);
}
return sound;
}