* Fixed bug: Discrete control lists are generally not played back
[ardour.git] / libs / pbd / filesystem.cc
index 2412b7676119db816139107a6185c45da3befff7..84d18cfcd242fc45b54b4bccc620ee7c7a1c6de2 100644 (file)
@@ -58,6 +58,30 @@ path::operator/=(const char* rhs)
        return *this;
 }
 
+string
+path::leaf () const
+{
+       return Glib::path_get_basename(m_path);
+}
+
+path
+path::branch_path () const
+{
+       string dir = Glib::path_get_dirname (m_path);
+
+       /*
+        * glib returns "." to signify that the path
+        * has no directory components(branch path)
+        * whereas boost::filesystem returns an empty
+        * string
+        */
+       if(dir == ".")
+       {
+               return "";
+       }
+       return dir;
+}
+
 bool
 exists (const path & p)
 {
@@ -112,11 +136,23 @@ remove(const path & p)
        return true;
 }
 
+void
+rename (const path & from_path, const path & to_path)
+{
+       // g_rename is a macro that evaluates to ::rename on
+       // POSIX systems, without the global namespace qualifier
+       // it would evaluate to a recursive call(if it compiled)
+       if ( ::g_rename( from_path.to_string().c_str(),
+                               to_path.to_string().c_str() ) == -1 )
+       {
+               throw filesystem_error(g_strerror(errno), errno);
+       }
+}
+
+// XXX character encoding.
 void
 copy_file(const path & from_path, const path & to_path)
 {
-       // this implementation could use mucho memory
-       // for big files.
        std::ifstream in(from_path.to_string().c_str());
        std::ofstream out(to_path.to_string().c_str());
        
@@ -128,17 +164,36 @@ copy_file(const path & from_path, const path & to_path)
        out << in.rdbuf();
        
        if (!in || !out) {
+               remove (to_path);
                throw filesystem_error(string_compose(_("Could not copy existing file %1 to %2"),
                                        from_path.to_string(), to_path.to_string()));
-               remove (to_path);
        }
 }
 
 string
 basename (const path & p)
 {
-       // I'm not sure if this works quite the same as boost::filesystem::basename
-       return Glib::path_get_basename (p.to_string ());
+       string base(p.leaf());
+
+       string::size_type n = base.rfind ('.');
+
+       return base.substr (0, n);
+}
+
+string
+extension (const path & p)
+{
+       string base(p.leaf());
+
+       string::size_type n = base.rfind ('.');
+
+       if (n != string::npos)
+       {
+               return base.substr(n);
+       }
+
+       return string();
+
 }
 
 } // namespace sys