2024-09-05 10:41:34 +02:00
|
|
|
/* SPDX-FileCopyrightText: 2024 Blender Authors
|
2023-05-31 16:19:06 +02:00
|
|
|
*
|
|
|
|
|
* SPDX-License-Identifier: GPL-2.0-or-later */
|
2021-03-05 16:35:08 +01:00
|
|
|
|
|
|
|
|
/** \file
|
2024-09-05 10:41:34 +02:00
|
|
|
* \ingroup animrig
|
2021-03-05 16:35:08 +01:00
|
|
|
*/
|
|
|
|
|
|
2024-09-05 10:41:34 +02:00
|
|
|
#include "ANIM_pose.hh"
|
2021-07-20 16:13:14 +02:00
|
|
|
#include "BKE_action.hh"
|
2021-03-05 16:35:08 +01:00
|
|
|
#include "BKE_animsys.h"
|
2021-07-20 16:13:14 +02:00
|
|
|
#include "BKE_armature.hh"
|
2024-09-05 10:41:34 +02:00
|
|
|
#include "BLI_listbase.h"
|
2021-03-05 16:35:08 +01:00
|
|
|
#include "DNA_anim_types.h"
|
|
|
|
|
#include "DNA_object_types.h"
|
2023-08-10 22:40:27 +02:00
|
|
|
#include "RNA_access.hh"
|
2021-03-05 16:35:08 +01:00
|
|
|
|
2024-09-13 11:37:44 +02:00
|
|
|
#include "ANIM_action.hh"
|
|
|
|
|
|
|
|
|
|
namespace blender::animrig {
|
|
|
|
|
|
2021-03-05 16:35:08 +01:00
|
|
|
namespace {
|
|
|
|
|
|
2024-09-13 11:37:44 +02:00
|
|
|
using ActionApplier = blender::FunctionRef<void(
|
|
|
|
|
PointerRNA *, bAction *, slot_handle_t, const AnimationEvalContext *)>;
|
2021-07-12 11:28:15 +02:00
|
|
|
|
2024-09-05 10:41:34 +02:00
|
|
|
void pose_apply_restore_fcurves(bAction *action)
|
2021-07-12 11:28:15 +02:00
|
|
|
{
|
2024-09-05 10:41:34 +02:00
|
|
|
/* TODO(Sybren): Restore the FCurve flags, instead of just erasing the 'disabled' flag. */
|
|
|
|
|
LISTBASE_FOREACH (FCurve *, fcu, &action->curves) {
|
|
|
|
|
fcu->flag &= ~FCURVE_DISABLED;
|
|
|
|
|
}
|
2021-07-12 11:28:15 +02:00
|
|
|
}
|
|
|
|
|
|
2024-09-05 10:41:34 +02:00
|
|
|
void pose_apply_disable_fcurves_for_unselected_bones(
|
|
|
|
|
bAction *action, const blender::bke::BoneNameSet &selected_bone_names)
|
2021-07-02 13:47:53 +02:00
|
|
|
{
|
2024-09-05 10:41:34 +02:00
|
|
|
auto disable_unselected_fcurve = [&](FCurve *fcu, const char *bone_name) {
|
|
|
|
|
const bool is_bone_selected = selected_bone_names.contains(bone_name);
|
|
|
|
|
if (!is_bone_selected) {
|
|
|
|
|
fcu->flag |= FCURVE_DISABLED;
|
|
|
|
|
}
|
2021-07-02 13:47:53 +02:00
|
|
|
};
|
2024-09-05 10:41:34 +02:00
|
|
|
blender::bke::BKE_action_find_fcurves_with_bones(action, disable_unselected_fcurve);
|
2021-07-02 13:47:53 +02:00
|
|
|
}
|
|
|
|
|
|
2023-06-03 08:36:28 +10:00
|
|
|
void pose_apply(Object *ob,
|
|
|
|
|
bAction *action,
|
2024-09-13 11:37:44 +02:00
|
|
|
const slot_handle_t slot_handle,
|
2024-04-04 10:55:10 +11:00
|
|
|
const AnimationEvalContext *anim_eval_context,
|
2021-07-12 11:28:15 +02:00
|
|
|
ActionApplier applier)
|
2021-03-05 16:35:08 +01:00
|
|
|
{
|
|
|
|
|
bPose *pose = ob->pose;
|
2021-03-06 16:54:09 +01:00
|
|
|
if (pose == nullptr) {
|
2021-03-05 16:35:08 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-23 14:54:41 +01:00
|
|
|
const bArmature *armature = (bArmature *)ob->data;
|
2024-09-05 10:41:34 +02:00
|
|
|
const blender::bke::BoneNameSet selected_bone_names =
|
|
|
|
|
blender::bke::BKE_armature_find_selected_bone_names(armature);
|
2021-03-05 16:35:08 +01:00
|
|
|
const bool limit_to_selected_bones = !selected_bone_names.is_empty();
|
|
|
|
|
|
|
|
|
|
if (limit_to_selected_bones) {
|
|
|
|
|
/* Mute all FCurves that are not associated with selected bones. This separates the concept of
|
|
|
|
|
* bone selection from the FCurve evaluation code. */
|
|
|
|
|
pose_apply_disable_fcurves_for_unselected_bones(action, selected_bone_names);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Apply the Action. */
|
2023-09-06 00:48:50 +02:00
|
|
|
PointerRNA pose_owner_ptr = RNA_id_pointer_create(&ob->id);
|
2021-07-12 11:28:15 +02:00
|
|
|
|
2024-09-13 11:37:44 +02:00
|
|
|
applier(&pose_owner_ptr, action, slot_handle, anim_eval_context);
|
2021-03-05 16:35:08 +01:00
|
|
|
|
|
|
|
|
if (limit_to_selected_bones) {
|
|
|
|
|
pose_apply_restore_fcurves(action);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-05 10:41:34 +02:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
void pose_apply_action_selected_bones(Object *ob,
|
|
|
|
|
bAction *action,
|
2024-09-13 11:37:44 +02:00
|
|
|
const int32_t slot_handle,
|
2024-09-05 10:41:34 +02:00
|
|
|
const AnimationEvalContext *anim_eval_context)
|
2021-03-05 16:35:08 +01:00
|
|
|
{
|
2024-09-13 11:37:44 +02:00
|
|
|
auto evaluate_and_apply = [](PointerRNA *ptr,
|
|
|
|
|
bAction *act,
|
|
|
|
|
const int32_t slot_handle,
|
|
|
|
|
const AnimationEvalContext *anim_eval_context) {
|
|
|
|
|
animsys_evaluate_action(ptr, act, slot_handle, anim_eval_context, false);
|
|
|
|
|
};
|
2024-09-05 10:41:34 +02:00
|
|
|
|
2024-09-13 11:37:44 +02:00
|
|
|
pose_apply(ob, action, slot_handle, anim_eval_context, evaluate_and_apply);
|
2021-03-05 16:35:08 +01:00
|
|
|
}
|
|
|
|
|
|
2024-09-05 10:41:34 +02:00
|
|
|
void pose_apply_action_all_bones(Object *ob,
|
|
|
|
|
bAction *action,
|
2024-09-13 11:37:44 +02:00
|
|
|
const int32_t slot_handle,
|
2024-09-05 10:41:34 +02:00
|
|
|
const AnimationEvalContext *anim_eval_context)
|
2021-03-05 16:35:08 +01:00
|
|
|
{
|
2024-09-05 10:41:34 +02:00
|
|
|
PointerRNA pose_owner_ptr = RNA_id_pointer_create(&ob->id);
|
2024-09-13 11:37:44 +02:00
|
|
|
animsys_evaluate_action(&pose_owner_ptr, action, slot_handle, anim_eval_context, false);
|
2024-09-05 10:41:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pose_apply_action_blend(Object *ob,
|
|
|
|
|
bAction *action,
|
2024-09-13 11:37:44 +02:00
|
|
|
const int32_t slot_handle,
|
2024-09-05 10:41:34 +02:00
|
|
|
const AnimationEvalContext *anim_eval_context,
|
|
|
|
|
const float blend_factor)
|
|
|
|
|
{
|
|
|
|
|
auto evaluate_and_blend = [blend_factor](PointerRNA *ptr,
|
|
|
|
|
bAction *act,
|
2024-09-13 11:37:44 +02:00
|
|
|
const int32_t slot_handle,
|
2024-09-05 10:41:34 +02:00
|
|
|
const AnimationEvalContext *anim_eval_context) {
|
2024-09-13 11:37:44 +02:00
|
|
|
animsys_blend_in_action(ptr, act, slot_handle, anim_eval_context, blend_factor);
|
2021-07-20 16:13:14 +02:00
|
|
|
};
|
2024-09-05 10:41:34 +02:00
|
|
|
|
2024-09-13 11:37:44 +02:00
|
|
|
pose_apply(ob, action, slot_handle, anim_eval_context, evaluate_and_blend);
|
2021-03-05 16:35:08 +01:00
|
|
|
}
|
|
|
|
|
|
2024-09-05 10:41:34 +02:00
|
|
|
} // namespace blender::animrig
|