-var _assert = function (expr) {
+/* XXX: this message is ultra-lame */
+var _assert = function (expr, value) {
     if (!expr) {
-        var message = "_assert(" + expr + ")";
-        alert(message);
+        var message = "_assert(" + value + ")";
+        console.log(message);
         throw message;
     }
 }
+
 // Compatibility {{{
 if (typeof Array.prototype.push != "function")
     Array.prototype.push = function (value) {
     if (this.magic_ != $.prototype.magic_)
         return new $(arg);
 
+    if (arg == null)
+        arg = [];
+
     var type = $.type(arg);
 
     if (type == "function")
         $.ready(arg);
     else if (type == "string") {
-        if (doc == undefined)
+        if (typeof doc == 'undefined')
             doc = document;
         if (arg.charAt(0) == '#') {
-            var node = doc.getElementById(arg.substring(1));
-            return $(node == null ? [] : [node]);
-        } else if (arg.charAt(0) == '.') {
-            var nodes = doc.getElementsByClassName(arg.substring(1));
-            return $(nodes == null ? [] : nodes);
-        } else
+            /* XXX: this is somewhat incorrect-a-porter */
+            var element = doc.getElementById(arg.substring(1));
+            return $(element == null ? [] : [element]);
+        } else if (arg.charAt(0) == '.')
+            return $(doc.getElementsByClassName(arg.substring(1)));
+        else
             return $([doc]).descendants(arg);
-    } else {
-        _assert(doc == undefined);
-        this.set($.array(arg));
+    } else if (typeof arg.length != 'undefined') {
+        _assert(typeof doc == 'undefined', "non-query with document to $");
+        this.set(arg);
         return this;
-    }
+    } else _assert(false, "unknown argument to $: " + typeof arg);
 };
 
 $.xml = function (value) {
 $.type = function (value) {
     var type = typeof value;
 
-    if (
-        type == "function" &&
-        value.toString != null &&
-        value.toString().substring(0, 8) == "[object "
-    )
-        return "object";
-    else
-        return type;
+    if ((type == "function" || type == "object") && value.toString != null) {
+        var string = value.toString();
+        if (string.substring(0, 8) == "[object ")
+            return string.substring(8, string.length - 1);
+    }
+
+    return type;
 };
 
 (function () {
 $.array = function (values) {
     if (values.constructor == Array)
         return values;
+    _assert(typeof values.length != 'undefined', "$.array on underlying non-array");
     var array = [];
     for (var i = 0; i != values.length; ++i)
         array.push(values[i]);
     }
 };
 
+$.reclass = function (_class) {
+    return new RegExp('(\\s|^)' + _class + '(\\s|$)');
+};
+
 $.prototype = {
     magic_: 2041085062,
 
     add: function (nodes) {
-        Array.prototype.push.apply(this, nodes);
+        Array.prototype.push.apply(this, $.array(nodes));
+    },
+
+    at: function (name, value) {
+        if (typeof value == 'undefined')
+            return $.map(this, function (node) {
+                return node.getAttribute(name);
+            });
+        else if (value == null)
+            $.each(this, function (node) {
+                node.removeAttribute();
+            });
+        else
+            $.each(this, function (node) {
+                node.setAttribute(name, value);
+            });
     },
 
     set: function (nodes) {
         this.add(nodes);
     },
 
+    /* XXX: verify arg3 overflow */
+    each: function (_function, arg0, arg1, arg2) {
+        $.each(this, function (node) {
+            _function($([node]), arg0, arg1, arg2);
+        });
+    },
+
     css: function (name, value) {
         $.each(this, function (node) {
             node.style[name] = value;
         });
     },
 
+    addClass: function (_class) {
+        $.each(this, function (node) {
+            if (!$([node]).hasClass(_class)[0])
+                node.className += " " + _class;
+        });
+    },
+
+    blur: function () {
+        $.each(this, function (node) {
+            node.blur();
+        });
+    },
+
+    focus: function () {
+        $.each(this, function (node) {
+            node.focus();
+        });
+    },
+
+    removeClass: function (_class) {
+        $.each(this, function (node) {
+            node.className = node.className.replace($.reclass(_class), ' ');
+        });
+    },
+
+    hasClass: function (_class) {
+        return $.map(this, function (node) {
+            return node.className.match($.reclass(_class));
+        });
+    },
+
     append: function (html) {
-        $.each(this, $.type(html) == "string" ? function (node) {
+        $.each(this, function (node) {
             var doc = $.document(node);
 
             // XXX: implement wrapper system
                 var child = div.childNodes[0];
                 node.appendChild(child);
             }
-        } : function (node) {
-            $.each(html, function (value) {
-                node.appendChild(value);
-            });
         });
     },
 
-    clone: function (deep) {
-        return $($.map(this, function (node) {
-            return node.cloneNode(deep);
-        }));
-    },
-
     descendants: function (expression) {
         var descendants = $([]);
 
         $.each(this, function (node) {
-            descendants.add(node.getElementsByTagName(expression));
+            var nodes = node.getElementsByTagName(expression);
+            descendants.add(nodes);
         });
 
         return descendants;
         $.each(this, function (node) {
             node.parentNode.removeChild(node);
         });
-    },
-
-    parent: function () {
-        return $($.map(this, function (node) {
-            return node.parentNode;
-        }));
-    },
-
-    xpath: function (expression) {
-        var value = $([]);
-
-        $.each(this, function (node) {
-            var doc = $.document(node);
-            var result = doc.evaluate(expression, node, null, XPathResult.ANY_TYPE, null);
-
-            if (result.resultType == XPathResult.UNORDERED_NODE_ITERATOR_TYPE)
-                for (;;) {
-                    var next = result.iterateNext();
-                    if (next == null)
-                        break;
-                    value.add([next]);
-                }
-        });
-
-        return value;
     }
 };
 
 
 // XXX: document.all?
 $.all = function (doc) {
-    if (doc == undefined)
+    if (typeof doc == 'undefined')
         doc = document;
     return $(doc.getElementsByTagName("*"));
 };
 $.inject = function (a, b) {
     if ($.type(a) == "string") {
         $.prototype[a] = function (value) {
-            if (value == undefined)
+            if (typeof value == 'undefined')
                 return $.map(this, function (node) {
                     return b.get(node);
                 });
 };
 
 $.inject({
+    _default: {
+        get: function (node) {
+            return node.style.defaultValue;
+        },
+        set: function (node, value) {
+            node.style.defaultValue = value;
+        }
+    },
+
     display: {
         get: function (node) {
             return node.style.display;
         }
     },
 
-    id: {
+    name: {
         get: function (node) {
-            return node.id;
+            return node.name;
         },
         set: function (node, value) {
-            node.id = value;
+            node.name = value;
+        }
+    },
+
+    parent: {
+        get: function (node) {
+            return node.parentNode;
         }
     },
 
         }
     },
 
+    type: {
+        get: function (node) {
+            return node.localName;
+        }
+    },
+
     value: {
         get: function (node) {
             return node.value;
         },
         set: function (node, value) {
-            node.value = value;
+            // XXX: do I really need this?
+            if (true || node.localName != "select")
+                node.value = value;
+            else {
+                var options = node.options;
+                for (var i = 0, e = options.length; i != e; ++i)
+                    if (options[i].value == value) {
+                        if (node.selectedIndex != i)
+                            node.selectedIndex = i;
+                        break;
+                    }
+            }
+        }
+    },
+
+    width: {
+        get: function (node) {
+            return node.offsetWidth;
         }
     }
 });
 
+// Query String Parsing {{{
+$.query = function () {
+    var args = {};
+
+    var search = location.search;
+    if (search != null) {
+        _assert(search[0] == "?", "query string without ?");
+
+        var values = search.substring(1).split("&");
+        for (var index in values) {
+            var value = values[index]
+            var equal = value.indexOf("=");
+            var name;
+
+            if (equal == -1) {
+                name = value;
+                value = null;
+            } else {
+                name = value.substring(0, equal);
+                value = value.substring(equal + 1);
+                value = decodeURIComponent(value);
+            }
+
+            name = decodeURIComponent(name);
+            if (typeof args[name] == "undefined")
+                args[name] = [];
+            if (value != null)
+                args[name].push(value);
+        }
+    }
+
+    return args;
+};
+// }}}
 // Event Registration {{{
 // XXX: unable to remove registration
 $.prototype.event = function (event, _function) {
     "click", "load", "submit"
 ], function (event) {
     $.prototype[event] = function (_function) {
-        if (_function == undefined)
-            _assert(false);
+        if (typeof _function == 'undefined')
+            _assert(false, "undefined function to $.[event]");
         else
             this.event(event, _function);
     };
 
     vertical-align: baseline;
 }
 
+sup {
+    font-size: smaller;
+    margin-top: -6px;
+    position: relative;
+    top: -6px;
+}
+
 select {
     border: 1px solid #999999;
 }
     letter-spacing: -2px;
 }
 
+.default {
+    color: #aaaabb;
+}
+
 /* #toolbar {{{ */
 dialog > toolbar {
     background: url(toolbar.png) #6d84a2 repeat-x;
     margin: 9px;
 }
 
+dialog > panel > input[type="submit"] {
+    /*-webkit-border-image: url(whiteButton.png) 0 12 0 12;
+    -webkit-border-radius: 0;
+    border-width: 0px 12px;*/
+    border: none;
+    color: #000000;
+    display: block;
+    font-size: 20px;
+    font-weight: bold;
+    margin: 9px;
+    height: 44px;
+    padding: 10px;
+    text-align: center;
+    width: 302px;
+}
+
 dialog > panel > label {
     display: block;
     margin: 13px 0 -4px 27px;
     margin-bottom: 0;
 }
 
-dialog > fieldset > a {
-    background: 295px 13px no-repeat url(listArrow.png);
-}
-
-dialog > panel > fieldset > a {
-    background: 275px 13px no-repeat url(listArrow.png);
-}
-
 fieldset > a {
     color: inherit;
     display: block;
 }
 
-dialog > fieldset > div > select {
+fieldset > textarea,
+fieldset > div > input,
+fieldset > div > select {
     background: none;
-    margin: -13px -17px -13px 86px;
-    border-left: 0;
-    border-right: 0;
-    height: 44px;
-    width: 217px;
+    -webkit-box-shadow: none;
+    -webkit-appearance: none;
 }
 
-dialog > panel > fieldset > div > select {
-    margin: -5px -10px -5px 86px;
-    -webkit-border-radius: 5px;
-    width: 190px;
+fieldset > a,
+fieldset > div > select {
+    background: no-repeat url(chevron.png);
+    background-position-y: 13px;
 }
 
-fieldset > textarea,
-fieldset > div > input {
-    background: none;
+dialog > fieldset > a {
+    background-position-x: 295px;
+}
+
+dialog > panel > fieldset > a {
+    background-position-x: 275px;
+}
+
+dialog > fieldset > div > select {
+    background-position-x: 192px;
+}
+
+dialog > panel > fieldset > div > select {
+    background-position-x: 172px;
 }
 
 fieldset > textarea,
     font-size: 16px;
 }
 
+fieldset > div > select,
 fieldset > div > input {
     border: none;
-    height: 45px;
-    margin: -13px -18px;
-    padding: 13px 10px 0 111px;
-}
-
-fieldset > textarea {
-    padding: 10px;
-    width: 320px;
+    height: 44px;
+    margin: -13px -17px -13px 86px;
 }
 
+dialog > panel > fieldset > div > select,
 dialog > panel > fieldset > div > input {
-    width: 302px;
+    width: 187px;
 }
 
+dialog > fieldset > div > select,
 dialog > fieldset > div > input {
-    width: 320px;
+    width: 207px;
 }
 
-fieldset > div > input[type="submit"] {
-    border-width: 0 12px;
-    color: #000000;
-    display: block;
-    font-size: 20px;
-    font-weight: bold;
+fieldset > div > input {
+    padding: 13px 7px;
+}
+
+fieldset > textarea {
     padding: 10px;
-    text-align: center;
-    -webkit-border-image: url(whiteButton.png) 0 12 0 12;
+    width: 320px;
 }
 
 fieldset > a > label,
 
 
 #include <WebKit/WebFrame.h>
 #include <WebKit/WebPolicyDelegate.h>
+#include <WebKit/WebPreferences.h>
 #include <WebKit/WebScriptObject.h>
 
 #import <WebKit/WebView.h>
 #import <WebKit/WebView-WebPrivate.h>
 
+#include <WebCore/Page.h>
+#include <WebCore/Settings.h>
+
 #import <JavaScriptCore/JavaScriptCore.h>
 
 #include <sstream>
 
 #import "BrowserView.h"
 #import "ResetView.h"
+
+#import "substrate.h"
 /* }}} */
 
 //#define _finline __attribute__((force_inline))
 
 #define lprintf(args...) fprintf(stderr, args)
 
-#define ForRelease 0
+#define ForRelease 1
 #define ForSaurik (1 && !ForRelease)
 #define RecycleWebViews 0
 #define AlwaysReload (1 && !ForRelease)
     return @"Cancel";
 }
 
-- (NSString *) _rightButtonTitle {
+- (id) _rightButtonTitle {
     return issues_ == nil ? @"Confirm" : nil;
 }
 
 }
 #endif
 
-- (NSString *) _rightButtonTitle {
+- (id) _rightButtonTitle {
     int count = [buttons_ count];
     return count == 0 ? nil : count != 1 ? @"Modify" : [buttons_ objectAtIndex:0];
 }
     return [[list_ table] isRowDeletionEnabled] ? @"Add" : nil;
 }
 
-- (NSString *) rightButtonTitle {
+- (id) rightButtonTitle {
     return [[list_ table] isRowDeletionEnabled] ? @"Done" : @"Edit";
 }
 
     return @"Packages";
 }
 
-- (NSString *) rightButtonTitle {
+- (id) rightButtonTitle {
     return Role_ != nil && [Role_ isEqualToString:@"Developer"] ? nil : expert_ ? @"Expert" : @"Simple";
 }
 
 }
 
 #if !AlwaysReload
-- (NSString *) _rightButtonTitle {
+- (id) _rightButtonTitle {
     return nil;
 }
 #endif
 + (NSString *) webScriptNameForSelector:(SEL)selector {
     if (selector == @selector(getPackageById:))
         return @"getPackageById";
+    else if (selector == @selector(setButtonImage:withStyle:toFunction:))
+        return @"setButtonImage";
     else if (selector == @selector(setButtonTitle:withStyle:toFunction:))
         return @"setButtonTitle";
     else if (selector == @selector(supports:))
     return [[Database sharedInstance] packageWithName:id];
 }
 
+- (void) setButtonImage:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
+    if (button_ != nil)
+        [button_ autorelease];
+    button_ = button == nil ? nil : [[UIImage imageWithData:[NSData dataWithContentsOfURL:[NSURL URLWithString:button]]] retain];
+
+    if (style_ != nil)
+        [style_ autorelease];
+    style_ = style == nil ? nil : [style retain];
+
+    if (function_ != nil)
+        [function_ autorelease];
+    function_ = function == nil ? nil : [function retain];
+}
+
 - (void) setButtonTitle:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
     if (button_ != nil)
         [button_ autorelease];
             webview = [webview_ webView];
 
             // XXX: this is terribly (too?) expensive
-            [webview_ setDrawsBackground:NO];
+            //[webview_ setDrawsBackground:NO];
+            [webview setPreferencesIdentifier:@"Cydia"];
 
             [webview_ setTileSize:CGSizeMake(webrect.size.width, 500)];
 
         reloading_ = true;
         [self reloadURL];
     } else {
+        WebView *webview([webview_ webView]);
+        WebFrame *frame([webview mainFrame]);
+
+        id _private(MSHookIvar<id>(webview, "_private"));
+        WebCore::Page *page(_private == nil ? NULL : MSHookIvar<WebCore::Page *>(_private, "page"));
+        WebCore::Settings *settings(page == NULL ? NULL : page->settings());
+
+        bool no;
+        if (settings == NULL)
+            no = 0;
+        else {
+            no = settings->JavaScriptCanOpenWindowsAutomatically();
+            settings->setJavaScriptCanOpenWindowsAutomatically(true);
+        }
+
         [delegate_ clearFirstResponder];
         JSObjectRef function([function_ JSObject]);
-        JSGlobalContextRef context([[[webview_ webView] mainFrame] globalContext]);
+        JSGlobalContextRef context([frame globalContext]);
         JSObjectCallAsFunction(context, function, NULL, 0, NULL, NULL);
+
+        if (settings != NULL)
+            settings->setJavaScriptCanOpenWindowsAutomatically(no);
     }
 }
 
-- (NSString *) _rightButtonTitle {
+- (id) _rightButtonTitle {
     return button_ != nil ? button_ : @"Reload";
 }
 
-- (NSString *) rightButtonTitle {
+- (id) rightButtonTitle {
     return [self _loading] ? @"" : [self _rightButtonTitle];
 }
 
     return @"Sections";
 }
 
-- (NSString *) rightButtonTitle {
+- (id) rightButtonTitle {
     return [sections_ count] == 0 ? nil : editing_ ? @"Done" : @"Edit";
 }
 
     return [(CYBook *)book_ updating] ? nil : @"Refresh";
 }
 
-- (NSString *) rightButtonTitle {
+- (id) rightButtonTitle {
     return upgrades_ == 0 ? nil : [NSString stringWithFormat:@"Upgrade (%u)", upgrades_];
 }