#include #include #import // XXX: fix the minimum requirement extern NSString * const kCAFilterNearest; #include #include #include "substrate.h" #define ForSaurik 0 static bool Wildcat_; static CFArrayRef (*$GSSystemCopyCapability)(CFStringRef); static CFArrayRef (*$GSSystemGetCapability)(CFStringRef); static Class $UIFormAssistant; static Class $UIWebBrowserView; @interface NSString (UIKit) - (NSString *) stringByAddingPercentEscapes; @end /* Indirect Delegate {{{ */ @interface IndirectDelegate : NSObject { _transient volatile id delegate_; } - (void) setDelegate:(id)delegate; - (id) initWithDelegate:(id)delegate; @end @implementation IndirectDelegate - (void) setDelegate:(id)delegate { delegate_ = delegate; } - (id) initWithDelegate:(id)delegate { delegate_ = delegate; return self; } - (void) webView:(WebView *)sender didClearWindowObject:(WebScriptObject *)window forFrame:(WebFrame *)frame { if (delegate_ != nil) return [delegate_ webView:sender didClearWindowObject:window forFrame:frame]; } - (void) webView:(WebView *)sender didCommitLoadForFrame:(WebFrame *)frame { if (delegate_ != nil) return [delegate_ webView:sender didCommitLoadForFrame:frame]; } - (void) webView:(WebView *)sender didFailLoadWithError:(NSError *)error forFrame:(WebFrame *)frame { if (delegate_ != nil) return [delegate_ webView:sender didFailLoadWithError:error forFrame:frame]; } - (void) webView:(WebView *)sender didFailProvisionalLoadWithError:(NSError *)error forFrame:(WebFrame *)frame { if (delegate_ != nil) return [delegate_ webView:sender didFailProvisionalLoadWithError:error forFrame:frame]; } - (void) webView:(WebView *)sender didFinishLoadForFrame:(WebFrame *)frame { if (delegate_ != nil) return [delegate_ webView:sender didFinishLoadForFrame:frame]; } - (void) webView:(WebView *)sender didReceiveTitle:(NSString *)title forFrame:(WebFrame *)frame { if (delegate_ != nil) return [delegate_ webView:sender didReceiveTitle:title forFrame:frame]; } - (void) webView:(WebView *)sender didStartProvisionalLoadForFrame:(WebFrame *)frame { if (delegate_ != nil) return [delegate_ webView:sender didStartProvisionalLoadForFrame:frame]; } - (void) webView:(WebView *)sender resource:(id)identifier didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge fromDataSource:(WebDataSource *)source { if (delegate_ != nil) return [delegate_ webView:sender resource:identifier didReceiveAuthenticationChallenge:challenge fromDataSource:source]; } - (NSURLRequest *) webView:(WebView *)sender resource:(id)identifier willSendRequest:(NSURLRequest *)request redirectResponse:(NSURLResponse *)redirectResponse fromDataSource:(WebDataSource *)source { if (delegate_ != nil) return [delegate_ webView:sender resource:identifier willSendRequest:request redirectResponse:redirectResponse fromDataSource:source]; return nil; } - (IMP) methodForSelector:(SEL)sel { if (IMP method = [super methodForSelector:sel]) return method; fprintf(stderr, "methodForSelector:[%s] == NULL\n", sel_getName(sel)); return NULL; } - (BOOL) respondsToSelector:(SEL)sel { if ([super respondsToSelector:sel]) return YES; // XXX: WebThreadCreateNSInvocation returns nil //fprintf(stderr, "[%s]R?%s\n", class_getName(self->isa), sel_getName(sel)); return delegate_ == nil ? NO : [delegate_ respondsToSelector:sel]; } - (NSMethodSignature *) methodSignatureForSelector:(SEL)sel { if (NSMethodSignature *method = [super methodSignatureForSelector:sel]) return method; //fprintf(stderr, "[%s]S?%s\n", class_getName(self->isa), sel_getName(sel)); if (delegate_ != nil) if (NSMethodSignature *sig = [delegate_ methodSignatureForSelector:sel]) return sig; // XXX: I fucking hate Apple so very very bad return [NSMethodSignature signatureWithObjCTypes:"v@:"]; } - (void) forwardInvocation:(NSInvocation *)inv { SEL sel = [inv selector]; if (delegate_ != nil && [delegate_ respondsToSelector:sel]) [inv invokeWithTarget:delegate_]; } @end /* }}} */ @interface WebView (UICaboodle) - (void) setScriptDebugDelegate:(id)delegate; - (void) _setFormDelegate:(id)delegate; - (void) _setUIKitDelegate:(id)delegate; - (void) setWebMailDelegate:(id)delegate; - (void) _setLayoutInterval:(float)interval; @end @implementation WebScriptObject (UICaboodle) - (unsigned) count { id length([self valueForKey:@"length"]); if ([length respondsToSelector:@selector(intValue)]) return [length intValue]; else return 0; } - (id) objectAtIndex:(unsigned)index { return [self webScriptValueAtIndex:index]; } @end @interface BrowserView : UIView { @private UIWebDocumentView *documentView; } @property (nonatomic, retain) UIWebDocumentView *documentView; @end @implementation BrowserView @synthesize documentView; - (void)dealloc { [documentView release]; [super dealloc]; } - (void)layoutSubviews { [super layoutSubviews]; if ([documentView respondsToSelector:@selector(setMinimumSize:)]) [documentView setMinimumSize:documentView.bounds.size]; } @end #define ShowInternals 0 #define LogBrowser 1 #define lprintf(args...) fprintf(stderr, args) @implementation BrowserController #if ShowInternals #include "UICaboodle/UCInternal.h" #endif + (void) _initialize { //[WebView enableWebThread]; WebPreferences *preferences([WebPreferences standardPreferences]); [preferences setCacheModel:WebCacheModelDocumentBrowser]; [preferences setOfflineWebApplicationCacheEnabled:YES]; [WebPreferences _setInitialDefaultTextEncodingToSystemEncoding]; $GSSystemCopyCapability = reinterpret_cast(dlsym(RTLD_DEFAULT, "GSSystemCopyCapability")); $GSSystemGetCapability = reinterpret_cast(dlsym(RTLD_DEFAULT, "GSSystemGetCapability")); $UIFormAssistant = objc_getClass("UIFormAssistant"); $UIWebBrowserView = objc_getClass("UIWebBrowserView"); if ($UIWebBrowserView == nil) { Wildcat_ = false; $UIWebBrowserView = objc_getClass("UIWebDocumentView"); } else { Wildcat_ = true; } } - (void) dealloc { #if LogBrowser NSLog(@"[BrowserController dealloc]"); #endif if (challenge_ != nil) [challenge_ release]; WebThreadLock(); WebView *webview = [document_ webView]; [webview setFrameLoadDelegate:nil]; [webview setResourceLoadDelegate:nil]; [webview setUIDelegate:nil]; [webview setScriptDebugDelegate:nil]; [webview setPolicyDelegate:nil]; /* XXX: these are set by UIWebDocumentView [webview setDownloadDelegate:nil]; [webview _setFormDelegate:nil]; [webview _setUIKitDelegate:nil]; [webview setEditingDelegate:nil];*/ /* XXX: no one sets this, ever [webview setWebMailDelegate:nil];*/ [document_ setDelegate:nil]; [document_ setGestureDelegate:nil]; if ([document_ respondsToSelector:@selector(setFormEditingDelegate:)]) [document_ setFormEditingDelegate:nil]; [document_ setInteractionDelegate:nil]; [indirect_ setDelegate:nil]; //NSNotificationCenter *center = [NSNotificationCenter defaultCenter]; [webview close]; #if RecycleWebViews [document_ removeFromSuperview]; [Documents_ addObject:[document_ autorelease]]; #else [document_ release]; #endif [indirect_ release]; WebThreadUnlock(); [scroller_ setDelegate:nil]; if (button_ != nil) [button_ release]; if (style_ != nil) [style_ release]; if (function_ != nil) [function_ release]; if (finish_ != nil) [finish_ release]; if (closer_ != nil) [closer_ release]; if (special_ != nil) [special_ release]; [scroller_ release]; [indicator_ release]; if (confirm_ != nil) [confirm_ release]; if (sensitive_ != nil) [sensitive_ release]; if (title_ != nil) [title_ release]; [super dealloc]; } - (void) loadURL:(NSURL *)url cachePolicy:(NSURLRequestCachePolicy)policy { [self loadRequest:[NSURLRequest requestWithURL:url cachePolicy:policy timeoutInterval:120.0 ]]; } - (void) loadURL:(NSURL *)url { [self loadURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy]; } - (void) loadRequest:(NSURLRequest *)request { pushed_ = true; error_ = false; WebThreadLock(); [document_ loadRequest:request]; WebThreadUnlock(); } - (void) reloadURL { if (request_ == nil) return; if ([request_ HTTPBody] == nil && [request_ HTTPBodyStream] == nil) [self loadRequest:request_]; else { UIActionSheet *sheet = [[[UIActionSheet alloc] initWithTitle:UCLocalize("RESUBMIT_FORM") buttons:[NSArray arrayWithObjects:UCLocalize("CANCEL"), UCLocalize("SUBMIT"), nil] defaultButtonIndex:0 delegate:self context:@"submit" ] autorelease]; [sheet setAutoresizingMask:(UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight)]; [sheet setNumberOfRows:1]; [sheet popupAlertAnimated:YES]; } } - (WebView *) webView { return [document_ webView]; } - (UIWebDocumentView *) documentView { return document_; } /* XXX: WebThreadLock? */ - (void) _fixScroller:(CGRect)bounds { float extra; if (!editing_ || $UIFormAssistant == nil) extra = 0; else { UIFormAssistant *assistant([$UIFormAssistant sharedFormAssistant]); CGRect peripheral([assistant peripheralFrame]); #if LogBrowser NSLog(@"per:%f", peripheral.size.height); #endif extra = peripheral.size.height; } CGRect subrect([scroller_ frame]); subrect.size.height -= extra; if ([scroller_ respondsToSelector:@selector(setScrollerIndicatorSubrect:)]) [scroller_ setScrollerIndicatorSubrect:subrect]; [document_ setValue:[NSValue valueWithSize:NSMakeSize(subrect.size.width, subrect.size.height)] forGestureAttribute:UIGestureAttributeVisibleSize]; CGSize size(size_); size.height += extra; [scroller_ setContentSize:size]; if ([scroller_ respondsToSelector:@selector(releaseRubberBandIfNecessary)]) [scroller_ releaseRubberBandIfNecessary]; } - (void) fixScroller { CGRect bounds([document_ documentBounds]); #if TrackResize NSLog(@"_fs:(%f,%f+%f,%f)", bounds.origin.x, bounds.origin.y, bounds.size.width, bounds.size.height); #endif [self _fixScroller:bounds]; } - (void) view:(UIView *)sender didSetFrame:(CGRect)frame { size_ = frame.size; #if TrackResize NSLog(@"dsf:(%f,%f+%f,%f)", frame.origin.x, frame.origin.y, frame.size.width, frame.size.height); #endif [self _fixScroller:frame]; } - (void) view:(UIView *)sender didSetFrame:(CGRect)frame oldFrame:(CGRect)old { [self view:sender didSetFrame:frame]; } - (void) pushPage:(UCViewController *)page { [page setDelegate:delegate_]; [[self navigationItem] setTitle:title_]; [[self navigationController] pushViewController:page animated:YES]; } - (void) _pushPage { if (pushed_) return; // WTR: [self autorelease]; pushed_ = true; [[self navigationController] pushViewController:self animated:YES]; } - (void) swapPage:(UCViewController *)page { [page setDelegate:delegate_]; if (pushed_) [[self navigationController] popViewControllerAnimated:NO]; [[self navigationController] pushViewController:page animated:NO]; } - (BOOL) getSpecial:(NSURL *)url swap:(BOOL)swap { #if LogBrowser NSLog(@"getSpecial:%@", url); #endif if (UCViewController *page = [delegate_ pageForURL:url hasTag:NULL]) { if (swap) [self swapPage:page]; else [self pushPage:page]; return true; } else return false; } - (void) formAssistant:(id)sender didBeginEditingFormNode:(id)node { } - (void) formAssistant:(id)sender didEndEditingFormNode:(id)node { [self fixScroller]; } - (void) webViewShow:(WebView *)sender { /* XXX: this is where I cry myself to sleep */ } - (bool) _allowJavaScriptPanel { return true; } - (bool) allowSensitiveRequests { return [self _allowJavaScriptPanel]; } - (void) _promptForSensitive:(NSMutableArray *)array { NSString *name([array objectAtIndex:0]); UIActionSheet *sheet = [[[UIActionSheet alloc] initWithTitle:nil buttons:[NSArray arrayWithObjects:UCLocalize("YES"), UCLocalize("NO"), nil] defaultButtonIndex:0 delegate:indirect_ context:@"sensitive" ] autorelease]; [sheet setAutoresizingMask:(UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight)]; NSString *host(@"XXX"); [sheet setNumberOfRows:1]; [sheet setBodyText:[NSString stringWithFormat:@"The website at %@ is requesting your phone's %@. This is almost certainly for product licensing purposes. Will you allow this?", host, name]]; [sheet popupAlertAnimated:YES]; NSRunLoop *loop([NSRunLoop currentRunLoop]); NSDate *future([NSDate distantFuture]); while (sensitive_ == nil && [loop runMode:NSDefaultRunLoopMode beforeDate:future]); NSNumber *sensitive([sensitive_ autorelease]); sensitive_ = nil; [self autorelease]; [array replaceObjectAtIndex:0 withObject:sensitive]; } - (bool) promptForSensitive:(NSString *)name { if (![self allowSensitiveRequests]) return false; NSMutableArray *array([NSMutableArray arrayWithCapacity:1]); [array addObject:name]; [self performSelectorOnMainThread:@selector(_promptForSensitive:) withObject:array waitUntilDone:YES]; return [[array lastObject] boolValue]; } - (void) webView:(WebView *)sender runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame { if (![self _allowJavaScriptPanel]) return; [self retain]; UIActionSheet *sheet = [[[UIActionSheet alloc] initWithTitle:nil buttons:[NSArray arrayWithObjects:UCLocalize("OK"), nil] defaultButtonIndex:0 delegate:self context:@"alert" ] autorelease]; [sheet setAutoresizingMask:(UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight)]; [sheet setBodyText:message]; [sheet popupAlertAnimated:YES]; } - (BOOL) webView:(WebView *)sender runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame { if (![self _allowJavaScriptPanel]) return NO; [self retain]; UIActionSheet *sheet = [[[UIActionSheet alloc] initWithTitle:nil buttons:[NSArray arrayWithObjects:UCLocalize("OK"), UCLocalize("CANCEL"), nil] defaultButtonIndex:0 delegate:indirect_ context:@"confirm" ] autorelease]; [sheet setAutoresizingMask:(UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight)]; [sheet setNumberOfRows:1]; [sheet setBodyText:message]; [sheet popupAlertAnimated:YES]; NSRunLoop *loop([NSRunLoop currentRunLoop]); NSDate *future([NSDate distantFuture]); while (confirm_ == nil && [loop runMode:NSDefaultRunLoopMode beforeDate:future]); NSNumber *confirm([confirm_ autorelease]); confirm_ = nil; [self autorelease]; return [confirm boolValue]; } - (void) setAutoPopup:(BOOL)popup { popup_ = popup; } - (void) setSpecial:(id)function { if (special_ != nil) [special_ autorelease]; special_ = function == nil ? nil : [function retain]; } - (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]; [self applyRightButton]; } - (void) setButtonTitle:(NSString *)button withStyle:(NSString *)style toFunction:(id)function { if (button_ != nil) [button_ autorelease]; button_ = button == nil ? nil : [button retain]; if (style_ != nil) [style_ autorelease]; style_ = style == nil ? nil : [style retain]; if (function_ != nil) [function_ autorelease]; function_ = function == nil ? nil : [function retain]; [self applyRightButton]; } - (void) setFinishHook:(id)function { if (finish_ != nil) [finish_ autorelease]; finish_ = function == nil ? nil : [function retain]; } - (void) setPopupHook:(id)function { if (closer_ != nil) [closer_ autorelease]; closer_ = function == nil ? nil : [function retain]; } - (void) _openMailToURL:(NSURL *)url { [UIApp openURL:url];// asPanel:YES]; } - (void) webView:(WebView *)sender willBeginEditingFormElement:(id)element { editing_ = true; } - (void) webView:(WebView *)sender didBeginEditingFormElement:(id)element { [self fixScroller]; } - (void) webViewDidEndEditingFormElements:(WebView *)sender { editing_ = false; [self fixScroller]; } - (void) webViewClose:(WebView *)sender { [self close]; } - (void) close { [[self navigationController] dismissModalViewControllerAnimated:YES]; } - (void) webView:(WebView *)sender didClearWindowObject:(WebScriptObject *)window forFrame:(WebFrame *)frame { } - (void) webView:(WebView *)sender unableToImplementPolicyWithError:(NSError *)error frame:(WebFrame *)frame { NSLog(@"err:%@", error); } - (void) webView:(WebView *)sender decidePolicyForNewWindowAction:(NSDictionary *)action request:(NSURLRequest *)request newFrameName:(NSString *)name decisionListener:(id)listener { #if LogBrowser NSLog(@"nwa:%@", name); #endif if (NSURL *url = [request URL]) { if (name == nil) unknown: { if (![self getSpecial:url swap:NO]) { NSString *scheme([[url scheme] lowercaseString]); if ([scheme isEqualToString:@"mailto"]) [self _openMailToURL:url]; else goto use; } } else if ([name isEqualToString:@"_open"]) [delegate_ openURL:url]; else if ([name isEqualToString:@"_popup"]) { NSString *scheme([[url scheme] lowercaseString]); if ([scheme isEqualToString:@"mailto"]) [self _openMailToURL:url]; else { UCNavigationController *navigation([[[UCNavigationController alloc] init] autorelease]); [navigation setHook:indirect_]; UCViewController *page([delegate_ pageForURL:url hasTag:NULL]); if (page == nil) { /* XXX: call createWebViewWithRequest instead? */ BrowserController *browser([[[class_ alloc] init] autorelease]); [browser loadURL:url]; page = browser; } [navigation setDelegate:delegate_]; [page setDelegate:delegate_]; [navigation setViewControllers:[NSArray arrayWithObject:page]]; UIBarButtonItem *closeItem = [[UIBarButtonItem alloc] initWithTitle:UCLocalize("CLOSE") style:UIBarButtonItemStylePlain target:page action:@selector(close) ]; [[page navigationItem] setLeftBarButtonItem:closeItem]; [closeItem release]; [[self navigationController] presentModalViewController:navigation animated:YES]; } } else goto unknown; [listener ignore]; } else use: [listener use]; } - (void) webView:(WebView *)sender decidePolicyForMIMEType:(NSString *)type request:(NSURLRequest *)request frame:(WebFrame *)frame decisionListener:(id)listener { if ([WebView canShowMIMEType:type]) [listener use]; else { // XXX: handle more mime types! [listener ignore]; WebView *webview([document_ webView]); if (frame == [webview mainFrame]) [UIApp openURL:[request URL]]; } } - (void) webView:(WebView *)sender decidePolicyForNavigationAction:(NSDictionary *)action request:(NSURLRequest *)request frame:(WebFrame *)frame decisionListener:(id)listener { if (request == nil) ignore: { [listener ignore]; return; } NSURL *url([request URL]); NSString *host([url host]); if (url == nil) use: { if (!error_ && [frame parentFrame] == nil) { if (request_ != nil) [request_ autorelease]; request_ = [request retain]; #if LogBrowser NSLog(@"dpn:%@", request_); #endif } [listener use]; WebView *webview([document_ webView]); if (frame == [webview mainFrame]) [self _pushPage]; return; } #if LogBrowser else NSLog(@"nav:%@:%@", url, [action description]); #endif const NSArray *capability; if ($GSSystemCopyCapability != NULL) { capability = reinterpret_cast((*$GSSystemCopyCapability)(kGSDisplayIdentifiersCapability)); capability = [capability autorelease]; } else if ($GSSystemGetCapability != NULL) { capability = reinterpret_cast((*$GSSystemGetCapability)(kGSDisplayIdentifiersCapability)); } else capability = nil; NSURL *open(nil); if (capability != nil && ( [url isGoogleMapsURL] && [capability containsObject:@"com.apple.Maps"] && (open = [url mapsURL]) != nil|| [host hasSuffix:@"youtube.com"] && [capability containsObject:@"com.apple.youtube"] && (open = [url youTubeURL]) != nil || [url respondsToSelector:@selector(phobosURL)] && (open = [url phobosURL]) != nil )) { url = open; open: [UIApp openURL:url]; goto ignore; } int store(_not(int)); if (NSURL *itms = [url itmsURL:&store]) { #if LogBrowser NSLog(@"itms#%@#%u#%@", url, store, itms); #endif if (capability != nil && ( store == 1 && [capability containsObject:@"com.apple.MobileStore"] || store == 2 && [capability containsObject:@"com.apple.AppStore"] )) { url = itms; goto open; } } NSString *scheme([[url scheme] lowercaseString]); if ([scheme isEqualToString:@"tel"]) { // XXX: intelligence goto open; } if ([scheme isEqualToString:@"mailto"]) { [self _openMailToURL:url]; goto ignore; } if ([self getSpecial:url swap:YES]) goto ignore; else if ([WebView _canHandleRequest:request]) goto use; else if ([url isSpringboardHandledURL]) goto open; else goto use; } - (void) webView:(WebView *)sender setStatusText:(NSString *)text { //lprintf("Status:%s\n", [text UTF8String]); } - (void) alertSheet:(UIActionSheet *)sheet buttonClicked:(int)button { NSString *context([sheet context]); if ([context isEqualToString:@"alert"]) { [self autorelease]; [sheet dismiss]; } else if ([context isEqualToString:@"confirm"]) { switch (button) { case 1: confirm_ = [NSNumber numberWithBool:YES]; break; case 2: confirm_ = [NSNumber numberWithBool:NO]; break; } [sheet dismiss]; } else if ([context isEqualToString:@"sensitive"]) { switch (button) { case 1: sensitive_ = [NSNumber numberWithBool:YES]; break; case 2: sensitive_ = [NSNumber numberWithBool:NO]; break; } [sheet dismiss]; } else if ([context isEqualToString:@"challenge"]) { id sender([challenge_ sender]); switch (button) { case 1: { NSString *username([[sheet textFieldAtIndex:0] text]); NSString *password([[sheet textFieldAtIndex:1] text]); NSURLCredential *credential([NSURLCredential credentialWithUser:username password:password persistence:NSURLCredentialPersistenceForSession]); [sender useCredential:credential forAuthenticationChallenge:challenge_]; } break; case 2: [sender cancelAuthenticationChallenge:challenge_]; break; _nodefault } [challenge_ release]; challenge_ = nil; [sheet dismiss]; } else if ([context isEqualToString:@"submit"]) { switch (button) { case 1: break; case 2: if (request_ != nil) { WebThreadLock(); [document_ loadRequest:request_]; WebThreadUnlock(); } break; _nodefault } [sheet dismiss]; } } - (void) webView:(WebView *)sender resource:(id)identifier didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge fromDataSource:(WebDataSource *)source { challenge_ = [challenge retain]; NSURLProtectionSpace *space([challenge protectionSpace]); NSString *realm([space realm]); if (realm == nil) realm = @""; UIActionSheet *sheet = [[[UIActionSheet alloc] initWithTitle:realm buttons:[NSArray arrayWithObjects:UCLocalize("LOGIN"), UCLocalize("CANCEL"), nil] defaultButtonIndex:0 delegate:self context:@"challenge" ] autorelease]; [sheet setAutoresizingMask:(UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight)]; [sheet setNumberOfRows:1]; [sheet addTextFieldWithValue:@"" label:UCLocalize("USERNAME")]; [sheet addTextFieldWithValue:@"" label:UCLocalize("PASSWORD")]; UITextField *username([sheet textFieldAtIndex:0]); { UITextInputTraits *traits([username textInputTraits]); [traits setAutocapitalizationType:UITextAutocapitalizationTypeNone]; [traits setAutocorrectionType:UITextAutocorrectionTypeNo]; [traits setKeyboardType:UIKeyboardTypeASCIICapable]; [traits setReturnKeyType:UIReturnKeyNext]; } UITextField *password([sheet textFieldAtIndex:1]); { UITextInputTraits *traits([password textInputTraits]); [traits setAutocapitalizationType:UITextAutocapitalizationTypeNone]; [traits setAutocorrectionType:UITextAutocorrectionTypeNo]; [traits setKeyboardType:UIKeyboardTypeASCIICapable]; // XXX: UIReturnKeyDone [traits setReturnKeyType:UIReturnKeyNext]; [traits setSecureTextEntry:YES]; } [sheet popupAlertAnimated:YES]; } - (NSURLRequest *) webView:(WebView *)sender resource:(id)identifier willSendRequest:(NSURLRequest *)request redirectResponse:(NSURLResponse *)redirectResponse fromDataSource:(WebDataSource *)source { return request; } - (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request windowFeatures:(NSDictionary *)features { //- (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request userGesture:(BOOL)gesture { #if LogBrowser NSLog(@"cwv:%@ (%@): %@", request, title_, features == nil ? @"{}" : [features description]); //NSLog(@"cwv:%@ (%@): %@", request, title_, gesture ? @"Yes" : @"No"); #endif NSNumber *value([features objectForKey:@"width"]); float width(value == nil ? 0 : [value floatValue]); UCNavigationController *navigation(!popup_ ? [self navigationController] : [[[UCNavigationController alloc] init] autorelease]); /* XXX: deal with cydia:// pages */ BrowserController *browser([[[class_ alloc] initWithWidth:width] autorelease]); if (features != nil && popup_) { [navigation setDelegate:delegate_]; [navigation setHook:indirect_]; [browser setDelegate:delegate_]; [browser loadRequest:request]; [navigation setViewControllers:[NSArray arrayWithObject:browser]]; UIBarButtonItem *closeItem = [[UIBarButtonItem alloc] initWithTitle:UCLocalize("CLOSE") style:UIBarButtonItemStylePlain target:browser action:@selector(close) ]; [[browser navigationItem] setLeftBarButtonItem:closeItem]; [closeItem release]; [[self navigationController] presentModalViewController:navigation animated:YES]; } /*else if (request == nil) { [[self navigationItem] setTitle:title_]; [browser setDelegate:delegate_]; [browser retain]; }*/ else { [self pushPage:browser]; [browser loadRequest:request]; } return [browser webView]; } - (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request { return [self webView:sender createWebViewWithRequest:request windowFeatures:nil]; //return [self webView:sender createWebViewWithRequest:request userGesture:YES]; } - (void) webView:(WebView *)sender didReceiveTitle:(NSString *)title forFrame:(WebFrame *)frame { if ([frame parentFrame] != nil) return; title_ = [title retain]; [[self navigationItem] setTitle:title_]; } - (void) webView:(WebView *)sender didStartProvisionalLoadForFrame:(WebFrame *)frame { /*if ([loading_ count] == 0) [self retain];*/ [loading_ addObject:[NSValue valueWithNonretainedObject:frame]]; if ([frame parentFrame] == nil) { [document_ resignFirstResponder]; reloading_ = false; if (title_ != nil) { [title_ release]; title_ = nil; } if (button_ != nil) { [button_ release]; button_ = nil; } if (style_ != nil) { [style_ release]; style_ = nil; } if (function_ != nil) { [function_ release]; function_ = nil; } if (finish_ != nil) { [finish_ release]; finish_ = nil; } if (closer_ != nil) { [closer_ release]; closer_ = nil; } if (special_ != nil) { [special_ release]; special_ = nil; } [[self navigationItem] setTitle:title_]; if (Wildcat_) { CGRect webrect = [scroller_ bounds]; webrect.size.height = 1; [document_ setFrame:webrect]; } if ([scroller_ respondsToSelector:@selector(scrollPointVisibleAtTopLeft:)]) [scroller_ scrollPointVisibleAtTopLeft:CGPointZero]; else [scroller_ scrollRectToVisible:CGRectZero animated:NO]; if ([scroller_ respondsToSelector:@selector(setZoomScale:duration:)]) [scroller_ setZoomScale:1 duration:0]; else if ([scroller_ respondsToSelector:@selector(_setZoomScale:duration:)]) [scroller_ _setZoomScale:1 duration:0]; /*else if ([scroller_ respondsToSelector:@selector(setZoomScale:animated:)]) [scroller_ setZoomScale:1 animated:NO];*/ if (!Wildcat_) { CGRect webrect = [scroller_ bounds]; webrect.size.height = 0; [document_ setFrame:webrect]; } } [self _startLoading]; } - (void) applyRightButton { if ([self isLoading]) { UIBarButtonItem *reloadItem = [[UIBarButtonItem alloc] initWithTitle:@" " style:UIBarButtonItemStylePlain target:self action:@selector(reloadButtonClicked) ]; [[self navigationItem] setRightBarButtonItem:reloadItem]; [[reloadItem view] addSubview:indicator_]; [[self navigationItem] setTitle:UCLocalize("LOADING")]; [reloadItem release]; } else { UIBarButtonItem *reloadItem = [[UIBarButtonItem alloc] initWithTitle:button_ ?: UCLocalize("RELOAD") style:[self rightButtonStyle] target:self action:button_ ? @selector(customButtonClicked) : @selector(reloadButtonClicked) ]; [[self navigationItem] setRightBarButtonItem:reloadItem animated:YES]; [reloadItem release]; } } - (void) _startLoading { [self applyRightButton]; } - (void) _finishLoading { size_t count([loading_ count]); /*if (count == 0) [self autorelease];*/ if (reloading_ || count != 0) return; if (finish_ != nil) [self callFunction:finish_]; [self applyRightButton]; if (![self isLoading]) [[self navigationItem] setTitle:title_]; } - (bool) isLoading { return [loading_ count] != 0; } - (BOOL) webView:(WebView *)sender shouldScrollToPoint:(struct CGPoint)point forFrame:(WebFrame *)frame { return [document_ webView:sender shouldScrollToPoint:point forFrame:frame]; } - (void) webView:(WebView *)sender didReceiveViewportArguments:(id)arguments forFrame:(WebFrame *)frame { return [document_ webView:sender didReceiveViewportArguments:arguments forFrame:frame]; } - (void) webView:(WebView *)sender needsScrollNotifications:(id)notifications forFrame:(WebFrame *)frame { return [document_ webView:sender needsScrollNotifications:notifications forFrame:frame]; } - (void) webView:(WebView *)sender didCommitLoadForFrame:(WebFrame *)frame { [self _pushPage]; return [document_ webView:sender didCommitLoadForFrame:frame]; } - (void) webView:(WebView *)sender didReceiveDocTypeForFrame:(WebFrame *)frame { return [document_ webView:sender didReceiveDocTypeForFrame:frame]; } - (void) webView:(WebView *)sender didFinishLoadForFrame:(WebFrame *)frame { [loading_ removeObject:[NSValue valueWithNonretainedObject:frame]]; [self _finishLoading]; if ([frame parentFrame] == nil) { if (DOMDocument *document = [frame DOMDocument]) if (DOMNodeList *bodies = [document getElementsByTagName:@"body"]) for (DOMHTMLBodyElement *body in bodies) { DOMCSSStyleDeclaration *style([document getComputedStyle:body pseudoElement:nil]); bool colored(false); if (DOMCSSPrimitiveValue *color = static_cast([style getPropertyCSSValue:@"background-color"])) { if ([color primitiveType] == DOM_CSS_RGBCOLOR) { DOMRGBColor *rgb([color getRGBColorValue]); float red([[rgb red] getFloatValue:DOM_CSS_NUMBER]); float green([[rgb green] getFloatValue:DOM_CSS_NUMBER]); float blue([[rgb blue] getFloatValue:DOM_CSS_NUMBER]); float alpha([[rgb alpha] getFloatValue:DOM_CSS_NUMBER]); UIColor *uic(nil); if (red == 0xc7 && green == 0xce && blue == 0xd5) uic = [UIColor pinStripeColor]; else if (alpha != 0) uic = [UIColor colorWithRed:(red / 255) green:(green / 255) blue:(blue / 255) alpha:alpha ]; if (uic != nil) { colored = true; [scroller_ setBackgroundColor:uic]; } } } if (!colored) [scroller_ setBackgroundColor:[UIColor pinStripeColor]]; break; } } return [document_ webView:sender didFinishLoadForFrame:frame]; } - (void) _didFailWithError:(NSError *)error forFrame:(WebFrame *)frame { _trace(); /*if ([frame parentFrame] == nil) [self autorelease];*/ [loading_ removeObject:[NSValue valueWithNonretainedObject:frame]]; [self _finishLoading]; if (reloading_) return; if ([frame parentFrame] == nil) { [self loadURL:[NSURL URLWithString:[NSString stringWithFormat:@"%@?%@", [[NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"error" ofType:@"html"]] absoluteString], [[error localizedDescription] stringByAddingPercentEscapes] ]]]; error_ = true; } } - (void) webView:(WebView *)sender didFailLoadWithError:(NSError *)error forFrame:(WebFrame *)frame { [self _didFailWithError:error forFrame:frame]; if ([document_ respondsToSelector:@selector(webView:didFailLoadWithError:forFrame:)]) [document_ webView:sender didFailLoadWithError:error forFrame:frame]; } - (void) webView:(WebView *)sender didFailProvisionalLoadWithError:(NSError *)error forFrame:(WebFrame *)frame { [self _didFailWithError:error forFrame:frame]; } - (void) webView:(WebView *)sender addMessageToConsole:(NSDictionary *)dictionary { #if LogBrowser || ForSaurik lprintf("Console:%s\n", [[dictionary description] UTF8String]); #endif } - (void) webView:(WebView *)sender didReceiveMessage:(NSDictionary *)dictionary { #if LogBrowser || ForSaurik lprintf("Console:%s\n", [[dictionary description] UTF8String]); #endif if ([document_ respondsToSelector:@selector(webView:didReceiveMessage:)]) [document_ webView:sender didReceiveMessage:dictionary]; } - (void) webView:(id)sender willCloseFrame:(id)frame { if ([document_ respondsToSelector:@selector(webView:willCloseFrame:)]) [document_ webView:sender willCloseFrame:frame]; } - (void) webView:(id)sender didFinishDocumentLoadForFrame:(id)frame { if ([document_ respondsToSelector:@selector(webView:didFinishDocumentLoadForFrame:)]) [document_ webView:sender didFinishDocumentLoadForFrame:frame]; } - (void) webView:(id)sender didFirstLayoutInFrame:(id)frame { if ([document_ respondsToSelector:@selector(webView:didFirstLayoutInFrame:)]) [document_ webView:sender didFirstLayoutInFrame:frame]; } - (void) webViewFormEditedStatusHasChanged:(id)changed { if ([document_ respondsToSelector:@selector(webViewFormEditedStatusHasChanged:)]) [document_ webViewFormEditedStatusHasChanged:changed]; } - (void) webView:(id)sender formStateDidFocusNode:(id)formState { if ([document_ respondsToSelector:@selector(webView:formStateDidFocusNode:)]) [document_ webView:sender formStateDidFocusNode:formState]; } - (void) webView:(id)sender formStateDidBlurNode:(id)formState { if ([document_ respondsToSelector:@selector(webView:formStateDidBlurNode:)]) [document_ webView:sender formStateDidBlurNode:formState]; } /* XXX: fix this stupid include file - (void) webView:(WebView *)sender frame:(WebFrame *)frame exceededDatabaseQuotaForSecurityOrigin:(WebSecurityOrigin *)origin database:(NSString *)database { [origin setQuota:0x500000]; }*/ - (void) webViewDidLayout:(id)sender { [document_ webViewDidLayout:sender]; } - (void) webView:(id)sender didFirstVisuallyNonEmptyLayoutInFrame:(id)frame { [document_ webView:sender didFirstVisuallyNonEmptyLayoutInFrame:frame]; } - (void) webView:(id)sender saveStateToHistoryItem:(id)item forFrame:(id)frame { [document_ webView:sender saveStateToHistoryItem:item forFrame:frame]; } - (void) webView:(id)sender restoreStateFromHistoryItem:(id)item forFrame:(id)frame force:(BOOL)force { [document_ webView:sender restoreStateFromHistoryItem:item forFrame:frame force:force]; } - (void) webView:(id)sender attachRootLayer:(id)layer { [document_ webView:sender attachRootLayer:layer]; } - (id) webView:(id)sender plugInViewWithArguments:(id)arguments fromPlugInPackage:(id)package { return [document_ webView:sender plugInViewWithArguments:arguments fromPlugInPackage:package]; } - (void) webView:(id)sender willShowFullScreenForPlugInView:(id)view { [document_ webView:sender willShowFullScreenForPlugInView:view]; } - (void) webView:(id)sender didHideFullScreenForPlugInView:(id)view { [document_ webView:sender didHideFullScreenForPlugInView:view]; } - (void) webView:(id)sender willAddPlugInView:(id)view { [document_ webView:sender willAddPlugInView:view]; } - (void) webView:(id)sender didObserveDeferredContentChange:(int)change forFrame:(id)frame { [document_ webView:sender didObserveDeferredContentChange:change forFrame:frame]; } - (void) webViewDidPreventDefaultForEvent:(id)sender { [document_ webViewDidPreventDefaultForEvent:sender]; } - (void) _setTileDrawingEnabled:(BOOL)enabled { //[document_ setTileDrawingEnabled:enabled]; } - (void) setViewportWidth:(float)width { width_ = width != 0 ? width : [[self class] defaultWidth]; [document_ setViewportSize:CGSizeMake(width_, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x10]; } - (void) willStartGesturesInView:(UIView *)view forEvent:(GSEventRef)event { [self _setTileDrawingEnabled:NO]; } - (void) didFinishGesturesInView:(UIView *)view forEvent:(GSEventRef)event { [self _setTileDrawingEnabled:YES]; [document_ redrawScaledDocument]; } - (void) scrollerWillStartDragging:(UIScroller *)scroller { [self _setTileDrawingEnabled:NO]; } - (void) scrollerDidEndDragging:(UIScroller *)scroller willSmoothScroll:(BOOL)smooth { [self _setTileDrawingEnabled:YES]; } - (void) scrollerDidEndDragging:(UIScroller *)scroller { [self _setTileDrawingEnabled:YES]; } - (id) initWithWidth:(float)width ofClass:(Class)_class { if ((self = [super init]) != nil) { class_ = _class; loading_ = [[NSMutableSet alloc] initWithCapacity:3]; popup_ = false; BrowserView *actualView = [[BrowserView alloc] initWithFrame:CGRectZero]; [self setView:actualView]; struct CGRect bounds = [[self view] bounds]; scroller_ = [[objc_getClass(Wildcat_ ? "UIScrollView" : "UIScroller") alloc] initWithFrame:bounds]; [[self view] addSubview:scroller_]; [scroller_ setFixedBackgroundPattern:YES]; [scroller_ setBackgroundColor:[UIColor pinStripeColor]]; [scroller_ setScrollingEnabled:YES]; [scroller_ setClipsSubviews:YES]; if (!Wildcat_) [scroller_ setAllowsRubberBanding:YES]; [scroller_ setDelegate:self]; [scroller_ setBounces:YES]; if (!Wildcat_) { [scroller_ setScrollHysteresis:8]; [scroller_ setThumbDetectionEnabled:NO]; [scroller_ setDirectionalScrolling:YES]; //[scroller_ setScrollDecelerationFactor:0.99]; /* 0.989324 */ [scroller_ setEventMode:YES]; } if (Wildcat_) { UIScrollView *scroller((UIScrollView *)scroller_); //[scroller setDirectionalLockEnabled:NO]; [scroller setDelaysContentTouches:NO]; //[scroller setScrollsToTop:NO]; //[scroller setCanCancelContentTouches:NO]; } [scroller_ setShowBackgroundShadow:NO]; /* YES */ //[scroller_ setAllowsRubberBanding:YES]; /* Vertical */ if (!Wildcat_) [scroller_ setAdjustForContentSizeChange:YES]; /* NO */ CGRect webrect = [scroller_ bounds]; webrect.size.height = 0; WebView *webview; WebThreadLock(); #if RecycleWebViews document_ = [Documents_ lastObject]; if (document_ != nil) { document_ = [document_ retain]; webview = [document_ webView]; [Documents_ removeLastObject]; [document_ setFrame:webrect]; } else { #else if (true) { #endif document_ = [[$UIWebBrowserView alloc] initWithFrame:webrect]; webview = [document_ webView]; // XXX: this is terribly (too?) expensive //[document_ setDrawsBackground:NO]; [webview setPreferencesIdentifier:@"Cydia"]; [document_ setTileSize:CGSizeMake(webrect.size.width, 500)]; if ([document_ respondsToSelector:@selector(enableReachability)]) [document_ enableReachability]; if ([document_ respondsToSelector:@selector(setAllowsMessaging:)]) [document_ setAllowsMessaging:YES]; if ([document_ respondsToSelector:@selector(useSelectionAssistantWithMode:)]) [document_ useSelectionAssistantWithMode:0]; [document_ setTilingEnabled:YES]; [document_ setDrawsGrid:NO]; [document_ setLogsTilingChanges:NO]; [document_ setTileMinificationFilter:kCAFilterNearest]; if ([document_ respondsToSelector:@selector(setDataDetectorTypes:)]) /* XXX: abstractify */ [document_ setDataDetectorTypes:0x80000000]; else [document_ setDetectsPhoneNumbers:NO]; [document_ setAutoresizes:YES]; [document_ setMinimumScale:0.25f forDocumentTypes:0x10]; [document_ setMaximumScale:5.00f forDocumentTypes:0x10]; [document_ setInitialScale:UIWebViewScalesToFitScale forDocumentTypes:0x10]; //[document_ setViewportSize:CGSizeMake(980, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x10]; [document_ setViewportSize:CGSizeMake(320, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x2]; [document_ setMinimumScale:1.00f forDocumentTypes:0x8]; [document_ setInitialScale:UIWebViewScalesToFitScale forDocumentTypes:0x8]; [document_ setViewportSize:CGSizeMake(320, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x8]; [document_ _setDocumentType:0x4]; if ([document_ respondsToSelector:@selector(setZoomsFocusedFormControl:)]) [document_ setZoomsFocusedFormControl:YES]; [document_ setContentsPosition:7]; [document_ setEnabledGestures:0xa]; [document_ setValue:[NSNumber numberWithBool:YES] forGestureAttribute:UIGestureAttributeIsZoomRubberBandEnabled]; [document_ setValue:[NSNumber numberWithBool:YES] forGestureAttribute:UIGestureAttributeUpdatesScroller]; [document_ setSmoothsFonts:YES]; [document_ setAllowsImageSheet:YES]; [webview _setUsesLoaderCache:YES]; [webview setGroupName:@"CydiaGroup"]; WebPreferences *preferences([webview preferences]); if ([webview respondsToSelector:@selector(_setLayoutInterval:)]) [webview _setLayoutInterval:0]; else [preferences _setLayoutInterval:0]; } actualView.documentView = document_; [actualView release]; [self setViewportWidth:width]; [document_ setDelegate:self]; [document_ setGestureDelegate:self]; if ([document_ respondsToSelector:@selector(setFormEditingDelegate:)]) [document_ setFormEditingDelegate:self]; [document_ setInteractionDelegate:self]; [scroller_ addSubview:document_]; //NSNotificationCenter *center = [NSNotificationCenter defaultCenter]; indirect_ = [[IndirectDelegate alloc] initWithDelegate:self]; [webview setFrameLoadDelegate:indirect_]; [webview setPolicyDelegate:indirect_]; [webview setResourceLoadDelegate:indirect_]; [webview setUIDelegate:indirect_]; /* XXX: do not turn this on under penalty of extreme pain */ [webview setScriptDebugDelegate:nil]; WebThreadUnlock(); CGSize indsize = [UIProgressIndicator defaultSizeForStyle:UIProgressIndicatorStyleMediumWhite]; indicator_ = [[UIProgressIndicator alloc] initWithFrame:CGRectMake(15, 5, indsize.width, indsize.height)]; [indicator_ setStyle:UIProgressIndicatorStyleMediumWhite]; [indicator_ startAnimation]; [scroller_ setAutoresizingMask:(UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight)]; [indicator_ setAutoresizingMask:UIViewAutoresizingFlexibleLeftMargin]; [document_ setAutoresizingMask:UIViewAutoresizingFlexibleWidth]; /*UIWebView *test([[[UIWebView alloc] initWithFrame:[[self view] bounds]] autorelease]); [test loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.saurik.com/"]]]; [[self view] addSubview:test];*/ } return self; } - (id) initWithWidth:(float)width { return [self initWithWidth:width ofClass:[self class]]; } - (id) init { return [self initWithWidth:0]; } - (NSString *) stringByEvaluatingJavaScriptFromString:(NSString *)script { WebThreadLock(); WebView *webview([document_ webView]); NSString *string([webview stringByEvaluatingJavaScriptFromString:script]); WebThreadUnlock(); return string; } - (void) callFunction:(WebScriptObject *)function { WebThreadLock(); WebView *webview([document_ webView]); WebFrame *frame([webview mainFrame]); id _private(MSHookIvar(webview, "_private")); WebCore::Page *page(_private == nil ? NULL : MSHookIvar(_private, "page")); WebCore::Settings *settings(page == NULL ? NULL : page->settings()); bool no; if (settings == NULL) no = 0; else { no = settings->JavaScriptCanOpenWindowsAutomatically(); settings->setJavaScriptCanOpenWindowsAutomatically(true); } if (UIWindow *window = [[self view] window]) if (UIResponder *responder = [window firstResponder]) [responder resignFirstResponder]; JSObjectRef object([function JSObject]); JSGlobalContextRef context([frame globalContext]); JSObjectCallAsFunction(context, object, NULL, 0, NULL, NULL); if (settings != NULL) settings->setJavaScriptCanOpenWindowsAutomatically(no); WebThreadUnlock(); } - (void) didDismissModalViewController { if (closer_ != nil) [self callFunction:closer_]; } - (void) reloadButtonClicked { reloading_ = true; [self reloadURL]; } - (void) customButtonClicked { #if !AlwaysReload if (function_ != nil) [self callFunction:function_]; else #endif [self reloadButtonClicked]; } - (UINavigationButtonStyle) rightButtonStyle { if (style_ == nil) normal: return UINavigationButtonStyleNormal; else if ([style_ isEqualToString:@"Normal"]) return UINavigationButtonStyleNormal; else if ([style_ isEqualToString:@"Back"]) return UINavigationButtonStyleBack; else if ([style_ isEqualToString:@"Highlighted"]) return UINavigationButtonStyleHighlighted; else if ([style_ isEqualToString:@"Destructive"]) return UINavigationButtonStyleDestructive; else goto normal; } - (void) setPageActive:(BOOL)active { if (!active) [indicator_ removeFromSuperview]; else [[[[self navigationItem] rightBarButtonItem] view] addSubview:indicator_]; } - (void) resetViewAnimated:(BOOL)animated { } - (void) setPushed:(bool)pushed { pushed_ = pushed; } + (float) defaultWidth { return 980; } @end