-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPlaylist.cpp
231 lines (186 loc) · 7.97 KB
/
Playlist.cpp
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
/*
==============================================================================
Playlist.cpp
Created: 6 Mar 2022 11:03:41am
Author: 13z79
==============================================================================
*/
#include <JuceHeader.h>
#include "Playlist.h"
#include <iostream>
//==============================================================================
Playlist::Playlist(juce::AudioFormatManager& _formatManager):formatManager(_formatManager)
{
// In your constructor, you should add any child components, and
// initialise any special settings that your component needs.
// build up the columns for the playlist
// column to show the tile of the audio file
tableComponent.getHeader().addColumn("Track title", 1, 300);
// column to show the URL file location
tableComponent.getHeader().addColumn("URL", 2, 400);
// column to show the length of the audio file
tableComponent.getHeader().addColumn("length", 3, 120);
// column to show the length of the audio file
tableComponent.getHeader().addColumn("type", 4, 100);
// button to load the audio file to the upper deck
tableComponent.getHeader().addColumn("Load to the Up Deck", 5, 180);
// button to load the audio file to the lower deck
tableComponent.getHeader().addColumn("Load to the down Deck", 6, 180);
// set the model to build the table for the playlist
tableComponent.setModel(this);
// make the table component visible in the constructor function
addAndMakeVisible(searchButton);
addAndMakeVisible(importButton);
addAndMakeVisible(exportButton);
addAndMakeVisible(tableComponent);
// add listeners to extra buttons
importButton.onClick = [this] {importButtonClicked(); };
exportButton.onClick = [this] {exportButtonClicked(); };
}
Playlist::~Playlist()
{
}
void Playlist::paint (juce::Graphics& g)
{
/* This demo code just fills the component's background and
draws some placeholder text to get you started.
You should replace everything in this method with your own
drawing code..
*/
g.fillAll (getLookAndFeel().findColour (juce::ResizableWindow::backgroundColourId)); // clear the background
g.setColour (juce::Colours::grey);
g.drawRect (getLocalBounds(), 1); // draw an outline around the component
g.setColour (juce::Colours::white);
g.setFont (20.0f);
g.drawText ("Playlist", getLocalBounds(),
juce::Justification::centred, true); // draw some placeholder text
}
void Playlist::resized()
{
// This method is where you should set the bounds of any child
// components that your component contains..
double rowH = getHeight() / 8;
double rowL = getWidth() / 8;
searchButton.setBounds(0, 0, rowL * 2, rowH);
importButton.setBounds(rowL*6, 0, rowL, rowH);
exportButton.setBounds(rowL * 7, 0, rowL, rowH);
tableComponent.setBounds(0, rowH, getWidth(), rowH*7);
searchButton.setEditable(true);
searchButton.setText("Search through the Playlist", juce::dontSendNotification);
searchButton.setColour(juce::Label::textColourId, juce::Colours::lightgreen);
searchButton.setJustificationType(juce::Justification::centred);
}
int Playlist::getNumRows() {
return trackTitles.size();
}
void Playlist::paintRowBackground(juce::Graphics& g,
int rowNumber,
int width,
int height,
bool rowIsSelected) {
if (rowIsSelected) {
g.fillAll(juce::Colours::blue);
}
else {
g.fillAll(juce::Colours::grey);
}
}
void Playlist::paintCell(juce::Graphics& g,
int rowNumber,
int columnId,
int width,
int height,
bool rowIsSelected) {
if (columnId == 1) {
g.drawText(trackTitles[rowNumber], 2, 0, width - 4, height, juce::Justification::centredLeft, true);
}
if (columnId == 2) {
g.drawText(trackPaths[rowNumber], 2, 0, width - 4, height, juce::Justification::centredLeft, true);
}
if (columnId == 3) {
g.drawText(trackDurations[rowNumber], 2, 0, width - 4, height, juce::Justification::centredLeft, true);
}
if (columnId == 4) {
g.drawText(trackTypes[rowNumber], 2, 0, width - 4, height, juce::Justification::centredLeft, true);
}
}
juce::Component* Playlist::refreshComponentForCell(int rowNumber,
int columnId,
bool isRowSelected,
Component* existingComponentToUpdate) {
if (columnId == 5) {
if (existingComponentToUpdate == nullptr) {
// if there is no load button on the playlist, add the load button at the end of each row
juce::TextButton* upbtn = new juce::TextButton{ "Load to Up Deck" };
// transform the int rowNumber to string and the get the component's ID
juce::String id{ std::to_string(rowNumber) };
upbtn->setComponentID(id);
// add the listener to the button
upbtn->addListener(this);
existingComponentToUpdate = upbtn;
}
}
if (columnId == 6) {
if (existingComponentToUpdate == nullptr) {
// if there is no load button on the playlist, add the load button at the end of each row
juce::TextButton *downbtn = new juce::TextButton{ "Load to Down Deck" };
// transform the int rowNumber to string and the get the component's ID
juce::String id{ std::to_string(rowNumber) };
downbtn->setComponentID(id);
// add the listener to the button
downbtn->addListener(this);
existingComponentToUpdate = downbtn;
}
}
// return the stored components
return existingComponentToUpdate;
}
// listener for the play button
void Playlist::buttonClicked(juce::Button* button) {
// convert the juce::string to the std::string
if (button == downbtn) {
juce::FileChooser chooser{ "What file do you want..." };
if (chooser.browseForFileToOpen()) {
int id = std::stoi(button->getComponentID().toStdString());
//player->loadURL(juce::URL{ trackFiles[id] });
//waveformDisplay.loadURL(juce::URL{ trackFiles[id] });
}
}
else if (button == upbtn) {
}
}
void Playlist::importButtonClicked() {
juce::FileChooser chooser{ "What file do you want..." };
if (chooser.browseForMultipleFilesToOpen()) {
setTracks(chooser.getResults());
}
}
void Playlist::exportButtonClicked() {
}
void Playlist::setTracks(juce::Array<juce::File> trackFiles){
if (searchButton.getText() == "Search through the Playlist") {
for (int i = 0; i < trackFiles.size(); i++) {
trackTitles.push_back(trackFiles[i].getFileName());
trackPaths.push_back(trackFiles[i].getFullPathName());
//trackDurations.push_back(getDuration(juce::URL{ trackFiles[i] }));
trackTypes.push_back(trackFiles[i].getFileExtension());
}
}
else {
DBG("Need to filter the trackTitles");
// so far. still not sure the searching algorithms.
// for instance, how can I type 'r' and get all titles with 'r' as the start?
// is the searching function embedded in the juce gui template?
}
tableComponent.updateContent();
}
std::string Playlist::getDuration(juce::URL audioURL) {
auto* reader = formatManager.createReaderFor(audioURL.createInputStream(false));
if (reader != nullptr) // good file!
{
std::unique_ptr<juce::AudioFormatReaderSource> newSource(new juce::AudioFormatReaderSource(reader, true));
transportSource.setSource(newSource.get(), 0, nullptr, reader->sampleRate);
}
double trackduration = transportSource.getLengthInSeconds();
return std::to_string(trackduration);
}