*** NEW CODING POLICY ***
[ardour.git] / libs / ardour / audio_library.cc
1 /*
2     Copyright (C) 2003-2006 Paul Davis
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include <sstream>
21
22 #include <libxml/uri.h>
23
24 #include <lrdf.h>
25 #include <glibmm/miscutils.h>
26
27 #include <glibmm/convert.h>
28
29 #include "pbd/compose.h"
30
31 #include "ardour/audio_library.h"
32 #include "ardour/utils.h"
33 #include "ardour/filesystem_paths.h"
34
35 #include "i18n.h"
36
37 using namespace std;
38 using namespace ARDOUR;
39
40 namespace {
41         const char* const sfdb_file_name = "sfdb";
42 } // anonymous namespace
43
44 static const char* TAG = "http://ardour.org/ontology/Tag";
45
46 AudioLibrary::AudioLibrary ()
47 {
48         sys::path sfdb_file_path(user_config_directory ());
49
50         sfdb_file_path /= sfdb_file_name;
51
52         src = Glib::filename_to_uri (sfdb_file_path.to_string ());
53         
54         // workaround for possible bug in raptor that crashes when saving to a
55         // non-existant file.
56
57         touch_file(sfdb_file_path.to_string());
58
59         lrdf_read_file(src.c_str());
60 }
61
62 AudioLibrary::~AudioLibrary ()
63 {
64 }
65
66 void
67 AudioLibrary::save_changes ()
68 {
69         if (lrdf_export_by_source(src.c_str(), src.substr(5).c_str())) {
70                 PBD::warning << string_compose(_("Could not open %1.  Audio Library not saved"), src) << endmsg;
71         }
72 }
73
74 void
75 AudioLibrary::set_tags (string member, vector<string> tags)
76 {
77         sort (tags.begin(), tags.end());
78         tags.erase (unique(tags.begin(), tags.end()), tags.end());
79         
80         const string file_uri(Glib::filename_to_uri (member));
81         
82         lrdf_remove_uri_matches (file_uri.c_str());
83         
84         for (vector<string>::iterator i = tags.begin(); i != tags.end(); ++i) {
85                 lrdf_add_triple (src.c_str(), file_uri.c_str(), TAG, (*i).c_str(), lrdf_literal);
86         }
87 }
88
89 vector<string>
90 AudioLibrary::get_tags (string member)
91 {
92         vector<string> tags;
93         
94         lrdf_statement pattern;
95         pattern.subject = strdup(Glib::filename_to_uri(member).c_str());
96         pattern.predicate = (char*)TAG;
97         pattern.object = 0;
98         pattern.object_type = lrdf_literal;
99         
100         lrdf_statement* matches = lrdf_matches (&pattern);
101         free (pattern.subject);
102         
103         lrdf_statement* current = matches;
104         while (current != 0) {
105                 tags.push_back (current->object);
106                 
107                 current = current->next;
108         }
109         
110         lrdf_free_statements (matches);
111         
112         sort (tags.begin(), tags.end());
113         
114         return tags;
115 }
116
117 void
118 AudioLibrary::search_members_and (vector<string>& members, const vector<string> tags)
119 {
120         lrdf_statement **head;
121         lrdf_statement* pattern = 0;
122         lrdf_statement* old = 0;
123         head = &pattern;
124
125         vector<string>::const_iterator i;
126         for (i = tags.begin(); i != tags.end(); ++i){
127                 pattern = new lrdf_statement;
128                 pattern->subject = (char*)"?";
129                 pattern->predicate = (char*)TAG;
130                 pattern->object = strdup((*i).c_str());
131                 pattern->next = old;
132
133                 old = pattern;
134         }
135
136         if (*head != 0) {
137                 lrdf_uris* ulist = lrdf_match_multi(*head);
138                 for (uint32_t j = 0; ulist && j < ulist->count; ++j) {
139                         members.push_back(Glib::filename_from_uri(ulist->items[j]));
140                 }
141                 lrdf_free_uris(ulist);
142
143             sort(members.begin(), members.end());
144             unique(members.begin(), members.end());
145         }
146
147         // memory clean up
148         pattern = *head;
149         while(pattern){
150                 free(pattern->object);
151                 old = pattern;
152                 pattern = pattern->next;
153                 delete old;
154         }
155 }