WIP: track creation and deletion both in frontend and backend.

This commit is contained in:
2025-07-28 01:03:40 +02:00
parent ecadfb7033
commit cb38a509b5
15 changed files with 192 additions and 49 deletions

View File

@@ -83,12 +83,11 @@ expand_mode = 2
unique_name_in_owner = true unique_name_in_owner = true
custom_minimum_size = Vector2(256, 0) custom_minimum_size = Vector2(256, 0)
layout_mode = 2 layout_mode = 2
size_flags_horizontal = 0 size_flags_horizontal = 3
size_flags_vertical = 4 size_flags_vertical = 4
theme_override_fonts/font = SubResource("FontVariation_lbdn3") theme_override_fonts/font = SubResource("FontVariation_lbdn3")
placeholder_text = "Empty Track" placeholder_text = "Empty Track"
context_menu_enabled = false context_menu_enabled = false
flat = true
draw_control_chars = true draw_control_chars = true
caret_blink = true caret_blink = true
script = ExtResource("4_2nwak") script = ExtResource("4_2nwak")

View File

@@ -37,23 +37,6 @@ func format_time_ms_minutes(ms: float) -> String:
var milliseconds = fmod(ms, 1000.0) var milliseconds = fmod(ms, 1000.0)
return "%02d:%02d.%03d" % [minutes, seconds, milliseconds] return "%02d:%02d.%03d" % [minutes, seconds, milliseconds]
func _ready():
await get_tree().process_frame
queue_redraw()
for c in track_list.get_children():
if c is Track:
var track = c as Track
track.on_deleted.connect(_track_deleted)
pass
pass
func _track_deleted(idx: int):
print("track deleted")
await get_tree().process_frame
queue_redraw()
pass
func get_track_idx_by_y(y: float): func get_track_idx_by_y(y: float):
var idx = 0 var idx = 0
for track in track_list.get_children(): for track in track_list.get_children():
@@ -64,6 +47,8 @@ func get_track_idx_by_y(y: float):
return -1 return -1
func get_track_by_idx(idx: int) -> Control: func get_track_by_idx(idx: int) -> Control:
if idx > track_list.get_child_count() - 1:
return null
return track_list.get_child(idx) return track_list.get_child(idx)
func get_pixels_per_unit() -> float: func get_pixels_per_unit() -> float:

View File

@@ -1,15 +1,19 @@
class_name Track class_name Track
extends Button extends Button
signal on_deleted signal on_deleted(idx: int)
signal on_renamed signal on_renamed
signal on_duplicated signal on_duplicated
func _ready(): func _ready():
var menu: PopupMenu = $PopupMenu var menu: PopupMenu = $PopupMenu
menu.add_item("Rename", 0) menu.add_item("Rename", 0)
menu.add_item("Delete", 1)
menu.add_item("Duplicate", 2) menu.add_item("Duplicate", 2)
menu.add_item("Delete", 1)
menu.add_separator()
menu.add_item("Properties")
menu.id_pressed.connect(_on_menu_option) menu.id_pressed.connect(_on_menu_option)
func _on_menu_option(id: int): func _on_menu_option(id: int):
@@ -23,10 +27,7 @@ func _on_menu_option(id: int):
pass pass
func delete(): func delete():
queue_free() on_deleted.emit(get_index())
var idx = get_index()
print(idx)
on_deleted.emit(idx)
pass pass
func rename(): func rename():

View File

@@ -4,6 +4,11 @@ func _ready() -> void:
text_submitted.connect(submit) text_submitted.connect(submit)
func submit(text: String): func submit(text: String):
caret_column = 0
release_focus() release_focus()
var owner = get_owner()
if owner is Control:
var c = owner as Control
c.grab_focus()
pass pass

44
Scripts/TrackView.gd Normal file
View File

@@ -0,0 +1,44 @@
extends Control
class_name TrackView
@export var controller: Node
@export var track_scene: PackedScene
@onready var track_list: VBoxContainer = %TrackList
signal track_added(idx: int)
signal track_deleted(idx: int)
signal tracks_updated()
func _ready() -> void:
pass
func add_track(idx: int):
track_added.emit(idx)
tracks_updated.emit()
pass
func delete_track(idx: int):
track_deleted.emit(idx)
_track_deleted(idx)
tracks_updated.emit()
pass
func _track_added():
var track = track_scene.instantiate() as Track
track.on_deleted.connect(delete_track)
track_list.add_child(track)
pass
func _track_duplicated(idx: int):
pass
func _track_renamed(idx: int):
pass
func _track_deleted(idx: int):
var track = track_list.get_child(idx)
track.call_deferred("free")
pass
func get_track_elements() -> Array[Node]:
return track_list.get_children()

1
Scripts/TrackView.gd.uid Normal file
View File

@@ -0,0 +1 @@
uid://y5e28thwmx7e

9
Source/Clip.cs Normal file
View File

@@ -0,0 +1,9 @@
using System;
namespace AudioEditor;
public class Clip
{
public TimeSpan StartTime { get; private set; }
public TimeSpan EndTime { get; private set; }
}

1
Source/Clip.cs.uid Normal file
View File

@@ -0,0 +1 @@
uid://b20juoqt3ac8x

46
Source/Project.cs Normal file
View File

@@ -0,0 +1,46 @@
using System.Collections.Generic;
namespace AudioEditor;
public class Project
{
public int TrackCount => _tracks.Count;
public void AddTrack()
{
var idx = _tracks.Count;
_tracks.Add(new Track(idx, string.Empty));
}
public void DuplicateTrack(int idx)
{
if (idx < 0 && idx > _tracks.Count - 1)
{
return;
}
var track = _tracks[idx];
var duplicate = track.Duplicate();
_tracks.Add(duplicate);
}
public void RenameTrack(int idx, string name)
{
if (idx < 0 && idx > _tracks.Count - 1)
{
return;
}
}
public void DeleteTrack(int idx)
{
if (idx < 0 && idx > _tracks.Count - 1)
{
return;
}
_tracks.RemoveAt(idx);
}
private List<Track> _tracks = new();
}

1
Source/Project.cs.uid Normal file
View File

@@ -0,0 +1 @@
uid://bvgn3tf07jspe

View File

@@ -7,6 +7,9 @@ namespace AudioEditor;
public partial class ProjectController : Node public partial class ProjectController : Node
{ {
[Signal] public delegate void AudioClipDroppedEventHandler(Vector2 atPosition, string path, string clipName, double startTime, double endTime); [Signal] public delegate void AudioClipDroppedEventHandler(Vector2 atPosition, string path, string clipName, double startTime, double endTime);
[Signal] public delegate void OnTrackAddedEventHandler(int idx);
[Signal] public delegate void OnTrackRenamedEventHandler(int idx, string name);
[Signal] public delegate void OnTrackDuplicatedEventHandler(int idx);
[Signal] public delegate void OnTrackDeletedEventHandler(int trackIdx); [Signal] public delegate void OnTrackDeletedEventHandler(int trackIdx);
public override void _Ready() public override void _Ready()
@@ -14,9 +17,28 @@ public partial class ProjectController : Node
GetWindow().FilesDropped += FilesDropped; GetWindow().FilesDropped += FilesDropped;
} }
public void DeleteTrack(int trackIdx) public void AddTrack(int idx = 0)
{ {
_project.AddTrack();
EmitSignal(SignalName.OnTrackAdded, _project.TrackCount - 1);
}
public void DuplicateTrack(int idx)
{
_project.DuplicateTrack(idx);
EmitSignal(SignalName.OnTrackDuplicated, idx);
}
public void RenameTrack(int idx, string name)
{
_project.RenameTrack(idx, name);
EmitSignal(SignalName.OnTrackRenamed, idx, name);
}
public void DeleteTrack(int idx)
{
_project.DeleteTrack(idx);
EmitSignal(SignalName.OnTrackDeleted, idx);
} }
public void CreateAudioClipPreview(Control audioClip) public void CreateAudioClipPreview(Control audioClip)
@@ -54,4 +76,6 @@ public partial class ProjectController : Node
EmitSignal(SignalName.AudioClipDropped, mousePosition, path, fileName, 0, waveformInfo.Length.TotalMilliseconds); EmitSignal(SignalName.AudioClipDropped, mousePosition, path, fileName, 0, waveformInfo.Length.TotalMilliseconds);
} }
private Project _project = new();
} }

33
Source/Track.cs Normal file
View File

@@ -0,0 +1,33 @@
using System.Collections.Generic;
using System.Linq;
namespace AudioEditor;
public class Track
{
public int Idx { get; private set; }
public string Name { get; private set; }
public Track(int idx, string name)
{
Idx = idx;
Name = name;
}
public Track(int idx, string name, IEnumerable<Clip> clips)
{
Idx = idx;
Name = name;
_clips = clips.ToList();
}
public bool TryAddClip(Clip clip)
{
_clips.Add(clip);
return false;
}
public Track Duplicate() => new(Idx, Name, _clips);
private List<Clip> _clips = new();
}

1
Source/Track.cs.uid Normal file
View File

@@ -0,0 +1 @@
uid://dtk58stvxxdyl

View File

@@ -188,5 +188,8 @@ mouse_filter = 2
[node name="ProjectController" type="Node" parent="."] [node name="ProjectController" type="Node" parent="."]
script = ExtResource("11_gc3ui") script = ExtResource("11_gc3ui")
[connection signal="track_added" from="VBoxContainer/VSplitContainer/HSplitContainer/LeftDock/Tracks" to="ProjectController" method="AddTrack"]
[connection signal="tracks_updated" from="VBoxContainer/VSplitContainer/HSplitContainer/LeftDock/Tracks" to="VBoxContainer/VSplitContainer/HSplitContainer/Timeline" method="queue_redraw"]
[connection signal="clip_added" from="VBoxContainer/VSplitContainer/HSplitContainer/Timeline" to="ProjectController" method="CreateAudioClipPreview"] [connection signal="clip_added" from="VBoxContainer/VSplitContainer/HSplitContainer/Timeline" to="ProjectController" method="CreateAudioClipPreview"]
[connection signal="AudioClipDropped" from="ProjectController" to="VBoxContainer/VSplitContainer/HSplitContainer/Timeline" method="clip_dropped"] [connection signal="AudioClipDropped" from="ProjectController" to="VBoxContainer/VSplitContainer/HSplitContainer/Timeline" method="clip_dropped"]
[connection signal="OnTrackAdded" from="ProjectController" to="VBoxContainer/VSplitContainer/HSplitContainer/LeftDock/Tracks" method="_track_added" unbinds=1]

View File

@@ -1,18 +1,20 @@
[gd_scene load_steps=3 format=3 uid="uid://bpd6g2b3s7tqa"] [gd_scene load_steps=4 format=3 uid="uid://bpd6g2b3s7tqa"]
[ext_resource type="Script" uid="uid://y5e28thwmx7e" path="res://Scripts/TrackView.gd" id="1_8piyd"]
[ext_resource type="PackedScene" uid="uid://c3kajrpp2ux7" path="res://Controls/Track.tscn" id="1_qr47w"] [ext_resource type="PackedScene" uid="uid://c3kajrpp2ux7" path="res://Controls/Track.tscn" id="1_qr47w"]
[ext_resource type="Texture2D" uid="uid://b3mydiolrvqlr" path="res://Assets/Icons/add.svg" id="2_rekuu"] [ext_resource type="Texture2D" uid="uid://b3mydiolrvqlr" path="res://Assets/Icons/add.svg" id="2_rekuu"]
[node name="Tracks" type="Control"] [node name="TrackView" type="Control"]
clip_children = 1 clip_children = 2
custom_minimum_size = Vector2(200, 0) custom_minimum_size = Vector2(264, 0)
layout_mode = 3 layout_mode = 3
anchors_preset = 9 anchors_preset = 9
anchor_bottom = 1.0 anchor_bottom = 1.0
grow_vertical = 2 grow_vertical = 2
script = ExtResource("1_8piyd")
track_scene = ExtResource("1_qr47w")
[node name="VBoxContainer" type="VBoxContainer" parent="."] [node name="VBoxContainer" type="VBoxContainer" parent="."]
custom_minimum_size = Vector2(128, 0)
layout_mode = 1 layout_mode = 1
anchors_preset = 15 anchors_preset = 15
anchor_right = 1.0 anchor_right = 1.0
@@ -22,27 +24,15 @@ grow_vertical = 2
mouse_filter = 2 mouse_filter = 2
[node name="TrackList" type="VBoxContainer" parent="VBoxContainer"] [node name="TrackList" type="VBoxContainer" parent="VBoxContainer"]
custom_minimum_size = Vector2(128, 0) unique_name_in_owner = true
clip_children = 1
layout_mode = 2 layout_mode = 2
size_flags_vertical = 3 size_flags_vertical = 3
mouse_filter = 2 mouse_filter = 2
[node name="Track" parent="VBoxContainer/TrackList" instance=ExtResource("1_qr47w")] [node name="AddTrack" type="Button" parent="VBoxContainer"]
layout_mode = 2
[node name="Track2" parent="VBoxContainer/TrackList" instance=ExtResource("1_qr47w")]
layout_mode = 2
[node name="Track3" parent="VBoxContainer/TrackList" instance=ExtResource("1_qr47w")]
layout_mode = 2
[node name="AddTrack" type="MenuButton" parent="VBoxContainer"]
layout_mode = 2 layout_mode = 2
icon = ExtResource("2_rekuu") icon = ExtResource("2_rekuu")
flat = false
icon_alignment = 1 icon_alignment = 1
item_count = 2
popup/item_0/text = "Audio Track" [connection signal="pressed" from="VBoxContainer/AddTrack" to="." method="add_track" binds= [-1]]
popup/item_0/id = 0
popup/item_1/text = "Label Track"
popup/item_1/id = 1