1 #include <BrowserView.h>
3 #include <WebCore/WebCoreThread.h>
5 /* Indirect Delegate {{{ */
6 @interface IndirectDelegate : NSObject {
7 _transient volatile id delegate_;
10 - (void) setDelegate:(id)delegate;
11 - (id) initWithDelegate:(id)delegate;
14 @implementation IndirectDelegate
16 - (void) setDelegate:(id)delegate {
20 - (id) initWithDelegate:(id)delegate {
25 - (void) webView:(WebView *)sender didClearWindowObject:(WebScriptObject *)window forFrame:(WebFrame *)frame {
27 return [delegate_ webView:sender didClearWindowObject:window forFrame:frame];
30 - (void) webView:(WebView *)sender didCommitLoadForFrame:(WebFrame *)frame {
32 return [delegate_ webView:sender didCommitLoadForFrame:frame];
35 - (void) webView:(WebView *)sender didFailLoadWithError:(NSError *)error forFrame:(WebFrame *)frame {
37 return [delegate_ webView:sender didFailLoadWithError:error forFrame:frame];
40 - (void) webView:(WebView *)sender didFailProvisionalLoadWithError:(NSError *)error forFrame:(WebFrame *)frame {
42 return [delegate_ webView:sender didFailProvisionalLoadWithError:error forFrame:frame];
45 - (void) webView:(WebView *)sender didFinishLoadForFrame:(WebFrame *)frame {
47 return [delegate_ webView:sender didFinishLoadForFrame:frame];
50 - (void) webView:(WebView *)sender didReceiveTitle:(NSString *)title forFrame:(WebFrame *)frame {
52 return [delegate_ webView:sender didReceiveTitle:title forFrame:frame];
55 - (void) webView:(WebView *)sender didStartProvisionalLoadForFrame:(WebFrame *)frame {
57 return [delegate_ webView:sender didStartProvisionalLoadForFrame:frame];
60 - (void) webView:(WebView *)sender resource:(id)identifier didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge fromDataSource:(WebDataSource *)source {
62 return [delegate_ webView:sender resource:identifier didReceiveAuthenticationChallenge:challenge fromDataSource:source];
65 - (NSURLRequest *) webView:(WebView *)sender resource:(id)identifier willSendRequest:(NSURLRequest *)request redirectResponse:(NSURLResponse *)redirectResponse fromDataSource:(WebDataSource *)source {
67 return [delegate_ webView:sender resource:identifier willSendRequest:request redirectResponse:redirectResponse fromDataSource:source];
71 - (IMP) methodForSelector:(SEL)sel {
72 if (IMP method = [super methodForSelector:sel])
74 fprintf(stderr, "methodForSelector:[%s] == NULL\n", sel_getName(sel));
78 - (BOOL) respondsToSelector:(SEL)sel {
79 if ([super respondsToSelector:sel])
81 // XXX: WebThreadCreateNSInvocation returns nil
82 //fprintf(stderr, "[%s]R?%s\n", class_getName(self->isa), sel_getName(sel));
83 return delegate_ == nil ? NO : [delegate_ respondsToSelector:sel];
86 - (NSMethodSignature *) methodSignatureForSelector:(SEL)sel {
87 if (NSMethodSignature *method = [super methodSignatureForSelector:sel])
89 //fprintf(stderr, "[%s]S?%s\n", class_getName(self->isa), sel_getName(sel));
91 if (NSMethodSignature *sig = [delegate_ methodSignatureForSelector:sel])
93 // XXX: I fucking hate Apple so very very bad
94 return [NSMethodSignature signatureWithObjCTypes:"v@:"];
97 - (void) forwardInvocation:(NSInvocation *)inv {
98 SEL sel = [inv selector];
99 if (delegate_ != nil && [delegate_ respondsToSelector:sel])
100 [inv invokeWithTarget:delegate_];
106 @interface WebView (Cydia)
107 - (void) setScriptDebugDelegate:(id)delegate;
108 - (void) _setFormDelegate:(id)delegate;
109 - (void) _setUIKitDelegate:(id)delegate;
110 - (void) setWebMailDelegate:(id)delegate;
111 - (void) _setLayoutInterval:(float)interval;
114 @interface WebScriptObject (Cydia)
117 - (id) objectAtIndex:(unsigned)index;
121 @implementation WebScriptObject (Cydia)
124 id length([self valueForKey:@"length"]);
125 if ([length respondsToSelector:@selector(intValue)])
126 return [length intValue];
131 - (id) objectAtIndex:(unsigned)index {
132 return [self webScriptValueAtIndex:index];
137 /* Web Scripting {{{ */
138 @interface CydiaObject : NSObject {
142 - (id) initWithDelegate:(IndirectDelegate *)indirect;
145 @implementation CydiaObject
152 - (id) initWithDelegate:(IndirectDelegate *)indirect {
153 if ((self = [super init]) != nil) {
154 indirect_ = [indirect retain];
158 + (NSArray *) _attributeKeys {
159 return [NSArray arrayWithObjects:@"device", @"firewire", @"imei", @"mac", @"serial", nil];
162 - (NSArray *) attributeKeys {
163 return [[self class] _attributeKeys];
166 + (BOOL) isKeyExcludedFromWebScript:(const char *)name {
167 return ![[self _attributeKeys] containsObject:[NSString stringWithUTF8String:name]] && [super isKeyExcludedFromWebScript:name];
170 - (NSString *) device {
171 return [[UIDevice currentDevice] uniqueIdentifier];
175 if (![indirect_ promptForSensitive:@"Mac Address"])
179 - (NSString *) serial {
180 if (![indirect_ promptForSensitive:@"Serial #"])
184 - (NSString *) firewire {
185 if (![indirect_ promptForSensitive:@"Firewire GUID"])
189 - (NSString *) imei {
190 if (![indirect_ promptForSensitive:@"IMEI"])
194 + (NSString *) webScriptNameForSelector:(SEL)selector {
195 if (selector == @selector(close))
197 else if (selector == @selector(getPackageById:))
198 return @"getPackageById";
199 else if (selector == @selector(setAutoPopup:))
200 return @"setAutoPopup";
201 else if (selector == @selector(setButtonImage:withStyle:toFunction:))
202 return @"setButtonImage";
203 else if (selector == @selector(setButtonTitle:withStyle:toFunction:))
204 return @"setButtonTitle";
205 else if (selector == @selector(setFinishHook:))
206 return @"setFinishHook";
207 else if (selector == @selector(setPopupHook:))
208 return @"setPopupHook";
209 else if (selector == @selector(setSpecial:))
210 return @"setSpecial";
211 else if (selector == @selector(setViewportWidth:))
212 return @"setViewportWidth";
213 else if (selector == @selector(supports:))
215 else if (selector == @selector(stringWithFormat:arguments:))
217 else if (selector == @selector(localizedStringForKey:value:table:))
219 else if (selector == @selector(du:))
221 else if (selector == @selector(statfs:))
227 + (BOOL) isSelectorExcludedFromWebScript:(SEL)selector {
228 return [self webScriptNameForSelector:selector] == nil;
231 - (BOOL) supports:(NSString *)feature {
232 return [feature isEqualToString:@"window.open"];
235 - (Package *) getPackageById:(NSString *)id {
236 return [[Database sharedInstance] packageWithName:id];
239 - (NSArray *) statfs:(NSString *)path {
242 if (path == nil || statfs([path UTF8String], &stat) == -1)
245 return [NSArray arrayWithObjects:
246 [NSNumber numberWithUnsignedLong:stat.f_bsize],
247 [NSNumber numberWithUnsignedLong:stat.f_blocks],
248 [NSNumber numberWithUnsignedLong:stat.f_bfree],
252 - (NSNumber *) du:(NSString *)path {
253 NSNumber *value(nil);
256 _assert(pipe(fds) != -1);
258 pid_t pid(ExecFork());
260 _assert(dup2(fds[1], 1) != -1);
261 _assert(close(fds[0]) != -1);
262 _assert(close(fds[1]) != -1);
263 /* XXX: this should probably not use du */
264 execl("/usr/libexec/cydia/du", "du", "-s", [path UTF8String], NULL);
269 _assert(close(fds[1]) != -1);
271 if (FILE *du = fdopen(fds[0], "r")) {
273 while (fgets(line, sizeof(line), du) != NULL) {
274 size_t length(strlen(line));
275 while (length != 0 && line[length - 1] == '\n')
276 line[--length] = '\0';
277 if (char *tab = strchr(line, '\t')) {
279 value = [NSNumber numberWithUnsignedLong:strtoul(line, NULL, 0)];
284 } else _assert(close(fds[0]));
288 if (waitpid(pid, &status, 0) == -1)
300 - (void) setAutoPopup:(BOOL)popup {
301 [indirect_ setAutoPopup:popup];
304 - (void) setButtonImage:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
305 [indirect_ setButtonImage:button withStyle:style toFunction:function];
308 - (void) setButtonTitle:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
309 [indirect_ setButtonTitle:button withStyle:style toFunction:function];
312 - (void) setSpecial:(id)function {
313 [indirect_ setSpecial:function];
316 - (void) setFinishHook:(id)function {
317 [indirect_ setFinishHook:function];
320 - (void) setPopupHook:(id)function {
321 [indirect_ setPopupHook:function];
324 - (void) setViewportWidth:(float)width {
325 [indirect_ setViewportWidth:width];
328 - (NSString *) stringWithFormat:(NSString *)format arguments:(WebScriptObject *)arguments {
329 //NSLog(@"SWF:\"%@\" A:%@", format, [arguments description]);
330 unsigned count([arguments count]);
332 for (unsigned i(0); i != count; ++i)
333 values[i] = [arguments objectAtIndex:i];
334 return [[[NSString alloc] initWithFormat:format arguments:reinterpret_cast<va_list>(values)] autorelease];
337 - (NSString *) localizedStringForKey:(NSString *)key value:(NSString *)value table:(NSString *)table {
338 if (reinterpret_cast<id>(table) == [WebUndefined undefined])
340 return [[NSBundle mainBundle] localizedStringForKey:key value:value table:table];
346 @implementation BrowserView
349 #include "Internals.h"
354 NSLog(@"[BrowserView dealloc]");
357 if (challenge_ != nil)
358 [challenge_ release];
362 WebView *webview = [webview_ webView];
363 [webview setFrameLoadDelegate:nil];
364 [webview setResourceLoadDelegate:nil];
365 [webview setUIDelegate:nil];
366 [webview setScriptDebugDelegate:nil];
367 [webview setPolicyDelegate:nil];
369 [webview setDownloadDelegate:nil];
371 /* XXX: these are set by UIWebDocumentView
372 [webview _setFormDelegate:nil];
373 [webview _setUIKitDelegate:nil];
374 [webview setEditingDelegate:nil];*/
376 /* XXX: no one sets this, ever
377 [webview setWebMailDelegate:nil];*/
379 [webview_ setDelegate:nil];
380 [webview_ setGestureDelegate:nil];
381 [webview_ setFormEditingDelegate:nil];
382 [webview_ setInteractionDelegate:nil];
384 [indirect_ setDelegate:nil];
386 //NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
391 [webview_ removeFromSuperview];
392 [Documents_ addObject:[webview_ autorelease]];
403 [scroller_ setDelegate:nil];
409 if (function_ != nil)
419 [indicator_ release];
422 if (sensitive_ != nil)
423 [sensitive_ release];
429 - (void) loadURL:(NSURL *)url cachePolicy:(NSURLRequestCachePolicy)policy {
430 [self loadRequest:[NSURLRequest
437 - (void) loadURL:(NSURL *)url {
438 [self loadURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy];
441 - (NSMutableURLRequest *) _addHeadersToRequest:(NSURLRequest *)request {
442 NSMutableURLRequest *copy = [request mutableCopy];
444 if (Machine_ != NULL)
445 [copy setValue:[NSString stringWithUTF8String:Machine_] forHTTPHeaderField:@"X-Machine"];
446 if (UniqueID_ != nil)
447 [copy setValue:UniqueID_ forHTTPHeaderField:@"X-Unique-ID"];
450 [copy setValue:Role_ forHTTPHeaderField:@"X-Role"];
455 - (void) loadRequest:(NSURLRequest *)request {
460 [webview_ loadRequest:request];
468 if ([request_ HTTPBody] == nil && [request_ HTTPBodyStream] == nil)
469 [self loadRequest:request_];
471 UIActionSheet *sheet = [[[UIActionSheet alloc]
472 initWithTitle:CYLocalize("RESUBMIT_FORM")
473 buttons:[NSArray arrayWithObjects:CYLocalize("CANCEL"), CYLocalize("SUBMIT"), nil]
479 [sheet setNumberOfRows:1];
480 [sheet popupAlertAnimated:YES];
484 - (WebView *) webView {
485 return [webview_ webView];
488 - (UIWebDocumentView *) documentView {
492 /* XXX: WebThreadLock? */
493 - (void) _fixScroller:(CGRect)bounds {
498 UIFormAssistant *assistant([UIFormAssistant sharedFormAssistant]);
499 CGRect peripheral([assistant peripheralFrame]);
501 NSLog(@"per:%f", peripheral.size.height);
503 extra = peripheral.size.height;
506 CGRect subrect([scroller_ frame]);
507 subrect.size.height -= extra;
508 [scroller_ setScrollerIndicatorSubrect:subrect];
510 NSSize visible(NSMakeSize(subrect.size.width, subrect.size.height));
511 [webview_ setValue:[NSValue valueWithSize:visible] forGestureAttribute:UIGestureAttributeVisibleSize];
514 size.height += extra;
515 [scroller_ setContentSize:size];
517 [scroller_ releaseRubberBandIfNecessary];
520 - (void) fixScroller {
521 CGRect bounds([webview_ documentBounds]);
523 NSLog(@"_fs:(%f,%f+%f,%f)", bounds.origin.x, bounds.origin.y, bounds.size.width, bounds.size.height);
525 [self _fixScroller:bounds];
528 - (void) view:(UIView *)sender didSetFrame:(CGRect)frame {
531 NSLog(@"dsf:(%f,%f+%f,%f)", frame.origin.x, frame.origin.y, frame.size.width, frame.size.height);
533 [self _fixScroller:frame];
536 - (void) view:(UIView *)sender didSetFrame:(CGRect)frame oldFrame:(CGRect)old {
537 [self view:sender didSetFrame:frame];
540 - (void) pushPage:(RVPage *)page {
541 [page setDelegate:delegate_];
542 [self setBackButtonTitle:title_];
543 [book_ pushPage:page];
549 // WTR: [self autorelease];
551 [book_ pushPage:self];
554 - (void) swapPage:(RVPage *)page {
555 [page setDelegate:delegate_];
557 [book_ swapPage:page];
559 [book_ pushPage:page];
562 - (BOOL) getSpecial:(NSURL *)url swap:(BOOL)swap {
564 NSLog(@"getSpecial:%@", url);
567 NSString *href([url absoluteString]);
568 NSString *scheme([[url scheme] lowercaseString]);
572 if ([href hasPrefix:@"apptapp://package/"])
573 page = [delegate_ pageForPackage:[href substringFromIndex:18]];
574 else if ([scheme isEqualToString:@"cydia"]) {
575 page = [delegate_ pageForURL:url hasTag:NULL];
578 } else if (![scheme isEqualToString:@"apptapp"])
583 [self swapPage:page];
585 [self pushPage:page];
589 - (void) webViewShow:(WebView *)sender {
590 /* XXX: this is where I cry myself to sleep */
593 - (bool) _allowJavaScriptPanel {
597 - (bool) allowSensitiveRequests {
598 [self _allowJavaScriptPanel];
601 - (void) _promptForSensitive:(NSMutableArray *)array {
602 NSString *name([array objectAtIndex:0]);
604 UIActionSheet *sheet = [[[UIActionSheet alloc]
606 buttons:[NSArray arrayWithObjects:CYLocalize("YES"), CYLocalize("NO"), nil]
612 NSString *host(@"XXX");
614 [sheet setNumberOfRows:1];
615 [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]];
616 [sheet popupAlertAnimated:YES];
618 NSRunLoop *loop([NSRunLoop currentRunLoop]);
619 NSDate *future([NSDate distantFuture]);
621 while (sensitive_ == nil && [loop runMode:NSDefaultRunLoopMode beforeDate:future]);
623 NSNumber *sensitive([sensitive_ autorelease]);
627 [array replaceObjectAtIndex:0 withObject:sensitive];
630 - (bool) promptForSensitive:(NSString *)name {
631 if (![self allowSensitiveRequests])
634 NSMutableArray *array([NSMutableArray arrayWithCapacity:1]);
635 [array addObject:name];
637 [self performSelectorOnMainThread:@selector(_promptForSensitive:) withObject:array waitUntilDone:YES];
638 return [[array lastObject] boolValue];
641 - (void) webView:(WebView *)sender runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame {
642 if (![self _allowJavaScriptPanel])
646 UIActionSheet *sheet = [[[UIActionSheet alloc]
648 buttons:[NSArray arrayWithObjects:CYLocalize("OK"), nil]
654 [sheet setBodyText:message];
655 [sheet popupAlertAnimated:YES];
658 - (BOOL) webView:(WebView *)sender runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame {
659 if (![self _allowJavaScriptPanel])
663 UIActionSheet *sheet = [[[UIActionSheet alloc]
665 buttons:[NSArray arrayWithObjects:CYLocalize("OK"), CYLocalize("CANCEL"), nil]
671 [sheet setNumberOfRows:1];
672 [sheet setBodyText:message];
673 [sheet popupAlertAnimated:YES];
675 NSRunLoop *loop([NSRunLoop currentRunLoop]);
676 NSDate *future([NSDate distantFuture]);
678 while (confirm_ == nil && [loop runMode:NSDefaultRunLoopMode beforeDate:future]);
680 NSNumber *confirm([confirm_ autorelease]);
684 return [confirm boolValue];
687 - (void) setAutoPopup:(BOOL)popup {
691 - (void) setSpecial:(id)function {
693 [special_ autorelease];
694 special_ = function == nil ? nil : [function retain];
697 - (void) setButtonImage:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
699 [button_ autorelease];
700 button_ = button == nil ? nil : [[UIImage imageWithData:[NSData dataWithContentsOfURL:[NSURL URLWithString:button]]] retain];
703 [style_ autorelease];
704 style_ = style == nil ? nil : [style retain];
706 if (function_ != nil)
707 [function_ autorelease];
708 function_ = function == nil ? nil : [function retain];
710 [self reloadButtons];
713 - (void) setButtonTitle:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
715 [button_ autorelease];
716 button_ = button == nil ? nil : [button retain];
719 [style_ autorelease];
720 style_ = style == nil ? nil : [style retain];
722 if (function_ != nil)
723 [function_ autorelease];
724 function_ = function == nil ? nil : [function retain];
726 [self reloadButtons];
729 - (void) setFinishHook:(id)function {
731 [finish_ autorelease];
732 finish_ = function == nil ? nil : [function retain];
735 - (void) setPopupHook:(id)function {
737 [closer_ autorelease];
738 closer_ = function == nil ? nil : [function retain];
741 - (void) webView:(WebView *)sender willBeginEditingFormElement:(id)element {
745 - (void) webView:(WebView *)sender didBeginEditingFormElement:(id)element {
749 - (void) webViewDidEndEditingFormElements:(WebView *)sender {
754 - (void) webViewClose:(WebView *)sender {
762 - (void) webView:(WebView *)sender didClearWindowObject:(WebScriptObject *)window forFrame:(WebFrame *)frame {
763 [window setValue:cydia_ forKey:@"cydia"];
766 - (void) webView:(WebView *)sender unableToImplementPolicyWithError:(NSError *)error frame:(WebFrame *)frame {
767 NSLog(@"err:%@", error);
770 - (void) webView:(WebView *)sender decidePolicyForNewWindowAction:(NSDictionary *)action request:(NSURLRequest *)request newFrameName:(NSString *)name decisionListener:(id<WebPolicyDecisionListener>)listener {
772 NSLog(@"nwa:%@", name);
775 if (NSURL *url = [request URL]) {
776 if (name == nil) unknown: {
777 if (![self getSpecial:url swap:NO]) {
778 NSString *scheme([[url scheme] lowercaseString]);
779 if ([scheme isEqualToString:@"mailto"])
780 [delegate_ openMailToURL:url];
783 } else if ([name isEqualToString:@"_open"])
784 [delegate_ openURL:url];
785 else if ([name isEqualToString:@"_popup"]) {
786 NSString *scheme([[url scheme] lowercaseString]);
787 if ([scheme isEqualToString:@"mailto"])
788 [delegate_ openMailToURL:url];
790 RVBook *book([[[RVPopUpBook alloc] initWithFrame:[delegate_ popUpBounds]] autorelease]);
791 [book setHook:indirect_];
793 RVPage *page([delegate_ pageForURL:url hasTag:NULL]);
795 /* XXX: call createWebViewWithRequest instead? */
797 [self setBackButtonTitle:title_];
799 BrowserView *browser([[[BrowserView alloc] initWithBook:book] autorelease]);
800 [browser loadURL:url];
804 [book setDelegate:delegate_];
805 [page setDelegate:delegate_];
808 [book_ pushBook:book];
817 - (void) webView:(WebView *)sender decidePolicyForMIMEType:(NSString *)type request:(NSURLRequest *)request frame:(WebFrame *)frame decisionListener:(id<WebPolicyDecisionListener>)listener {
818 if ([WebView canShowMIMEType:type])
821 // XXX: handle more mime types!
824 WebView *webview([webview_ webView]);
825 if (frame == [webview mainFrame])
826 [UIApp openURL:[request URL]];
830 - (void) webView:(WebView *)sender decidePolicyForNavigationAction:(NSDictionary *)action request:(NSURLRequest *)request frame:(WebFrame *)frame decisionListener:(id<WebPolicyDecisionListener>)listener {
831 if (request == nil) ignore: {
836 NSURL *url([request URL]);
838 if (url == nil) use: {
839 if (!error_ && [frame parentFrame] == nil) {
841 [request_ autorelease];
842 request_ = [request retain];
844 NSLog(@"dpn:%@", request_);
850 WebView *webview([webview_ webView]);
851 if (frame == [webview mainFrame])
856 else NSLog(@"nav:%@:%@", url, [action description]);
859 const NSArray *capability;
861 #if 0 // XXX:3:GSSystemCopyCapability
862 capability = reinterpret_cast<const NSArray *>(GSSystemGetCapability(kGSDisplayIdentifiersCapability));
867 if (capability != nil && (
868 [capability containsObject:@"com.apple.Maps"] && [url mapsURL] ||
869 [capability containsObject:@"com.apple.youtube"] && [url youTubeURL]
876 int store(_not(int));
877 if (NSURL *itms = [url itmsURL:&store]) {
879 NSLog(@"itms#%@#%u#%@", url, store, itms);
882 if (capability != nil && (
883 store == 1 && [capability containsObject:@"com.apple.MobileStore"] ||
884 store == 2 && [capability containsObject:@"com.apple.AppStore"]
891 NSString *scheme([[url scheme] lowercaseString]);
893 if ([scheme isEqualToString:@"tel"]) {
898 if ([scheme isEqualToString:@"mailto"]) {
899 [delegate_ openMailToURL:url];
903 if ([self getSpecial:url swap:YES])
905 else if ([WebView _canHandleRequest:request])
907 else if ([url isSpringboardHandledURL])
913 - (void) webView:(WebView *)sender setStatusText:(NSString *)text {
914 //lprintf("Status:%s\n", [text UTF8String]);
917 - (void) alertSheet:(UIActionSheet *)sheet buttonClicked:(int)button {
918 NSString *context([sheet context]);
920 if ([context isEqualToString:@"alert"]) {
923 } else if ([context isEqualToString:@"confirm"]) {
926 confirm_ = [NSNumber numberWithBool:YES];
930 confirm_ = [NSNumber numberWithBool:NO];
935 } else if ([context isEqualToString:@"sensitive"]) {
938 sensitive_ = [NSNumber numberWithBool:YES];
942 sensitive_ = [NSNumber numberWithBool:NO];
947 } else if ([context isEqualToString:@"challenge"]) {
948 id<NSURLAuthenticationChallengeSender> sender([challenge_ sender]);
952 NSString *username([[sheet textFieldAtIndex:0] text]);
953 NSString *password([[sheet textFieldAtIndex:1] text]);
955 NSURLCredential *credential([NSURLCredential credentialWithUser:username password:password persistence:NSURLCredentialPersistenceForSession]);
957 [sender useCredential:credential forAuthenticationChallenge:challenge_];
961 [sender cancelAuthenticationChallenge:challenge_];
968 [challenge_ release];
972 } else if ([context isEqualToString:@"submit"]) {
978 if (request_ != nil) {
980 [webview_ loadRequest:request_];
993 - (void) webView:(WebView *)sender resource:(id)identifier didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge fromDataSource:(WebDataSource *)source {
994 challenge_ = [challenge retain];
996 NSURLProtectionSpace *space([challenge protectionSpace]);
997 NSString *realm([space realm]);
1001 UIActionSheet *sheet = [[[UIActionSheet alloc]
1003 buttons:[NSArray arrayWithObjects:CYLocalize("LOGIN"), CYLocalize("CANCEL"), nil]
1004 defaultButtonIndex:0
1006 context:@"challenge"
1009 [sheet setNumberOfRows:1];
1011 [sheet addTextFieldWithValue:@"" label:CYLocalize("USERNAME")];
1012 [sheet addTextFieldWithValue:@"" label:CYLocalize("PASSWORD")];
1014 UITextField *username([sheet textFieldAtIndex:0]); {
1015 UITextInputTraits *traits([username textInputTraits]);
1016 [traits setAutocapitalizationType:UITextAutocapitalizationTypeNone];
1017 [traits setAutocorrectionType:UITextAutocorrectionTypeNo];
1018 [traits setKeyboardType:UIKeyboardTypeASCIICapable];
1019 [traits setReturnKeyType:UIReturnKeyNext];
1022 UITextField *password([sheet textFieldAtIndex:1]); {
1023 UITextInputTraits *traits([password textInputTraits]);
1024 [traits setAutocapitalizationType:UITextAutocapitalizationTypeNone];
1025 [traits setAutocorrectionType:UITextAutocorrectionTypeNo];
1026 [traits setKeyboardType:UIKeyboardTypeASCIICapable];
1027 // XXX: UIReturnKeyDone
1028 [traits setReturnKeyType:UIReturnKeyNext];
1029 [traits setSecureTextEntry:YES];
1032 [sheet popupAlertAnimated:YES];
1035 - (NSURLRequest *) webView:(WebView *)sender resource:(id)identifier willSendRequest:(NSURLRequest *)request redirectResponse:(NSURLResponse *)redirectResponse fromDataSource:(WebDataSource *)source {
1036 return [self _addHeadersToRequest:request];
1039 - (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request windowFeatures:(NSDictionary *)features {
1040 //- (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request userGesture:(BOOL)gesture {
1042 NSLog(@"cwv:%@ (%@): %@", request, title_, features == nil ? @"{}" : [features description]);
1043 //NSLog(@"cwv:%@ (%@): %@", request, title_, gesture ? @"Yes" : @"No");
1046 NSNumber *value([features objectForKey:@"width"]);
1047 float width(value == nil ? 0 : [value floatValue]);
1049 RVBook *book(!popup_ ? book_ : [[[RVPopUpBook alloc] initWithFrame:[delegate_ popUpBounds]] autorelease]);
1051 /* XXX: deal with cydia:// pages */
1052 BrowserView *browser([[[BrowserView alloc] initWithBook:book forWidth:width] autorelease]);
1054 if (features != nil && popup_) {
1055 [book setDelegate:delegate_];
1056 [book setHook:indirect_];
1057 [browser setDelegate:delegate_];
1059 [browser loadRequest:request];
1061 [book setPage:browser];
1062 [book_ pushBook:book];
1063 } else if (request == nil) {
1064 [self setBackButtonTitle:title_];
1065 [browser setDelegate:delegate_];
1068 [self pushPage:browser];
1069 [browser loadRequest:request];
1072 return [browser webView];
1075 - (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request {
1076 return [self webView:sender createWebViewWithRequest:request windowFeatures:nil];
1077 //return [self webView:sender createWebViewWithRequest:request userGesture:YES];
1080 - (void) webView:(WebView *)sender didReceiveTitle:(NSString *)title forFrame:(WebFrame *)frame {
1081 if ([frame parentFrame] != nil)
1084 title_ = [title retain];
1085 [book_ reloadTitleForPage:self];
1088 - (void) webView:(WebView *)sender didStartProvisionalLoadForFrame:(WebFrame *)frame {
1089 if ([loading_ count] == 0)
1091 [loading_ addObject:[NSValue valueWithNonretainedObject:frame]];
1093 if ([frame parentFrame] == nil) {
1094 [webview_ resignFirstResponder];
1098 if (title_ != nil) {
1103 if (button_ != nil) {
1108 if (style_ != nil) {
1113 if (function_ != nil) {
1114 [function_ release];
1118 if (finish_ != nil) {
1123 if (closer_ != nil) {
1128 if (special_ != nil) {
1133 [book_ reloadTitleForPage:self];
1135 [scroller_ scrollPointVisibleAtTopLeft:CGPointZero];
1137 if ([scroller_ respondsToSelector:@selector(setZoomScale:duration:)])
1138 [scroller_ setZoomScale:1 duration:0];
1139 else if ([scroller_ respondsToSelector:@selector(_setZoomScale:duration:)])
1140 [scroller_ _setZoomScale:1 duration:0];
1141 /*else if ([scroller_ respondsToSelector:@selector(setZoomScale:animated:)])
1142 [scroller_ setZoomScale:1 animated:NO];*/
1144 CGRect webrect = [scroller_ bounds];
1145 webrect.size.height = 0;
1146 [webview_ setFrame:webrect];
1149 [self reloadButtons];
1152 - (void) _finishLoading {
1153 size_t count([loading_ count]);
1156 if (reloading_ || count != 0)
1159 [self callFunction:finish_];
1160 [self reloadButtons];
1163 - (bool) isLoading {
1164 return [loading_ count] != 0;
1167 - (void) reloadButtons {
1168 if ([self isLoading])
1169 [indicator_ startAnimation];
1171 [indicator_ stopAnimation];
1172 [super reloadButtons];
1175 - (BOOL) webView:(WebView *)sender shouldScrollToPoint:(struct CGPoint)point forFrame:(WebFrame *)frame {
1176 return [webview_ webView:sender shouldScrollToPoint:point forFrame:frame];
1179 - (void) webView:(WebView *)sender didReceiveViewportArguments:(id)arguments forFrame:(WebFrame *)frame {
1180 return [webview_ webView:sender didReceiveViewportArguments:arguments forFrame:frame];
1183 - (void) webView:(WebView *)sender needsScrollNotifications:(id)notifications forFrame:(WebFrame *)frame {
1184 return [webview_ webView:sender needsScrollNotifications:notifications forFrame:frame];
1187 - (void) webView:(WebView *)sender didCommitLoadForFrame:(WebFrame *)frame {
1189 return [webview_ webView:sender didCommitLoadForFrame:frame];
1192 - (void) webView:(WebView *)sender didReceiveDocTypeForFrame:(WebFrame *)frame {
1193 return [webview_ webView:sender didReceiveDocTypeForFrame:frame];
1196 - (void) webView:(WebView *)sender didFinishLoadForFrame:(WebFrame *)frame {
1197 [loading_ removeObject:[NSValue valueWithNonretainedObject:frame]];
1198 [self _finishLoading];
1200 if ([frame parentFrame] == nil) {
1201 if (DOMDocument *document = [frame DOMDocument])
1202 if (DOMNodeList<NSFastEnumeration> *bodies = [document getElementsByTagName:@"body"])
1203 for (DOMHTMLBodyElement *body in bodies) {
1204 DOMCSSStyleDeclaration *style([document getComputedStyle:body pseudoElement:nil]);
1206 bool colored(false);
1208 if (DOMCSSPrimitiveValue *color = static_cast<DOMCSSPrimitiveValue *>([style getPropertyCSSValue:@"background-color"])) {
1209 if ([color primitiveType] == DOM_CSS_RGBCOLOR) {
1210 DOMRGBColor *rgb([color getRGBColorValue]);
1212 float red([[rgb red] getFloatValue:DOM_CSS_NUMBER]);
1213 float green([[rgb green] getFloatValue:DOM_CSS_NUMBER]);
1214 float blue([[rgb blue] getFloatValue:DOM_CSS_NUMBER]);
1215 float alpha([[rgb alpha] getFloatValue:DOM_CSS_NUMBER]);
1219 if (red == 0xc7 && green == 0xce && blue == 0xd5)
1220 uic = [UIColor pinStripeColor];
1221 else if (alpha != 0)
1223 colorWithRed:(red / 255)
1231 [scroller_ setBackgroundColor:uic];
1237 [scroller_ setBackgroundColor:[UIColor pinStripeColor]];
1242 return [webview_ webView:sender didFinishLoadForFrame:frame];
1245 - (void) _didFailWithError:(NSError *)error forFrame:(WebFrame *)frame {
1246 if ([frame parentFrame] == nil)
1249 [loading_ removeObject:[NSValue valueWithNonretainedObject:frame]];
1250 [self _finishLoading];
1255 if ([frame parentFrame] == nil) {
1256 [self loadURL:[NSURL URLWithString:[NSString stringWithFormat:@"%@?%@",
1257 [[NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"error" ofType:@"html"]] absoluteString],
1258 [[error localizedDescription] stringByAddingPercentEscapes]
1265 - (void) webView:(WebView *)sender didFailLoadWithError:(NSError *)error forFrame:(WebFrame *)frame {
1266 [self _didFailWithError:error forFrame:frame];
1269 - (void) webView:(WebView *)sender didFailProvisionalLoadWithError:(NSError *)error forFrame:(WebFrame *)frame {
1270 [self _didFailWithError:error forFrame:frame];
1273 - (void) webView:(WebView *)sender addMessageToConsole:(NSDictionary *)dictionary {
1274 #if LogBrowser || ForSaurik
1275 lprintf("Console:%s\n", [[dictionary description] UTF8String]);
1279 /* XXX: fix this stupid include file
1280 - (void) webView:(WebView *)sender frame:(WebFrame *)frame exceededDatabaseQuotaForSecurityOrigin:(WebSecurityOrigin *)origin database:(NSString *)database {
1281 [origin setQuota:0x500000];
1284 - (void) _setTileDrawingEnabled:(BOOL)enabled {
1285 //[webview_ setTileDrawingEnabled:enabled];
1288 - (void) setViewportWidth:(float)width {
1289 width_ = width ? width != 0 : [[self class] defaultWidth];
1290 [webview_ setViewportSize:CGSizeMake(width_, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x10];
1293 - (void) willStartGesturesInView:(UIView *)view forEvent:(GSEventRef)event {
1294 [self _setTileDrawingEnabled:NO];
1297 - (void) didFinishGesturesInView:(UIView *)view forEvent:(GSEventRef)event {
1298 [self _setTileDrawingEnabled:YES];
1299 [webview_ redrawScaledDocument];
1302 - (void) scrollerWillStartDragging:(UIScroller *)scroller {
1303 [self _setTileDrawingEnabled:NO];
1306 - (void) scrollerDidEndDragging:(UIScroller *)scroller willSmoothScroll:(BOOL)smooth {
1307 [self _setTileDrawingEnabled:YES];
1310 - (void) scrollerDidEndDragging:(UIScroller *)scroller {
1311 [self _setTileDrawingEnabled:YES];
1314 - (id) initWithBook:(RVBook *)book forWidth:(float)width {
1315 if ((self = [super initWithBook:book]) != nil) {
1316 loading_ = [[NSMutableSet alloc] initWithCapacity:3];
1319 struct CGRect bounds = [self bounds];
1321 scroller_ = [[UIScroller alloc] initWithFrame:bounds];
1322 [self addSubview:scroller_];
1324 [scroller_ setFixedBackgroundPattern:YES];
1325 [scroller_ setBackgroundColor:[UIColor pinStripeColor]];
1327 [scroller_ setScrollingEnabled:YES];
1328 [scroller_ setClipsSubviews:YES];
1329 [scroller_ setAllowsRubberBanding:YES];
1331 [scroller_ setDelegate:self];
1332 [scroller_ setBounces:YES];
1333 [scroller_ setScrollHysteresis:8];
1334 [scroller_ setThumbDetectionEnabled:NO];
1335 [scroller_ setDirectionalScrolling:YES];
1336 [scroller_ setScrollDecelerationFactor:0.99]; /* 0.989324 */
1337 [scroller_ setEventMode:YES];
1338 [scroller_ setShowBackgroundShadow:NO]; /* YES */
1339 [scroller_ setAllowsRubberBanding:YES]; /* Vertical */
1340 [scroller_ setAdjustForContentSizeChange:YES]; /* NO */
1342 CGRect webrect = [scroller_ bounds];
1343 webrect.size.height = 0;
1350 webview_ = [Documents_ lastObject];
1351 if (webview_ != nil) {
1352 webview_ = [webview_ retain];
1353 webview = [webview_ webView];
1354 [Documents_ removeLastObject];
1355 [webview_ setFrame:webrect];
1360 webview_ = [[UIWebDocumentView alloc] initWithFrame:webrect];
1361 webview = [webview_ webView];
1363 // XXX: this is terribly (too?) expensive
1364 //[webview_ setDrawsBackground:NO];
1365 [webview setPreferencesIdentifier:@"Cydia"];
1367 [webview_ setTileSize:CGSizeMake(webrect.size.width, 500)];
1369 [webview_ setAllowsMessaging:YES];
1371 [webview_ setTilingEnabled:YES];
1372 [webview_ setDrawsGrid:NO];
1373 [webview_ setLogsTilingChanges:NO];
1374 [webview_ setTileMinificationFilter:kCAFilterNearest];
1375 if ([webview_ respondsToSelector:@selector(setDataDetectorTypes:)])
1376 /* XXX: abstractify */
1377 [webview_ setDataDetectorTypes:0x80000000];
1379 [webview_ setDetectsPhoneNumbers:NO];
1380 [webview_ setAutoresizes:YES];
1382 [webview_ setMinimumScale:0.25f forDocumentTypes:0x10];
1383 [webview_ setMaximumScale:5.00f forDocumentTypes:0x10];
1384 [webview_ setInitialScale:UIWebViewScalesToFitScale forDocumentTypes:0x10];
1385 //[webview_ setViewportSize:CGSizeMake(980, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x10];
1387 [webview_ setViewportSize:CGSizeMake(320, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x2];
1389 [webview_ setMinimumScale:1.00f forDocumentTypes:0x8];
1390 [webview_ setInitialScale:UIWebViewScalesToFitScale forDocumentTypes:0x8];
1391 [webview_ setViewportSize:CGSizeMake(320, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x8];
1393 [webview_ _setDocumentType:0x4];
1395 if ([webview_ respondsToSelector:@selector(UIWebDocumentView:)])
1396 [webview_ setZoomsFocusedFormControl:YES];
1397 [webview_ setContentsPosition:7];
1398 [webview_ setEnabledGestures:0xa];
1399 [webview_ setValue:[NSNumber numberWithBool:YES] forGestureAttribute:UIGestureAttributeIsZoomRubberBandEnabled];
1400 [webview_ setValue:[NSNumber numberWithBool:YES] forGestureAttribute:UIGestureAttributeUpdatesScroller];
1402 [webview_ setSmoothsFonts:YES];
1403 [webview_ setAllowsImageSheet:YES];
1404 [webview _setUsesLoaderCache:YES];
1406 [webview setGroupName:@"CydiaGroup"];
1407 if ([webview respondsToSelector:@selector(_setLayoutInterval:)])
1408 [webview _setLayoutInterval:0];
1411 [self setViewportWidth:width];
1413 [webview_ setDelegate:self];
1414 [webview_ setGestureDelegate:self];
1415 [webview_ setFormEditingDelegate:self];
1416 [webview_ setInteractionDelegate:self];
1418 [scroller_ addSubview:webview_];
1420 //NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
1422 Package *package([[Database sharedInstance] packageWithName:@"cydia"]);
1423 NSString *application = package == nil ? @"Cydia" : [NSString
1424 stringWithFormat:@"Cydia/%@",
1428 if (Product_ != nil)
1429 application = [NSString stringWithFormat:@"%@ Version/%@", application, Product_];
1431 application = [NSString stringWithFormat:@"%@ Mobile/%@", application, Build_];
1433 application = [NSString stringWithFormat:@"%@ Safari/%@", application, Safari_];
1435 [webview setApplicationNameForUserAgent:application];
1437 indirect_ = [[IndirectDelegate alloc] initWithDelegate:self];
1438 cydia_ = [[CydiaObject alloc] initWithDelegate:indirect_];
1440 [webview setFrameLoadDelegate:indirect_];
1441 [webview setResourceLoadDelegate:indirect_];
1442 [webview setUIDelegate:indirect_];
1443 [webview setScriptDebugDelegate:indirect_];
1444 [webview setPolicyDelegate:indirect_];
1448 CGSize indsize = [UIProgressIndicator defaultSizeForStyle:UIProgressIndicatorStyleMediumWhite];
1449 indicator_ = [[UIProgressIndicator alloc] initWithFrame:CGRectMake(281, 12, indsize.width, indsize.height)];
1450 [indicator_ setStyle:UIProgressIndicatorStyleMediumWhite];
1452 [self setAutoresizingMask:UIViewAutoresizingFlexibleHeight];
1453 [scroller_ setAutoresizingMask:UIViewAutoresizingFlexibleHeight];
1455 /*UIWebView *test([[[UIWebView alloc] initWithFrame:[self bounds]] autorelease]);
1456 [test loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.saurik.com/"]]];
1457 [self addSubview:test];*/
1461 - (id) initWithBook:(RVBook *)book {
1462 return [self initWithBook:book forWidth:0];
1465 - (NSString *) stringByEvaluatingJavaScriptFromString:(NSString *)script {
1467 WebView *webview([webview_ webView]);
1468 NSString *string([webview stringByEvaluatingJavaScriptFromString:script]);
1473 - (void) callFunction:(WebScriptObject *)function {
1476 WebView *webview([webview_ webView]);
1477 WebFrame *frame([webview mainFrame]);
1479 id _private(MSHookIvar<id>(webview, "_private"));
1480 WebCore::Page *page(_private == nil ? NULL : MSHookIvar<WebCore::Page *>(_private, "page"));
1481 WebCore::Settings *settings(page == NULL ? NULL : page->settings());
1484 if (settings == NULL)
1487 no = settings->JavaScriptCanOpenWindowsAutomatically();
1488 settings->setJavaScriptCanOpenWindowsAutomatically(true);
1491 [delegate_ clearFirstResponder];
1492 JSObjectRef object([function JSObject]);
1493 JSGlobalContextRef context([frame globalContext]);
1494 JSObjectCallAsFunction(context, object, NULL, 0, NULL, NULL);
1496 if (settings != NULL)
1497 settings->setJavaScriptCanOpenWindowsAutomatically(no);
1502 - (void) didCloseBook:(RVBook *)book {
1504 [self callFunction:closer_];
1507 - (void) __rightButtonClicked {
1512 - (void) _rightButtonClicked {
1514 if (function_ != nil)
1515 [self callFunction:function_];
1518 [self __rightButtonClicked];
1521 - (id) _rightButtonTitle {
1522 return CYLocalize("RELOAD");
1525 - (id) rightButtonTitle {
1526 return [self isLoading] ? @"" : button_ != nil ? button_ : [self _rightButtonTitle];
1529 - (UINavigationButtonStyle) rightButtonStyle {
1530 if (style_ == nil) normal:
1531 return UINavigationButtonStyleNormal;
1532 else if ([style_ isEqualToString:@"Normal"])
1533 return UINavigationButtonStyleNormal;
1534 else if ([style_ isEqualToString:@"Back"])
1535 return UINavigationButtonStyleBack;
1536 else if ([style_ isEqualToString:@"Highlighted"])
1537 return UINavigationButtonStyleHighlighted;
1538 else if ([style_ isEqualToString:@"Destructive"])
1539 return UINavigationButtonStyleDestructive;
1543 - (NSString *) title {
1544 return title_ == nil ? CYLocalize("LOADING") : title_;
1547 - (NSString *) backButtonTitle {
1548 return CYLocalize("BROWSER");
1551 - (void) setPageActive:(BOOL)active {
1553 [indicator_ removeFromSuperview];
1555 [[book_ navigationBar] addSubview:indicator_];
1558 - (void) resetViewAnimated:(BOOL)animated {
1561 - (void) setPushed:(bool)pushed {
1565 + (float) defaultWidth {