Switch to side-by-side view

--- a/src/utils/pathut.cpp
+++ b/src/utils/pathut.cpp
@@ -52,6 +52,7 @@
 #include <stack>
 #include <set>
 #include <vector>
+#include <regex>
 
 #include "pathut.h"
 #include "smallut.h"
@@ -761,6 +762,96 @@
     return url.find("file://") == 0;
 }
 
+static std::regex
+re_uriparse("^(([^:/?#]+):)?(//([^/?#]*))?([^?#]*)(\\?([^#]*))?(#(.*))?",
+            std::regex::extended);
+
+ParsedUri::ParsedUri(std::string uri)
+{
+    std::smatch mr;
+    parsed = regex_match(uri, mr, re_uriparse);
+    if (!parsed)
+        return;
+    // cf http://www.ietf.org/rfc/rfc2396.txt
+    // scheme    = $2
+    // authority = $4
+    // path      = $5
+    // query     = $7
+    // fragment  = $9
+    if (mr[2].matched) {
+        scheme = mr[2].str();
+    }
+    if (mr[4].matched) {
+        string auth = mr[4].str();
+        // user:pass@host, user@host
+        string::size_type at = auth.find_first_of('@');
+        if (at != string::npos) {
+            host = auth.substr(at+1);
+            string::size_type colon = auth.find_first_of(':');
+            if (colon != string::npos && colon < at) {
+                user = auth.substr(0, colon);
+                pass = auth.substr(colon+1, at-colon-1);
+            } else {
+                user = auth.substr(0, at);
+            }
+        } else {
+            host.swap(auth);
+        }
+        string::size_type pc = host.find_first_of(':');
+        if (pc != string::npos) {
+            port = host.substr(pc+1);
+            host = host.substr(0, pc);
+        }
+    }
+    if (mr[5].matched) {
+        path = mr[5].str();
+    }
+    if (mr[7].matched) {
+        query = mr[7].str();
+        string::size_type pos=0, amp, eq;
+        string nm, val;
+        for (;;) {
+            nm.clear();
+            val.clear();
+            amp = query.find_first_of('&', pos);
+            //cerr << "pos " << pos << " amp " << amp << endl;
+            if (amp > pos && amp != string::npos) {
+                eq = query.find_first_of('=', pos);
+                if (eq > amp || eq == string::npos) {
+                    nm = query.substr(pos, amp-pos);
+                } else {
+                    nm = query.substr(pos, eq-pos);
+                    val = query.substr(eq+1, amp-eq-1);
+                }
+                pos = amp + 1;
+            } else if (amp == string::npos) {
+                if (pos < query.size()-1) {
+                    eq = query.find_first_of('=', pos);
+                    if (eq == string::npos) {
+                        nm = query.substr(pos);
+                    } else {
+                        nm = query.substr(pos, eq-pos);
+                        val = query.substr(eq+1);
+                    }
+                }
+                pos = query.size()-1;
+            } else {
+                pos++;
+            }
+            if (!nm.empty()) {
+                parsedquery.push_back(pair<string,string>(nm, val));
+            }
+            if (pos >= query.size()-1) {
+                break;
+            }
+        }
+        
+    }
+    if (mr[9].matched) {
+        fragment = mr[9].str();
+    }
+}
+
 bool readdir(const string& dir, string& reason, set<string>& entries)
 {
     struct stat st;