1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
use crate::*;
use crate::aegp::*;
use ae_sys::{ AEGP_EffectRefH, AEGP_LayerH };
define_suite!(
/// Access the effects applied to a layer. This suite provides access to all parameter data streams.
///
/// Use the [`suites::Stream`](aegp::suites::Stream) to work with those streams.
///
/// An [`EffectRefHandle`] is a reference to an applied effect. An [`InstalledEffectKey`] is a reference to an installed effect, which may or may not be currently applied to anything.
///
/// If Foobarocity is applied to a layer twice, there will be two distinct [`EffectRefHandle`]s, but they'll both return the same [`InstalledEffectKey`].
EffectSuite,
AEGP_EffectSuite4,
kAEGPEffectSuite,
kAEGPEffectSuiteVersion4
);
define_suite!(
EffectSuite5,
AEGP_EffectSuite5,
kAEGPEffectSuite,
kAEGPEffectSuiteVersion5
);
impl EffectSuite {
/// Acquire this suite from the host. Returns error if the suite is not available.
/// Suite is released on drop.
pub fn new() -> Result<Self, Error> {
crate::Suite::new()
}
/// Get the number of effects applied to a layer.
pub fn layer_num_effects(&self, layer: impl AsPtr<AEGP_LayerH>) -> Result<i32, Error> {
Ok(call_suite_fn_single!(self, AEGP_GetLayerNumEffects -> ae_sys::A_long, layer.as_ptr())? as i32)
}
/// Retrieves (by index) a reference to an effect applied to the layer.
pub fn layer_effect_by_index(&self, layer: impl AsPtr<AEGP_LayerH>, plugin_id: PluginId, index: i32) -> Result<EffectRefHandle, Error> {
Ok(EffectRefHandle::from_raw(
call_suite_fn_single!(self, AEGP_GetLayerEffectByIndex -> ae_sys::AEGP_EffectRefH, plugin_id, layer.as_ptr(), index)?
))
}
/// Given an [`EffectRefHandle`], retrieves its associated [`InstalledEffectKey`].
pub fn installed_key_from_layer_effect(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>) -> Result<InstalledEffectKey, Error> {
Ok(call_suite_fn_single!(self, AEGP_GetInstalledKeyFromLayerEffect -> ae_sys::AEGP_InstalledEffectKey, effect_ref.as_ptr())?.into())
}
/// Returns description of effect parameter.
///
/// Do not use the value(s) in the Param returned by this function (Use [`suites::Stream::new_stream_value()`](aegp::suites::Stream::new_stream_value) instead);
/// it's provided so AEGPs can access parameter defaults, checkbox names, and pop-up strings.
///
/// Use [`suites::Stream::effect_num_param_streams()`](aegp::suites::Stream::effect_num_param_streams) to get the stream count, useful for determining the maximum `param_index`.
pub fn effect_param_union_by_index(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, plugin_id: PluginId, param_index: i32) -> Result<pf::Param<'_>, Error> {
let (param_type, u) = call_suite_fn_double!(self, AEGP_GetEffectParamUnionByIndex -> ae_sys::PF_ParamType, ae_sys::PF_ParamDefUnion, plugin_id, effect_ref.as_ptr(), param_index)?;
unsafe {
match param_type {
ae_sys::PF_Param_ANGLE => Ok(Param::Angle (AngleDef ::from_owned(u.ad))),
ae_sys::PF_Param_ARBITRARY_DATA => Ok(Param::Arbitrary (ArbitraryDef ::from_owned(u.arb_d))),
ae_sys::PF_Param_BUTTON => Ok(Param::Button (ButtonDef ::from_owned(u.button_d))),
ae_sys::PF_Param_CHECKBOX => Ok(Param::CheckBox (CheckBoxDef ::from_owned(u.bd))),
ae_sys::PF_Param_COLOR => Ok(Param::Color (ColorDef ::from_owned(u.cd))),
ae_sys::PF_Param_FLOAT_SLIDER => Ok(Param::FloatSlider(FloatSliderDef::from_owned(u.fs_d))),
ae_sys::PF_Param_POPUP => Ok(Param::Popup (PopupDef ::from_owned(u.pd))),
ae_sys::PF_Param_SLIDER => Ok(Param::Slider (SliderDef ::from_owned(u.sd))),
ae_sys::PF_Param_POINT => Ok(Param::Point (PointDef ::from_owned(u.td))),
ae_sys::PF_Param_POINT_3D => Ok(Param::Point3D (Point3DDef ::from_owned(u.point3d_d))),
ae_sys::PF_Param_PATH => Ok(Param::Path (PathDef ::from_owned(u.path_d))),
_ => Err(Error::InvalidParms),
}
}
}
/// Obtains the flags for the given [`EffectRefHandle`].
pub fn effect_flags(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>) -> Result<EffectFlags, Error> {
Ok(EffectFlags::from_bits_truncate(call_suite_fn_single!(self, AEGP_GetEffectFlags -> ae_sys::AEGP_EffectFlags, effect_ref.as_ptr())?))
}
/// Sets the flags for the given [`EffectRefHandle`], masked by a different set of effect flags.
pub fn set_effect_flags(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, set_mask: EffectFlags, flags: EffectFlags) -> Result<(), Error> {
call_suite_fn!(self, AEGP_SetEffectFlags, effect_ref.as_ptr(), set_mask.bits(), flags.bits())
}
/// Change the order of applied effects (pass the requested index).
pub fn reorder_effect(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, index: i32) -> Result<(), Error> {
call_suite_fn!(self, AEGP_ReorderEffect, effect_ref.as_ptr(), index)
}
/// Call an effect plug-in, and pass it a pointer to any data you like; the effect can modify it.
///
/// This is how AEGPs communicate with effects.
///
/// Pass [`Command::CompletelyGeneral`](crate::Command::CompletelyGeneral) for `command` to get the old behaviour.
pub fn effect_call_generic<T: Sized>(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, plugin_id: PluginId, time: Time, command: &pf::Command, extra_payload: Option<&T>) -> Result<(), Error> {
// T is Sized so it can never be a fat pointer which means we are safe to transmute here.
// Alternatively we could write extra_payload.map(|p| p as *const _).unwrap_or(core::ptr::null())
call_suite_fn!(self, AEGP_EffectCallGeneric, plugin_id, effect_ref.as_ptr(), &time.into() as *const _, command.as_raw(), std::mem::transmute(extra_payload))
}
/// Disposes of an [`EffectRefHandle`]. Use this to dispose of any [`EffectRefHandle`] returned by After Effects.
pub fn dispose_effect(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>) -> Result<(), Error> {
call_suite_fn!(self, AEGP_DisposeEffect, effect_ref.as_ptr())
}
/// Apply an effect to a given layer. Returns the newly-created [`EffectRefHandle`].
pub fn apply_effect(&self, layer: impl AsPtr<AEGP_LayerH>, plugin_id: PluginId, installed_effect_key: InstalledEffectKey) -> Result<EffectRefHandle, Error> {
Ok(EffectRefHandle::from_raw(
call_suite_fn_single!(self, AEGP_ApplyEffect -> ae_sys::AEGP_EffectRefH, plugin_id, layer.as_ptr(), installed_effect_key.into())?
))
}
/// Remove an applied effect.
pub fn delete_layer_effect(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>) -> Result<(), Error> {
call_suite_fn!(self, AEGP_DeleteLayerEffect, effect_ref.as_ptr())
}
/// Returns the count of effects installed in After Effects.
pub fn num_installed_effects(&self) -> Result<i32, Error> {
Ok(call_suite_fn_single!(self, AEGP_GetNumInstalledEffects -> ae_sys::A_long)? as i32)
}
/// Returns the [`InstalledEffectKey`] of the next installed effect.
///
/// Pass [`InstalledEffectKey::None`] as the first parameter to obtain the first [`InstalledEffectKey`].
pub fn next_installed_effect(&self, installed_effect_key: InstalledEffectKey) -> Result<InstalledEffectKey, Error> {
Ok(call_suite_fn_single!(self, AEGP_GetNextInstalledEffect -> ae_sys::AEGP_InstalledEffectKey, installed_effect_key.into())?.into())
}
/// Get name of the effect. `name` can be up to `48` characters long.
///
/// Note: use [`suites::DynamicStream::set_stream_name()`](aegp::suites::DynamicStream::set_stream_name) to change the display name of an effect.
pub fn effect_name(&self, installed_effect_key: InstalledEffectKey) -> Result<String, Error> {
let mut name = [0i8; ae_sys::AEGP_MAX_EFFECT_NAME_SIZE as usize + 1];
call_suite_fn!(self, AEGP_GetEffectName, installed_effect_key.into(), name.as_mut_ptr() as _)?;
Ok(unsafe { std::ffi::CStr::from_ptr(name.as_ptr()) }.to_string_lossy().into_owned())
}
/// Get match name of an effect (defined in PiPL). `match_name` up to `48` characters long.
///
/// Match names are in 7-bit ASCII. UI names are in the current application runtime encoding;
/// for example, ISO 8859-1 for most languages on Windows.
pub fn effect_match_name(&self, installed_effect_key: InstalledEffectKey) -> Result<String, Error> {
let mut name = [0i8; ae_sys::AEGP_MAX_EFFECT_MATCH_NAME_SIZE as usize + 1];
call_suite_fn!(self, AEGP_GetEffectMatchName, installed_effect_key.into(), name.as_mut_ptr() as _)?;
// TODO: It's not UTF-8
Ok(unsafe { std::ffi::CStr::from_ptr(name.as_ptr()) }.to_string_lossy().into_owned())
}
/// Menu category of effect. `category` can be up to `48` characters long.
pub fn effect_category(&self, installed_effect_key: InstalledEffectKey) -> Result<String, Error> {
let mut name = [0i8; ae_sys::AEGP_MAX_EFFECT_CATEGORY_NAME_SIZE as usize + 1];
call_suite_fn!(self, AEGP_GetEffectCategory, installed_effect_key.into(), name.as_mut_ptr() as _)?;
Ok(unsafe { std::ffi::CStr::from_ptr(name.as_ptr()) }.to_string_lossy().into_owned())
}
/// Duplicates a given [`EffectRefHandle`]. Caller must dispose of duplicate when finished.
pub fn duplicate_effect(&self, original_effect_ref: impl AsPtr<AEGP_EffectRefH>) -> Result<EffectRefHandle, Error> {
Ok(EffectRefHandle::from_raw(
call_suite_fn_single!(self, AEGP_DuplicateEffect -> ae_sys::AEGP_EffectRefH, original_effect_ref.as_ptr())?
))
}
/// New in CC 2014. How many masks are on this effect?
pub fn num_effect_mask(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>) -> Result<usize, Error> {
Ok(call_suite_fn_single!(self, AEGP_NumEffectMask -> ae_sys::A_u_long, effect_ref.as_ptr())? as usize)
}
/// New in CC 2014. For a given `mask_index`, returns the corresponding `AEGP_MaskIDVal` for use in uniquely identifying the mask.
pub fn effect_mask_id(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, mask_index: usize) -> Result<ae_sys::AEGP_MaskIDVal, Error> {
call_suite_fn_single!(self, AEGP_GetEffectMaskID -> ae_sys::AEGP_MaskIDVal, effect_ref.as_ptr(), mask_index as ae_sys::A_u_long)
}
/// New in CC 2014. Add an effect mask, which may be created using the [`suites::Mask`](aegp::suites::Mask).
///
/// Returns the local stream of the effect ref - useful if you want to add keyframes.
///
/// Caller must dispose of [`StreamReferenceHandle`] when finished.
///
/// Undoable.
pub fn add_effect_mask(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, id_val: ae_sys::AEGP_MaskIDVal) -> Result<StreamReferenceHandle, Error> {
Ok(StreamReferenceHandle::from_raw(
call_suite_fn_single!(self, AEGP_AddEffectMask -> ae_sys::AEGP_StreamRefH, effect_ref.as_ptr(), id_val)?
))
}
/// New in CC 2014. Remove an effect mask.
///
/// Undoable.
pub fn remove_effect_mask(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, id_val: ae_sys::AEGP_MaskIDVal) -> Result<(), Error> {
call_suite_fn!(self, AEGP_RemoveEffectMask, effect_ref.as_ptr(), id_val)
}
/// New in CC 2014. Set an effect mask on an existing index.
///
/// Returns the local stream of the effect ref - useful if you want to add keyframes.
///
/// Caller must dispose of [`StreamReferenceHandle`] when finished.
///
/// Undoable.
pub fn set_effect_mask(&self, effect_ref: impl AsPtr<AEGP_EffectRefH>, mask_index: usize, id_val: ae_sys::AEGP_MaskIDVal) -> Result<StreamReferenceHandle, Error> {
Ok(StreamReferenceHandle::from_raw(
call_suite_fn_single!(self, AEGP_SetEffectMask -> ae_sys::AEGP_StreamRefH, effect_ref.as_ptr(), mask_index as ae_sys::A_u_long, id_val)?
))
}
pub fn is_internal_effect(&self, installed_effect_key: ae_sys::AEGP_InstalledEffectKey) -> Result<bool, Error> {
let v5 = EffectSuite5::new()?;
Ok(call_suite_fn_single!(v5, AEGP_GetIsInternalEffect -> ae_sys::A_Boolean, installed_effect_key)? != 0)
}
}
// ――――――――――――――――――――――――――――――――――――――― Types ――――――――――――――――――――――――――――――――――――――――
register_handle!(AEGP_EffectRefH);
define_handle_wrapper!(EffectRefHandle, AEGP_EffectRefH);
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
pub enum InstalledEffectKey {
None,
Key(i32)
}
impl From<ae_sys::AEGP_InstalledEffectKey> for InstalledEffectKey {
fn from(key: ae_sys::AEGP_InstalledEffectKey) -> Self {
if key == ae_sys::AEGP_InstalledEffectKey_NONE as ae_sys::AEGP_InstalledEffectKey {
InstalledEffectKey::None
} else {
InstalledEffectKey::Key(key as _)
}
}
}
impl Into<ae_sys::AEGP_InstalledEffectKey> for InstalledEffectKey {
fn into(self) -> ae_sys::AEGP_InstalledEffectKey {
match self {
InstalledEffectKey::None => ae_sys::AEGP_InstalledEffectKey_NONE as ae_sys::AEGP_InstalledEffectKey,
InstalledEffectKey::Key(key) => key as ae_sys::AEGP_InstalledEffectKey,
}
}
}
bitflags::bitflags! {
pub struct EffectFlags: ae_sys::A_long {
const NONE = ae_sys::AEGP_EffectFlags_NONE as ae_sys::A_long;
const ACTIVE = ae_sys::AEGP_EffectFlags_ACTIVE as ae_sys::A_long;
const AUDIO_ONLY = ae_sys::AEGP_EffectFlags_AUDIO_ONLY as ae_sys::A_long;
const AUDIO_TOO = ae_sys::AEGP_EffectFlags_AUDIO_TOO as ae_sys::A_long;
const MISSING = ae_sys::AEGP_EffectFlags_MISSING as ae_sys::A_long;
}
}
define_suite_item_wrapper!(
ae_sys::AEGP_EffectRefH, EffectRefHandle,
suite: EffectSuite,
stream: aegp::suites::Stream,
/// Access the effects applied to a layer. This suite provides access to all parameter data streams.
///
/// Use the [`suites::Stream`](aegp::suites::Stream) to work with those streams.
///
/// An [`EffectRefHandle`] is a reference to an applied effect. An [`InstalledEffectKey`] is a reference to an installed effect, which may or may not be currently applied to anything.
///
/// If Foobarocity is applied to a layer twice, there will be two distinct [`EffectRefHandle`]s, but they'll both return the same [`InstalledEffectKey`].
Effect {
dispose: suite.dispose_effect;
/// Retrieves its associated [`InstalledEffectKey`].
installed_key() -> InstalledEffectKey => suite.installed_key_from_layer_effect,
/// Returns description of effect parameter.
///
/// Do not use the value(s) in the Param returned by this function (Use [`suites::Stream::new_stream_value()`](aegp::suites::Stream::new_stream_value) instead);
/// it's provided so AEGPs can access parameter defaults, checkbox names, and pop-up strings.
///
/// Use [`suites::Stream::effect_num_param_streams()`](aegp::suites::Stream::effect_num_param_streams) to get the stream count, useful for determining the maximum `param_index`.
param_union_by_index(plugin_id: PluginId, param_index: i32) -> pf::Param<'_> => suite.effect_param_union_by_index,
/// Obtains the flags for this [`Effect`].
flags() -> EffectFlags => suite.effect_flags,
/// Sets the flags for this [`Effect`], masked by a different set of effect flags.
set_flags(set_mask: EffectFlags, flags: EffectFlags) -> () => suite.set_effect_flags,
/// Change the order of applied effects (pass the requested index).
reorder(index: i32) -> () => suite.reorder_effect,
/// Remove an applied effect.
delete_layert() -> () => suite.delete_layer_effect,
/// Duplicates this [`Effect`]. Caller must dispose of duplicate when finished.
duplicate() -> EffectRefHandle => suite.duplicate_effect,
/// New in CC 2014. How many masks are on this effect?
num_mask() -> usize => suite.num_effect_mask,
/// New in CC 2014. For a given `mask_index`, returns the corresponding `AEGP_MaskIDVal` for use in uniquely identifying the mask.
mask_id(mask_index: usize) -> ae_sys::AEGP_MaskIDVal => suite.effect_mask_id,
/// New in CC 2014. Add an effect mask, which may be created using the [`suites::Mask`](aegp::suites::Mask).
///
/// Returns the local stream of the effect ref - useful if you want to add keyframes.
///
/// Undoable.
add_mask(id_val: ae_sys::AEGP_MaskIDVal) -> Stream => suite.add_effect_mask,
/// New in CC 2014. Remove an effect mask.
///
/// Undoable.
remove_mask(id_val: ae_sys::AEGP_MaskIDVal) -> () => suite.remove_effect_mask,
/// New in CC 2014. Set an effect mask on an existing index.
///
/// Returns the local stream of the effect ref - useful if you want to add keyframes.
///
/// Undoable.
set_mask(mask_index: usize, id_val: ae_sys::AEGP_MaskIDVal) -> Stream => suite.set_effect_mask,
// ―――――――――――――――――――――――――――― Stream suite functions ――――――――――――――――――――――――――――
/// Get number of parameter streams associated with an effect.
num_param_streams() -> i32 => stream.effect_num_param_streams,
/// Get an effect's parameter stream.
new_stream_by_index(plugin_id: PluginId, index: i32) -> Stream => stream.new_effect_stream_by_index,
}
);
impl Effect {
/// Returns a new AEGP effect from the provided `effect_ref` and AEGP plugin ID.
pub fn new(effect_ref: impl AsPtr<ae_sys::PF_ProgPtr>, plugin_id: PluginId) -> Result<Self, Error> {
Ok(Self::from_handle(
aegp::suites::PFInterface::new()?.new_effect_for_effect(effect_ref, plugin_id)?,
true
))
}
/// Creates a new [`aegp::LayerRenderOptions`] from this layer.
pub fn layer_render_options(&self, plugin_id: PluginId) -> Result<aegp::LayerRenderOptions, Error> {
aegp::LayerRenderOptions::from_upstream_of_effect(self.handle.as_ptr(), plugin_id)
}
/// Returns the count of effects installed in After Effects.
pub fn num_installed_effects() -> Result<i32, Error> {
EffectSuite::new()?.num_installed_effects()
}
/// Returns the [`InstalledEffectKey`] of the next installed effect.
///
/// Pass [`InstalledEffectKey::None`] as the first parameter to obtain the first [`InstalledEffectKey`].
pub fn next_installed_effect(installed_effect_key: InstalledEffectKey) -> Result<InstalledEffectKey, Error> {
EffectSuite::new()?.next_installed_effect(installed_effect_key)
}
/// Get name of the effect. `name` can be up to `48` characters long.
///
/// Note: use [`suites::DynamicStream::set_stream_name()`](aegp::suites::DynamicStream::set_stream_name) to change the display name of an effect.
pub fn name_of(installed_effect_key: InstalledEffectKey) -> Result<String, Error> {
EffectSuite::new()?.effect_name(installed_effect_key)
}
/// Get match name of an effect (defined in PiPL). Can be up to `48` characters long.
///
/// Match names are in 7-bit ASCII. UI names are in the current application runtime encoding;
/// for example, ISO 8859-1 for most languages on Windows.
pub fn match_name_of(installed_effect_key: InstalledEffectKey) -> Result<String, Error> {
EffectSuite::new()?.effect_match_name(installed_effect_key)
}
/// Menu category of effect. Can be up to `48` characters long.
pub fn category_of(installed_effect_key: InstalledEffectKey) -> Result<String, Error> {
EffectSuite::new()?.effect_category(installed_effect_key)
}
}