68 lines
2.2 KiB
Plaintext
68 lines
2.2 KiB
Plaintext
doc = App.activeDocument()
|
|
|
|
# Get the base object (C2)
|
|
# Naming format: <note>-<object-type>
|
|
c2 = doc.getObjectsByLabel('C2-Base')[0]
|
|
|
|
# Create a list of notes from C2 to C7
|
|
notes = ["C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B"]
|
|
notes = [f"{k}{str(oct)}" for oct in range(2, 8) for k in notes]
|
|
notes = notes[notes.index("C2"):notes.index("C7") + 1]
|
|
non_sharp = [note for note in notes if "#" not in note]
|
|
|
|
MIN_WIDTH = 78
|
|
MAX_WIDTH = 194
|
|
|
|
|
|
# Calculate position offset for a note
|
|
def calc_pos_offset(note: str):
|
|
is_sharp = "#" in note
|
|
|
|
# Find the index of the note in the list of non-sharp notes
|
|
note_index = non_sharp.index(note.replace("#", ""))
|
|
|
|
# If it's a sharp note, place it in between the note and the previous note
|
|
if is_sharp:
|
|
note_index += 0.5
|
|
|
|
# Calculate the offset
|
|
return 37 * note_index
|
|
|
|
|
|
# Loop through C#2 to C7
|
|
for i, note in enumerate(notes):
|
|
if i == 0:
|
|
continue
|
|
|
|
is_sharp = "#" in note
|
|
|
|
# Create a copy of the base object
|
|
copy: list[fc.Part] = list(doc.copyObject(c2, True, True))
|
|
|
|
# Since freecad will automatically name the copy ORIGINAL001, we need to rename it
|
|
for obj in copy:
|
|
obj.Label = obj.Label.replace('C2', note).replace("001", "")
|
|
|
|
def find_ends_with(label: str) -> fc.Part:
|
|
return [obj for obj in copy if obj.Label.endswith(label)][0]
|
|
|
|
# Find the objects by label
|
|
base, cube, hole = [find_ends_with(label) for label in ["Base", "Cube", "Hole"]]
|
|
|
|
# Make the cube wider
|
|
# width = MIN_WIDTH + (MAX_WIDTH - MIN_WIDTH) * ((len(notes) - i) / len(notes))
|
|
x = len(notes) - i - 1
|
|
# Calculate width with a quadratic function 0.022172619047619x^2 + 0.60297619047619x + 78
|
|
width = 0.016592261904762 * x ** 2 + 0.93779761904762 * x + 78
|
|
cube.Width = fc.Units.Quantity(f"{width} mm")
|
|
|
|
# The hole's y placement should be width - width / 4.2
|
|
plac: list[float] = list(hole.Placement.Base)
|
|
plac[1] = width - width / 4.2
|
|
hole.Placement.Base = App.Vector(*plac)
|
|
|
|
# Move the base object. For sharp notes, move it to the top
|
|
y = MAX_WIDTH - 6 if is_sharp else MAX_WIDTH - width
|
|
base.Placement.move(App.Vector(calc_pos_offset(note), y, 0))
|
|
|
|
doc.recompute() |