Merge branch 'blender-v4.0-release'
[blender-addons.git] / power_sequencer / operators / expand_to_surrounding_cuts.py
blobad31ab2730512a225c194b6be0b0de9b126b8134
1 # SPDX-FileCopyrightText: 2016-2020 by Nathan Lovato, Daniel Oakey, Razvan Radulescu, and contributors
3 # SPDX-License-Identifier: GPL-3.0-or-later
5 import bpy
7 from .utils.functions import slice_selection
8 from .utils.doc import doc_name, doc_idname, doc_brief, doc_description
11 class POWER_SEQUENCER_OT_expand_to_surrounding_cuts(bpy.types.Operator):
12 """
13 *Brief* Expand selected strips to surrounding cuts
15 Finds potential gaps surrounding each block of selected sequences and extends the corresponding
16 sequence handle to it
17 """
19 doc = {
20 "name": doc_name(__qualname__),
21 "demo": "",
22 "description": doc_description(__doc__),
23 "shortcuts": [
25 {"type": "E", "value": "PRESS", "ctrl": True},
26 {},
27 "Expand to Surrounding Cuts",
30 "keymap": "Sequencer",
32 bl_idname = doc_idname(__qualname__)
33 bl_label = doc["name"]
34 bl_description = doc_brief(doc["description"])
35 bl_options = {"REGISTER", "UNDO"}
37 margin: bpy.props.FloatProperty(
38 name="Trim margin",
39 description="Margin to leave on either sides of the trim in seconds",
40 default=0.2,
41 min=0,
43 gap_remove: bpy.props.BoolProperty(
44 name="Remove gaps",
45 description="When trimming the sequences, remove gaps automatically",
46 default=True,
49 @classmethod
50 def poll(cls, context):
51 return context.selected_sequences
53 def invoke(self, context, event):
54 sequence_blocks = slice_selection(context, context.selected_sequences)
55 for sequences in sequence_blocks:
56 sequences_frame_start = min(
57 sequences, key=lambda s: s.frame_final_start
58 ).frame_final_start
59 sequences_frame_end = max(sequences, key=lambda s: s.frame_final_end).frame_final_end
61 frame_left, frame_right = find_closest_cuts(
62 context, sequences_frame_start, sequences_frame_end
64 if sequences_frame_start == frame_left and sequences_frame_end == frame_right:
65 continue
67 to_extend_left = [s for s in sequences if s.frame_final_start == sequences_frame_start]
68 to_extend_right = [s for s in sequences if s.frame_final_end == sequences_frame_end]
70 for s in to_extend_left:
71 s.frame_final_start = (
72 frame_left if frame_left < sequences_frame_start else sequences_frame_start
74 for s in to_extend_right:
75 s.frame_final_end = (
76 frame_right if frame_right > sequences_frame_end else sequences_frame_end
78 return {"FINISHED"}
81 def find_closest_cuts(context, frame_min, frame_max):
82 frame_left = max(
83 context.sequences,
84 key=lambda s: s.frame_final_end if s.frame_final_end <= frame_min else -1,
85 ).frame_final_end
86 frame_right = min(
87 context.sequences,
88 key=lambda s: s.frame_final_start if s.frame_final_start >= frame_max else 1000000,
89 ).frame_final_start
90 return frame_left, frame_right