summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBen Loftis <ben@glw.com>2008-03-24 00:19:03 +0000
committerBen Loftis <ben@glw.com>2008-03-24 00:19:03 +0000
commitcd89c6e3bf1e6c4b786b6ee89bc5fa9a0e780d61 (patch)
treea4cd10f6bcba3ca7a09639b1a49fb740f14b78a4
parent2ac3d8656515e67209a8f5cd82e9e5a2167dbb44 (diff)
initial checkin of freesound integration
git-svn-id: svn://localhost/ardour2/branches/2.0-ongoing@3168 d708f5d6-7413-0410-9779-e7cbd77b26cf
-rw-r--r--gtk2_ardour/SConscript12
-rw-r--r--gtk2_ardour/sfdb_freesound_mootcher.cc489
-rw-r--r--gtk2_ardour/sfdb_freesound_mootcher.h79
-rw-r--r--gtk2_ardour/sfdb_ui.cc281
-rw-r--r--gtk2_ardour/sfdb_ui.h24
5 files changed, 840 insertions, 45 deletions
diff --git a/gtk2_ardour/SConscript b/gtk2_ardour/SConscript
index e390845d9d..7991500fa5 100644
--- a/gtk2_ardour/SConscript
+++ b/gtk2_ardour/SConscript
@@ -79,6 +79,10 @@ if gtkardour['FFT_ANALYSIS']:
gtkardour.Merge ([libraries['fftw3f']])
gtkardour.Append(CCFLAGS='-DFFT_ANALYSIS')
+if gtkardour['FREESOUND']:
+ gtkardour.Merge ([libraries['curl']])
+ gtkardour.Append(CCFLAGS='-DFREESOUND')
+
if gtkardour['RUBBERBAND']:
gtkardour.Merge ([ libraries['rubberband'] ])
else:
@@ -237,6 +241,10 @@ fft_graph.cc
fft_result.cc
""")
+freesound_files=Split("""
+sfdb_freesound_mootcher.cc
+""")
+
pixmap_files = glob.glob('pixmaps/*.xpm')
icon_files = glob.glob ('icons/*.png')
@@ -297,6 +305,9 @@ else:
if env['FFT_ANALYSIS']:
extra_sources += fft_analysis_files
+if env['FREESOUND']:
+ extra_sources += freesound_files
+
intl_files += extra_sources
gtkardour.Append(CCFLAGS="-D_REENTRANT -D_LARGEFILE_SOURCE -D_LARGEFILE64_SOURCE")
@@ -533,6 +544,7 @@ env.Alias ('tarball', env.Distribute (env['DISTTREE'],
gtkosx_files +
x11_files +
fft_analysis_files +
+ freesound_files +
glob.glob('po/*.po') + glob.glob('*.h')))
# generate a prototype full-featured ardour_ui.rc file
diff --git a/gtk2_ardour/sfdb_freesound_mootcher.cc b/gtk2_ardour/sfdb_freesound_mootcher.cc
new file mode 100644
index 0000000000..c9e04c14d5
--- /dev/null
+++ b/gtk2_ardour/sfdb_freesound_mootcher.cc
@@ -0,0 +1,489 @@
+/* sfdb_freesound_mootcher.cpp **********************************************************************
+
+ Adapted for Ardour by Ben Loftis, March 2008
+
+ Mootcher 23-8-2005
+
+ Mootcher Online Access to thefreesoundproject website
+ http://freesound.iua.upf.edu/
+
+ GPL 2005 Jorn Lemon
+ mail for questions/remarks: mootcher@twistedlemon.nl
+ or go to the freesound website forum
+
+ -----------------------------------------------------------------
+
+ Includes:
+ curl.h (version 7.14.0)
+ Librarys:
+ libcurl.lib
+
+ -----------------------------------------------------------------
+ Licence GPL:
+
+ This program is free software; you can redistribute it and/or
+ modify it under the terms of the GNU General Public License
+ as published by the Free Software Foundation; either version 2
+ of the License, or (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+
+
+*************************************************************************************/
+#include "sfdb_freesound_mootcher.h"
+
+#include <pbd/xml++.h>
+
+#include <sys/stat.h>
+#include <sys/types.h>
+
+#include <ardour/audio_library.h>
+
+#define TRUE 1
+
+//------------------------------------------------------------------------
+Mootcher:: Mootcher(const char *saveLocation)
+ : curl( NULL )
+ , connection( NULL )
+{
+ changeWorkingDir(saveLocation);
+};
+//------------------------------------------------------------------------
+Mootcher:: ~Mootcher()
+{
+ remove( "cookiejar.txt" );
+}
+//------------------------------------------------------------------------
+const char* Mootcher::changeWorkingDir(const char *saveLocation)
+{
+ basePath = saveLocation;
+#ifdef __WIN32__
+ std::string replace = "/";
+ int pos = (int)basePath.find("\\");
+ while( pos != std::string::npos ){
+ basePath.replace(pos, 1, replace);
+ pos = (int)basePath.find("\\");
+ }
+#endif
+ //
+ int pos2 = basePath.find_last_of("/");
+ if(basePath.length() != (pos2+1)) basePath += "/";
+ // add a check if the given directory exists
+ createResourceLocation();
+ return basePath.c_str();
+}
+
+//------------------------------------------------------------------------
+void Mootcher::createResourceLocation()
+{
+ // create a snd directory
+ std::string sndLocation = basePath + "snd";
+ mkdir(sndLocation.c_str(), 0x777);
+}
+
+//------------------------------------------------------------------------
+size_t Mootcher::WriteMemoryCallback(void *ptr, size_t size, size_t nmemb, void *data)
+{
+ register int realsize = (int)(size * nmemb);
+ struct MemoryStruct *mem = (struct MemoryStruct *)data;
+
+ // There might be a realloc() out there that doesn't like
+ // reallocing NULL pointers, so we take care of it here
+ if(mem->memory) mem->memory = (char *)realloc(mem->memory, mem->size + realsize + 1);
+ else mem->memory = (char *)malloc(mem->size + realsize + 1);
+
+ if (mem->memory) {
+ memcpy(&(mem->memory[mem->size]), ptr, realsize);
+ mem->size += realsize;
+ mem->memory[mem->size] = 0;
+ }
+ return realsize;
+}
+
+
+//------------------------------------------------------------------------
+void Mootcher::toLog(std::string input)
+{
+//printf("%s", input.c_str());// for debugging
+}
+
+
+//------------------------------------------------------------------------
+void Mootcher::setcUrlOptions()
+{
+ // basic init for curl
+ curl_global_init(CURL_GLOBAL_ALL);
+ // some servers don't like requests that are made without a user-agent field, so we provide one
+ curl_easy_setopt(curl, CURLOPT_USERAGENT, "libcurl-agent/1.0");
+ // setup curl error buffer
+ CURLcode res = curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorBuffer);
+ // always use the cookie with session id which is received at the login
+ curl_easy_setopt(curl, CURLOPT_COOKIEFILE, "cookiejar.txt");
+ // Allow redirection
+ curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1);
+}
+
+//------------------------------------------------------------------------
+int Mootcher::doLogin(std::string login, std::string password)
+{
+ if(connection==1)
+ return 1;
+
+ struct MemoryStruct xml_page;
+ xml_page.memory = NULL;
+ xml_page.size = NULL;
+
+ // create the post message from the login and password
+ std::string postMessage;
+ postMessage += "username=";
+ postMessage += curl_escape(login.c_str(), 0);
+ postMessage += "&password=";
+ postMessage += curl_escape(password.c_str(), 0);
+ postMessage += "&login=";
+ postMessage += curl_escape("1", 0);
+ postMessage += "&redirect=";
+ postMessage += curl_escape("../tests/login.php", 0);
+
+ // Do the setup for libcurl
+ curl = curl_easy_init();
+
+ if(curl)
+ {
+ setcUrlOptions();
+ curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteMemoryCallback);
+ curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&xml_page);
+ // use POST for login variables
+ curl_easy_setopt(curl, CURLOPT_POST, TRUE);
+ curl_easy_setopt(curl, CURLOPT_POSTFIELDS, postMessage.c_str());
+ curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, -1);
+
+ // the url to get
+ std::string login_url = "http://freesound.iua.upf.edu/forum/login.php";
+ curl_easy_setopt(curl, CURLOPT_URL, login_url.c_str() );
+
+ // perform online request
+ CURLcode res = curl_easy_perform(curl);
+ // if something goes wrong 'connection' is set to '0'
+ if(strcmp(curl_easy_strerror(res), "no error") == 0) connection = 1;
+ else connection = 0;
+
+ if (connection == 1){
+ std::string check_page = xml_page.memory;
+ int test = (int)check_page.find("login"); //logged
+ if( test != -1){
+ sprintf(message, "Login: %s\n", xml_page.memory);
+ toLog(message);
+ }
+ else {
+ sprintf(message, "Login: Check username and password\n");
+ toLog(message);
+ connection = 0;
+ }
+ }
+
+ // free the memory
+ if(xml_page.memory){
+ free( xml_page.memory );
+ xml_page.memory = NULL;
+ xml_page.size = NULL;
+ }
+
+ return connection;
+ }
+ else return 3; // will be returned if a curl related problem ocurrs
+}
+//------------------------------------------------------------------------
+std::string Mootcher::searchText(std::string word)
+{
+ struct MemoryStruct xml_page;
+ xml_page.memory = NULL;
+ xml_page.size = NULL;
+
+ std::string result;
+
+ if(connection != 0)
+ {
+ // create a url encoded post message
+ std::string postMessage;
+ char tempString[ 128 ];
+ char *tempPointer = &tempString[0];
+
+ postMessage = "search=";
+ postMessage += curl_escape(word.c_str(), 0);
+ sprintf( tempPointer, "&searchDescriptions=1");
+ postMessage += tempPointer;
+ sprintf( tempPointer, "&searchtags=1");
+ postMessage += tempPointer;
+
+ if(curl)
+ {
+ // basic init for curl
+ setcUrlOptions();
+ curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteMemoryCallback);
+ curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&xml_page);
+ // setup the post message
+ curl_easy_setopt(curl, CURLOPT_POST, TRUE);
+ curl_easy_setopt(curl, CURLOPT_POSTFIELDS, postMessage.c_str());
+ curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, -1);
+
+ // the url to get
+ std::string search_url = "http://freesound.iua.upf.edu/searchTextXML.php";
+ curl_easy_setopt(curl, CURLOPT_URL, search_url.c_str());
+
+ // perform the online search
+ CURLcode res = curl_easy_perform(curl);
+ if(strcmp(curl_easy_strerror(res), "no error") == 0) connection = 1;
+ else connection = 0;
+
+ result = xml_page.memory;
+//printf("%s/n", result.c_str());
+
+ // free the memory
+ if(xml_page.memory){
+ free( xml_page.memory );
+ xml_page.memory = NULL;
+ xml_page.size = NULL;
+ }
+
+ }
+ }
+
+ return result;
+}
+
+//------------------------------------------------------------------------
+std::string Mootcher::changeExtension(std::string filename)
+{
+ std::string aiff = ".aiff";
+ std::string aif = ".aif";
+ std::string wav = ".wav";
+ std::string mp3 = ".mp3";
+ std::string ogg = ".ogg";
+ std::string flac = ".flac";
+
+ std::string replace = ".xml";
+ int pos = 0;
+
+ pos = (int)filename.find(aiff);
+ if(pos != std::string::npos) filename.replace(pos, aiff.size(), replace);
+ pos = (int)filename.find(aif);
+ if(pos != std::string::npos) filename.replace(pos, aif.size(), replace);
+ pos = (int)filename.find(wav);
+ if(pos != std::string::npos) filename.replace(pos, wav.size(), replace);
+ pos = (int)filename.find(mp3);
+ if(pos != std::string::npos) filename.replace(pos, mp3.size(), replace);
+ pos = (int)filename.find(ogg);
+ if(pos != std::string::npos) filename.replace(pos, ogg.size(), replace);
+ pos = (int)filename.find(flac);
+ if(pos != std::string::npos) filename.replace(pos, flac.size(), replace);
+
+ return filename;
+}
+//------------------------------------------------------------------------
+void Mootcher::GetXml(std::string ID, struct MemoryStruct &xml_page)
+{
+
+ if(curl) {
+ setcUrlOptions();
+ curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteMemoryCallback);
+ curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&xml_page);
+
+ // URL to get
+ std::string getxml_url = "http://freesound.iua.upf.edu/samplesViewSingleXML.php?id=";
+ getxml_url += ID;
+
+ curl_easy_setopt(curl, CURLOPT_URL, getxml_url.c_str() );
+
+ // get it!
+ CURLcode res = curl_easy_perform(curl);
+ if(strcmp(curl_easy_strerror(res), "no error") == 0) connection = 1;
+ else connection = 0;
+ }
+}
+//------------------------------------------------------------------------
+std::string Mootcher::getXmlFile(std::string ID, int &length)
+{
+ struct MemoryStruct xml_page;
+ xml_page.memory = NULL;
+ xml_page.size = NULL;
+
+ std::string xmlFileName;
+ std::string audioFileName;
+ std::string filename;
+
+ if(connection != 0) {
+ // download the xmlfile into xml_page
+ GetXml(ID, xml_page);
+
+ // if sample ID does not exist on the freesound website
+ if(strcmp(xml_page.memory, "sample non existant") == 0){
+ free( xml_page.memory );
+ sprintf(message, "getXmlFile: sample with ID:%s does not exist!\n", ID.c_str() );
+ toLog(message);
+ return filename;
+ } else {
+ XMLTree doc;
+ doc.read_buffer( xml_page.memory );
+ XMLNode *freesound = doc.root();
+
+ // if the page is not a valid xml document with a 'freesound' root
+ if( freesound == NULL){
+ sprintf(message, "getXmlFile: There is no valid root in the xml file");
+ toLog(message);
+ } else {
+ XMLNode *sample = freesound->child("sample");
+ XMLNode *name = NULL;
+ XMLNode *filesize = NULL;
+ if (sample) {
+ name = sample->child("originalFilename");
+ filesize = sample->child("filesize");
+ }
+
+ // get the file name and size from xml file
+ if (sample && name && filesize) {
+
+ audioFileName = name->child("text")->content();
+ sprintf( message, "getXmlFile: %s needs to be downloaded\n", audioFileName.c_str() );
+ toLog(message);
+
+ length = atoi(filesize->child("text")->content().c_str());
+
+ // create new filename with the ID number
+ filename = basePath;
+ filename += "snd/";
+ filename += sample->property("id")->value();
+ filename += "-";
+ filename += audioFileName;
+ // change the extention into .xml
+ xmlFileName = changeExtension( filename );
+
+ sprintf(message, "getXmlFile: saving XML: %s\n", xmlFileName.c_str() );
+ toLog(message);
+
+ // save the xml file to disk
+ doc.write(xmlFileName.c_str());
+
+ //store all the tags in the database
+ XMLNode *tags = sample->child("tags");
+ if (tags) {
+ XMLNodeList children = tags->children();
+ XMLNodeConstIterator niter;
+ vector<string> strings;
+ for (niter = children.begin(); niter != children.end(); ++niter) {
+ XMLNode *node = *niter;
+ if( strcmp( node->name().c_str(), "tag") == 0 ) {
+ XMLNode *text = node->child("text");
+ if (text) strings.push_back(text->content());
+ }
+ }
+ ARDOUR::Library->set_tags (string("//")+filename, strings);
+ ARDOUR::Library->save_changes ();
+ }
+ }
+
+ // clear the memory
+ if(xml_page.memory){
+ free( xml_page.memory );
+ xml_page.memory = NULL;
+ xml_page.size = 0;
+ }
+ return audioFileName;
+ }
+ }
+ }
+ else {
+ return audioFileName;
+ }
+
+}
+
+int audioFileWrite(void *buffer, size_t size, size_t nmemb, void *file)
+{
+ return (int)fwrite(buffer, size, nmemb, (FILE*) file);
+};
+
+//------------------------------------------------------------------------
+std::string Mootcher::getFile(std::string ID)
+{
+ CURLcode result_curl;
+
+ std::string audioFileName;
+
+ if(connection != 0)
+ {
+ int length;
+ std::string name = getXmlFile(ID, length);
+ if( name != "" ){
+
+ // create new filename with the ID number
+ audioFileName += basePath;
+ audioFileName += "snd/";
+ audioFileName += ID;
+ audioFileName += "-";
+ audioFileName += name;
+
+ //check to see if audio file already exists
+ FILE *testFile = fopen(audioFileName.c_str(), "r");
+ if (testFile) { //TODO: should also check length to see if file is complete
+ fseek (testFile , 0 , SEEK_END);
+ if (ftell (testFile) == length) {
+ sprintf(message, "%s already exists\n", audioFileName.c_str() );
+ toLog(message);
+ fclose (testFile);
+ return audioFileName;
+ } else {
+ remove( audioFileName.c_str() ); //file was not correct length, delete it and try again
+ }
+ }
+
+
+ //now download the actual file
+ if (curl) {
+
+ FILE* theFile;
+ theFile = fopen( audioFileName.c_str(), "wb" );
+
+ // create the download url, this url will also update the download statics on the site
+ std::string audioURL;
+ audioURL += "http://freesound.iua.upf.edu/samplesDownload.php?id=";
+ audioURL += ID;
+
+ setcUrlOptions();
+ curl_easy_setopt(curl, CURLOPT_URL, audioURL.c_str() );
+ curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, audioFileWrite);
+ curl_easy_setopt(curl, CURLOPT_WRITEDATA, theFile);
+ CURLcode res = curl_easy_perform(curl);
+ if(strcmp(curl_easy_strerror(res), "no error") == 0) connection = 1;
+ else connection = 0;
+
+ fclose(theFile);
+ }
+
+/*
+ bar.dlnowMoo = 0;
+ bar.dltotalMoo = 0;
+ curl_easy_setopt (curl, CURLOPT_NOPROGRESS, 0); // turn on the process bar thingy
+ curl_easy_setopt (curl, CURLOPT_PROGRESSFUNCTION, progress_callback);
+ curl_easy_setopt (curl, CURLOPT_PROGRESSDATA, &bar);
+*/
+ }
+ }
+
+ return audioFileName;
+}
+
+//---------
+int Mootcher::progress_callback(void *bar, double dltotal, double dlnow, double ultotal, double ulnow)
+{
+ struct dlprocess *lbar = (struct dlprocess *) bar;
+ lbar->dltotalMoo = dltotal;
+ lbar->dlnowMoo = dlnow;
+ return 0;
+}
diff --git a/gtk2_ardour/sfdb_freesound_mootcher.h b/gtk2_ardour/sfdb_freesound_mootcher.h
new file mode 100644
index 0000000000..af5f4e0bab
--- /dev/null
+++ b/gtk2_ardour/sfdb_freesound_mootcher.h
@@ -0,0 +1,79 @@
+/*sfdb_freesound_mootcher.h****************************************************************************
+
+ Adapted for Ardour by Ben Loftis, March 2008
+
+ Mootcher Online Access to thefreesoundproject website
+ http://freesound.iua.upf.edu/
+
+ GPL 2005 Jorn Lemon
+ mail for questions/remarks: mootcher@twistedlemon.nl
+ or go to the freesound website forum
+
+*****************************************************************************/
+
+#include <string>
+#include <fstream>
+#include <iostream>
+#include <stdio.h>
+#include <cstring>
+#include <string>
+#include <sstream>
+#include <vector>
+//#include <ctime>
+
+#include "curl/curl.h"
+
+
+// mootcher version
+#define ___VERSION___ 1.3
+
+//--- struct to store XML file
+struct MemoryStruct {
+ char *memory;
+ size_t size;
+};
+
+//--- for download process viewing
+struct dlprocess {
+ double dltotalMoo;
+ double dlnowMoo;
+};
+
+class Mootcher
+{
+public:
+ Mootcher(const char *saveLocation);
+ ~Mootcher();
+
+ int doLogin(std::string login, std::string password);
+ std::string getFile(std::string ID);
+ std::string searchText(std::string word);
+
+
+ struct dlprocess bar;
+
+private:
+
+ const char* changeWorkingDir(const char *saveLocation);
+ void createResourceLocation();
+
+ std::string getXmlFile(std::string ID, int &length);
+ void GetXml(std::string ID, struct MemoryStruct &xml_page);
+ std::string changeExtension(std::string filename);
+
+ void toLog(std::string input);
+
+ void setcUrlOptions();
+
+ static size_t WriteMemoryCallback(void *ptr, size_t size, size_t nmemb, void *data);
+ static int progress_callback(void *clientp, double dltotal, double dlnow, double ultotal, double ulnow);
+
+ CURL *curl;
+ char errorBuffer[CURL_ERROR_SIZE]; // storage for cUrl error message
+
+ int connection; // is 0 if no connection
+ char message[128]; // storage for messages that are send to the logfile
+
+ std::string basePath;
+ std::string xmlLocation;
+};
diff --git a/gtk2_ardour/sfdb_ui.cc b/gtk2_ardour/sfdb_ui.cc
index d40487e5f5..7b8babca58 100644
--- a/gtk2_ardour/sfdb_ui.cc
+++ b/gtk2_ardour/sfdb_ui.cc
@@ -32,6 +32,8 @@
#include <pbd/convert.h>
#include <pbd/tokenizer.h>
#include <pbd/enumwriter.h>
+#include <pbd/pthread_utils.h>
+#include <pbd/xml++.h>
#include <gtkmm2ext/utils.h>
@@ -52,6 +54,10 @@
#include "utils.h"
#include "gain_meter.h"
+#ifdef FREESOUND
+#include "sfdb_freesound_mootcher.h"
+#endif
+
#include "i18n.h"
using namespace ARDOUR;
@@ -386,70 +392,138 @@ SoundFileBox::save_tags (const vector<string>& tags)
SoundFileBrowser::SoundFileBrowser (Gtk::Window& parent, string title, ARDOUR::Session* s, bool persistent)
: ArdourDialog (parent, title, false, false),
found_list (ListStore::create(found_list_columns)),
+ freesound_list (ListStore::create(freesound_list_columns)),
chooser (FILE_CHOOSER_ACTION_OPEN),
found_list_view (found_list),
+ freesound_list_view (freesound_list),
preview (persistent),
- found_search_btn (_("Search"))
+ found_search_btn (_("Search")),
+ freesound_search_btn (_("Start Downloading"))
{
+ resetting_ourselves = false;
+ gm = 0;
+
if (ARDOUR::Profile->get_sae()) {
chooser.add_shortcut_folder_uri("file:///Library/GarageBand/Apple Loops");
chooser.add_shortcut_folder_uri("file:///Library/Application Support/GarageBand/Instrument Library/Sampler/Sampler Files");
}
- VBox* vbox;
- HBox* hbox;
+ //add the file chooser
+ {
+ chooser.set_border_width (12);
+ custom_filter.add_custom (FILE_FILTER_FILENAME, mem_fun(*this, &SoundFileBrowser::on_custom));
+ custom_filter.set_name (_("Audio files"));
+
+ matchall_filter.add_pattern ("*.*");
+ matchall_filter.set_name (_("All files"));
+
+ chooser.add_filter (custom_filter);
+ chooser.add_filter (matchall_filter);
+ chooser.set_select_multiple (true);
+ chooser.signal_update_preview().connect(mem_fun(*this, &SoundFileBrowser::update_preview));
+ chooser.signal_file_activated().connect (mem_fun (*this, &SoundFileBrowser::chooser_file_activated));
+
+ if (!persistent_folder.empty()) {
+ chooser.set_current_folder (persistent_folder);
+ }
+ notebook.append_page (chooser, _("Browse Files"));
+ }
+
+ //add tag search
+ {
+ VBox* vbox;
+ HBox* hbox;
- gm = 0;
+ hpacker.set_spacing (6);
+ hpacker.pack_start (notebook, true, true);
+ hpacker.pack_start (preview, false, false);
- set_session (s);
- resetting_ourselves = false;
-
- hpacker.set_spacing (6);
- hpacker.pack_start (notebook, true, true);
- hpacker.pack_start (preview, false, false);
+ get_vbox()->pack_start (hpacker, true, true);
- get_vbox()->pack_start (hpacker, true, true);
+ hbox = manage(new HBox);
+ hbox->pack_start (found_entry);
+ hbox->pack_start (found_search_btn);
+
+ Gtk::ScrolledWindow *scroll = manage(new ScrolledWindow);
+ scroll->add(found_list_view);
+ scroll->set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
- hbox = manage(new HBox);
- hbox->pack_start (found_entry);
- hbox->pack_start (found_search_btn);
-
- vbox = manage(new VBox);
- vbox->pack_start (*hbox, PACK_SHRINK);
- vbox->pack_start (found_list_view);
- found_list_view.append_column(_("Paths"), found_list_columns.pathname);
-
- chooser.set_border_width (12);
+ vbox = manage(new VBox);
+ vbox->pack_start (*hbox, PACK_SHRINK);
+ vbox->pack_start (*scroll);
+
+ found_list_view.append_column(_("Paths"), found_list_columns.pathname);
- notebook.append_page (chooser, _("Browse Files"));
- notebook.append_page (*vbox, _("Search Tags"));
+ found_list_view.get_selection()->signal_changed().connect(mem_fun(*this, &SoundFileBrowser::found_list_view_selected));
+
+ found_list_view.signal_row_activated().connect (mem_fun (*this, &SoundFileBrowser::found_list_view_activated));
- notebook.set_size_request (500, -1);
+ found_search_btn.signal_clicked().connect(mem_fun(*this, &SoundFileBrowser::found_search_clicked));
+ found_entry.signal_activate().connect(mem_fun(*this, &SoundFileBrowser::found_search_clicked));
+
+ notebook.append_page (*vbox, _("Search Tags"));
+ }
+
+ //add freesound search
+#ifdef FREESOUND
+ {
+ VBox* vbox;
+ HBox* passbox;
+ Label* label;
+
+ hpacker.set_spacing (6);
+ hpacker.pack_start (notebook, true, true);
+ hpacker.pack_start (preview, false, false);
+
+ get_vbox()->pack_start (hpacker, true, true);
+
+ passbox = manage(new HBox);
+ passbox->set_border_width (12);
+ passbox->set_spacing (6);
+
+ label = manage (new Label);
+ label->set_text (_("User:"));
+ passbox->pack_start (*label, false, false);
+ passbox->pack_start (freesound_name_entry);
+ label = manage (new Label);
+ label->set_text (_("Password:"));
+ passbox->pack_start (*label, false, false);
+ passbox->pack_start (freesound_pass_entry);
+ label = manage (new Label);
+ label->set_text (_("Tags:"));
+ passbox->pack_start (*label, false, false);
+ passbox->pack_start (freesound_entry, false, false);
+ passbox->pack_start (freesound_search_btn, false, false);
+
+ Gtk::ScrolledWindow *scroll = manage(new ScrolledWindow);
+ scroll->add(freesound_list_view);
+ scroll->set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
- found_list_view.get_selection()->set_mode (SELECTION_MULTIPLE);
- found_list_view.signal_row_activated().connect (mem_fun (*this, &SoundFileBrowser::found_list_view_activated));
+ vbox = manage(new VBox);
+ vbox->pack_start (*passbox, PACK_SHRINK);
+ vbox->pack_start(*scroll);
+
+ //vbox->pack_start (freesound_list_view);
- custom_filter.add_custom (FILE_FILTER_FILENAME, mem_fun(*this, &SoundFileBrowser::on_custom));
- custom_filter.set_name (_("Audio files"));
+ freesound_list_view.append_column(_("Paths"), freesound_list_columns.pathname);
- matchall_filter.add_pattern ("*.*");
- matchall_filter.set_name (_("All files"));
+ freesound_list_view.get_selection()->signal_changed().connect(mem_fun(*this, &SoundFileBrowser::freesound_list_view_selected));
+
+ //freesound_list_view.get_selection()->set_mode (SELECTION_MULTIPLE);
+ freesound_list_view.signal_row_activated().connect (mem_fun (*this, &SoundFileBrowser::freesound_list_view_activated));
- chooser.add_filter (custom_filter);
- chooser.add_filter (matchall_filter);
- chooser.set_select_multiple (true);
- chooser.signal_update_preview().connect(mem_fun(*this, &SoundFileBrowser::update_preview));
- chooser.signal_file_activated().connect (mem_fun (*this, &SoundFileBrowser::chooser_file_activated));
+ freesound_search_btn.signal_clicked().connect(mem_fun(*this, &SoundFileBrowser::freesound_search_clicked));
+ freesound_entry.signal_activate().connect(mem_fun(*this, &SoundFileBrowser::freesound_search_clicked));
- if (!persistent_folder.empty()) {
- chooser.set_current_folder (persistent_folder);
+ notebook.append_page (*vbox, _("Search Freesound"));
}
-
- found_list_view.get_selection()->signal_changed().connect(mem_fun(*this, &SoundFileBrowser::found_list_view_selected));
+#endif
- found_search_btn.signal_clicked().connect(mem_fun(*this, &SoundFileBrowser::found_search_clicked));
- found_entry.signal_activate().connect(mem_fun(*this, &SoundFileBrowser::found_search_clicked));
+
+ notebook.set_size_request (500, -1);
+
+ set_session (s);
add_button (Stock::CANCEL, RESPONSE_CANCEL);
add_button (Stock::APPLY, RESPONSE_APPLY);
@@ -490,6 +564,12 @@ SoundFileBrowser::found_list_view_activated (const TreeModel::Path& path, TreeVi
}
void
+SoundFileBrowser::freesound_list_view_activated (const TreeModel::Path& path, TreeViewColumn* col)
+{
+ preview.audition ();
+}
+
+void
SoundFileBrowser::set_session (Session* s)
{
ArdourDialog::set_session (s);
@@ -588,6 +668,29 @@ SoundFileBrowser::found_list_view_selected ()
}
void
+SoundFileBrowser::freesound_list_view_selected ()
+{
+ if (!reset_options ()) {
+ set_response_sensitive (RESPONSE_OK, false);
+ } else {
+ ustring file;
+
+ TreeView::Selection::ListHandle_Path rows = freesound_list_view.get_selection()->get_selected_rows ();
+
+ if (!rows.empty()) {
+ TreeIter iter = freesound_list->get_iter(*rows.begin());
+ file = (*iter)[freesound_list_columns.pathname];
+ chooser.set_filename (file);
+ set_response_sensitive (RESPONSE_OK, true);
+ } else {
+ set_response_sensitive (RESPONSE_OK, false);
+ }
+
+ preview.setup_labels (file);
+ }
+}
+
+void
SoundFileBrowser::found_search_clicked ()
{
string tag_string = found_entry.get_text ();
@@ -611,6 +714,88 @@ SoundFileBrowser::found_search_clicked ()
}
}
+void*
+freesound_search_thread_entry (void* arg)
+{
+ PBD::ThreadCreated (pthread_self(), X_("Freesound Search"));
+
+ static_cast<SoundFileBrowser*>(arg)->freesound_search_thread ();
+
+ return 0;
+}
+
+bool searching = false;
+bool canceling = false;
+
+void
+SoundFileBrowser::freesound_search_clicked ()
+{
+ if (canceling) //already canceling, button does nothing
+ return;
+
+ if ( searching ) {
+ freesound_search_btn.set_label(_("Canceling.."));
+ canceling = true;
+ } else {
+ searching = true;
+ freesound_search_btn.set_label(_("Cancel"));
+ pthread_t freesound_thr;
+ pthread_create_and_store ("freesound_search", &freesound_thr, 0, freesound_search_thread_entry, this);
+ }
+}
+
+void
+SoundFileBrowser::freesound_search_thread()
+{
+#ifdef FREESOUND
+ freesound_list->clear();
+
+ string path;
+ path = Glib::get_home_dir();
+ path += "/Freesound/";
+ Mootcher theMootcher(path.c_str());
+
+ string name_string = freesound_name_entry.get_text ();
+ string pass_string = freesound_pass_entry.get_text ();
+ string search_string = freesound_entry.get_text ();
+
+ if ( theMootcher.doLogin( name_string, pass_string ) ) {
+
+ string theString = theMootcher.searchText(search_string);
+
+ XMLTree doc;
+ doc.read_buffer( theString );
+ XMLNode *root = doc.root();
+
+ if (root==NULL) return;
+
+ if ( strcmp(root->name().c_str(), "freesound") == 0) {
+
+ XMLNode *node = 0;
+ XMLNodeList children = root->children();
+ XMLNodeConstIterator niter;
+ for (niter = children.begin(); niter != children.end() && !canceling; ++niter) {
+ node = *niter;
+ if( strcmp( node->name().c_str(), "sample") == 0 ){
+ XMLProperty *prop=node->property ("id");
+ string filename = theMootcher.getFile( prop->value().c_str() );
+ if ( filename != "" ) {
+ TreeModel::iterator new_row = freesound_list->append();
+ TreeModel::Row row = *new_row;
+ string path = Glib::filename_from_uri (string ("file:") + filename);
+ row[freesound_list_columns.pathname] = path;
+ }
+ }
+ }
+ }
+ }
+
+ searching = false;
+ canceling = false;
+ freesound_search_btn.set_label(_("Start Downloading"));
+#endif
+}
+
vector<ustring>
SoundFileBrowser::get_paths ()
{
@@ -629,7 +814,7 @@ SoundFileBrowser::get_paths ()
}
}
- } else {
+ } else if (n==1){
typedef TreeView::Selection::ListHandle_Path ListPath;
@@ -640,6 +825,17 @@ SoundFileBrowser::get_paths ()
results.push_back (str);
}
+ } else {
+
+ typedef TreeView::Selection::ListHandle_Path ListPath;
+
+ ListPath rows = freesound_list_view.get_selection()->get_selected_rows ();
+ for (ListPath::iterator i = rows.begin() ; i != rows.end(); ++i) {
+ TreeIter iter = freesound_list->get_iter(*i);
+ ustring str = (*iter)[freesound_list_columns.pathname];
+
+ results.push_back (str);
+ }
}
return results;
@@ -936,6 +1132,7 @@ SoundFileChooser::SoundFileChooser (Gtk::Window& parent, string title, ARDOUR::S
{
chooser.set_select_multiple (false);
found_list_view.get_selection()->set_mode (SELECTION_SINGLE);
+ freesound_list_view.get_selection()->set_mode (SELECTION_SINGLE);
}
void
diff --git a/gtk2_ardour/sfdb_ui.h b/gtk2_ardour/sfdb_ui.h
index 43f76a9ea2..0f164bcb11 100644
--- a/gtk2_ardour/sfdb_ui.h
+++ b/gtk2_ardour/sfdb_ui.h
@@ -117,6 +117,9 @@ class SoundFileBrowser : public ArdourDialog
FoundTagColumns found_list_columns;
Glib::RefPtr<Gtk::ListStore> found_list;
+ FoundTagColumns freesound_list_columns;
+ Glib::RefPtr<Gtk::ListStore> freesound_list;
+
public:
SoundFileBrowser (Gtk::Window& parent, std::string title, ARDOUR::Session* _s, bool persistent);
virtual ~SoundFileBrowser ();
@@ -127,20 +130,30 @@ class SoundFileBrowser : public ArdourDialog
void clear_selection ();
Gtk::FileChooserWidget chooser;
+
+ SoundFileBox preview;
+
+ Gtk::Entry found_entry;
+ Gtk::Button found_search_btn;
Gtk::TreeView found_list_view;
+ Gtk::Entry freesound_name_entry;
+ Gtk::Entry freesound_pass_entry;
+ Gtk::Entry freesound_entry;
+ Gtk::Button freesound_search_btn;
+ Gtk::TreeView freesound_list_view;
+
+ void freesound_search_thread();
+
protected:
bool resetting_ourselves;
Gtk::FileFilter custom_filter;
Gtk::FileFilter matchall_filter;
- SoundFileBox preview;
Gtk::HBox hpacker;
static Glib::ustring persistent_folder;
- Gtk::Entry found_entry;
- Gtk::Button found_search_btn;
Gtk::Notebook notebook;
GainMeter* gm;
@@ -153,10 +166,15 @@ class SoundFileBrowser : public ArdourDialog
sigc::connection metering_connection;
void update_preview ();
+
void found_list_view_selected ();
void found_list_view_activated (const Gtk::TreeModel::Path& path, Gtk::TreeViewColumn*);
void found_search_clicked ();
+ void freesound_list_view_selected ();
+ void freesound_list_view_activated (const Gtk::TreeModel::Path& path, Gtk::TreeViewColumn*);
+ void freesound_search_clicked ();
+
void chooser_file_activated ();
bool on_custom (const Gtk::FileFilter::Info& filter_info);