1 #include <BrowserView.h>
3 @interface WebView (Cydia)
4 - (void) setScriptDebugDelegate:(id)delegate;
5 - (void) _setFormDelegate:(id)delegate;
6 - (void) _setUIKitDelegate:(id)delegate;
7 - (void) setWebMailDelegate:(id)delegate;
8 - (void) _setLayoutInterval:(float)interval;
11 @implementation BrowserView
14 #include "Internals.h"
18 if (challenge_ != nil)
21 WebView *webview = [webview_ webView];
22 [webview setFrameLoadDelegate:nil];
23 [webview setResourceLoadDelegate:nil];
24 [webview setUIDelegate:nil];
25 [webview setScriptDebugDelegate:nil];
26 [webview setPolicyDelegate:nil];
28 [webview setDownloadDelegate:nil];
30 [webview _setFormDelegate:nil];
31 [webview _setUIKitDelegate:nil];
32 [webview setWebMailDelegate:nil];
33 [webview setEditingDelegate:nil];
35 [webview_ setDelegate:nil];
36 [webview_ setGestureDelegate:nil];
38 //NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
43 [webview_ removeFromSuperview];
44 [Documents_ addObject:[webview_ autorelease]];
49 [indirect_ setDelegate:nil];
52 [scroller_ setDelegate:nil];
70 - (void) loadURL:(NSURL *)url cachePolicy:(NSURLRequestCachePolicy)policy {
71 [self loadRequest:[NSURLRequest
78 - (void) loadURL:(NSURL *)url {
79 [self loadURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy];
82 - (NSMutableURLRequest *) _addHeadersToRequest:(NSURLRequest *)request {
83 NSMutableURLRequest *copy = [request mutableCopy];
86 [copy setValue:[NSString stringWithUTF8String:Machine_] forHTTPHeaderField:@"X-Machine"];
88 [copy setValue:UniqueID_ forHTTPHeaderField:@"X-Unique-ID"];
91 [copy setValue:Role_ forHTTPHeaderField:@"X-Role"];
96 - (void) loadRequest:(NSURLRequest *)request {
99 [webview_ loadRequest:request];
103 NSLog(@"rlu:%@", request_);
107 if ([request_ HTTPBody] == nil && [request_ HTTPBodyStream] == nil)
108 [self loadRequest:request_];
110 UIActionSheet *sheet = [[[UIActionSheet alloc]
111 initWithTitle:@"Are you sure you want to submit this form again?"
112 buttons:[NSArray arrayWithObjects:@"Cancel", @"Submit", nil]
118 [sheet setNumberOfRows:1];
119 [sheet popupAlertAnimated:YES];
123 - (WebView *) webView {
124 return [webview_ webView];
127 - (void) view:(UIView *)sender didSetFrame:(CGRect)frame {
128 [scroller_ setContentSize:frame.size];
131 - (void) view:(UIView *)sender didSetFrame:(CGRect)frame oldFrame:(CGRect)old {
132 [self view:sender didSetFrame:frame];
135 - (void) pushPage:(RVPage *)page {
136 [self setBackButtonTitle:title_];
137 [page setDelegate:delegate_];
138 [book_ pushPage:page];
141 - (BOOL) getSpecial:(NSURL *)url {
142 NSString *href([url absoluteString]);
143 NSString *scheme([[url scheme] lowercaseString]);
147 if ([href hasPrefix:@"apptapp://package/"])
148 page = [delegate_ pageForPackage:[href substringFromIndex:18]];
149 else if ([scheme isEqualToString:@"cydia"]) {
150 page = [delegate_ pageForURL:url hasTag:NULL];
153 } else if (![scheme isEqualToString:@"apptapp"])
157 [self pushPage:page];
161 - (void) webView:(WebView *)sender runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame {
162 UIActionSheet *sheet = [[[UIActionSheet alloc]
164 buttons:[NSArray arrayWithObjects:@"OK", nil]
170 [sheet setBodyText:message];
171 [sheet popupAlertAnimated:YES];
174 - (BOOL) webView:(WebView *)sender runJavaScriptConfirmPanelWithMessage:(NSString *)message initiatedByFrame:(WebFrame *)frame {
175 UIActionSheet *sheet = [[[UIActionSheet alloc]
177 buttons:[NSArray arrayWithObjects:@"OK", @"Cancel", nil]
183 [sheet setNumberOfRows:1];
184 [sheet setBodyText:message];
185 [sheet popupAlertAnimated:YES];
187 NSRunLoop *loop([NSRunLoop currentRunLoop]);
188 NSDate *future([NSDate distantFuture]);
190 while (confirm_ == nil && [loop runMode:NSDefaultRunLoopMode beforeDate:future]);
192 NSNumber *confirm([confirm_ autorelease]);
194 return [confirm boolValue];
197 /* Web Scripting {{{ */
198 + (NSString *) webScriptNameForSelector:(SEL)selector {
199 if (selector == @selector(getPackageById:))
200 return @"getPackageById";
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(supports:))
211 + (BOOL) isSelectorExcludedFromWebScript:(SEL)selector {
212 return [self webScriptNameForSelector:selector] == nil;
215 - (BOOL) supports:(NSString *)feature {
216 return [feature isEqualToString:@"window.open"];
219 - (Package *) getPackageById:(NSString *)id {
220 return [[Database sharedInstance] packageWithName:id];
223 - (void) setButtonImage:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
225 [button_ autorelease];
226 button_ = button == nil ? nil : [[UIImage imageWithData:[NSData dataWithContentsOfURL:[NSURL URLWithString:button]]] retain];
229 [style_ autorelease];
230 style_ = style == nil ? nil : [style retain];
232 if (function_ != nil)
233 [function_ autorelease];
234 function_ = function == nil ? nil : [function retain];
237 - (void) setButtonTitle:(NSString *)button withStyle:(NSString *)style toFunction:(id)function {
239 [button_ autorelease];
240 button_ = button == nil ? nil : [button retain];
243 [style_ autorelease];
244 style_ = style == nil ? nil : [style retain];
246 if (function_ != nil)
247 [function_ autorelease];
248 function_ = function == nil ? nil : [function retain];
252 - (void) webView:(WebView *)sender didClearWindowObject:(WebScriptObject *)window forFrame:(WebFrame *)frame {
253 [window setValue:self forKey:@"cydia"];
256 - (void) webView:(WebView *)sender unableToImplementPolicyWithError:(NSError *)error frame:(WebFrame *)frame {
257 NSLog(@"err:%@", error);
260 - (void) webView:(WebView *)sender decidePolicyForNewWindowAction:(NSDictionary *)action request:(NSURLRequest *)request newFrameName:(NSString *)name decisionListener:(id<WebPolicyDecisionListener>)listener {
261 if (NSURL *url = [request URL]) {
262 if (name != nil && [name isEqualToString:@"_open"])
263 [delegate_ openURL:url];
265 NSLog(@"win:%@:%@", url, [action description]);
266 if (![self getSpecial:url]) {
267 NSString *scheme([[url scheme] lowercaseString]);
268 if ([scheme isEqualToString:@"mailto"])
269 [delegate_ openMailToURL:url];
278 - (void) webView:(WebView *)webView decidePolicyForMIMEType:(NSString *)type request:(NSURLRequest *)request frame:(WebFrame *)frame decisionListener:(id<WebPolicyDecisionListener>)listener {
279 if ([WebView canShowMIMEType:type])
282 // XXX: handle more mime types!
284 if (frame == [webView mainFrame])
285 [UIApp openURL:[request URL]];
289 - (void) webView:(WebView *)sender decidePolicyForNavigationAction:(NSDictionary *)action request:(NSURLRequest *)request frame:(WebFrame *)frame decisionListener:(id<WebPolicyDecisionListener>)listener {
290 if (request == nil) ignore: {
295 NSURL *url([request URL]);
297 if (url == nil) use: {
298 if (!error_ && [frame parentFrame] == nil) {
300 [request_ autorelease];
301 request_ = [request retain];
302 NSLog(@"dpn:%@", request_);
309 else NSLog(@"nav:%@:%@", url, [action description]);
312 const NSArray *capability(reinterpret_cast<const NSArray *>(GSSystemGetCapability(kGSDisplayIdentifiersCapability)));
315 [capability containsObject:@"com.apple.Maps"] && [url mapsURL] ||
316 [capability containsObject:@"com.apple.youtube"] && [url youTubeURL]
323 int store(_not(int));
324 if (NSURL *itms = [url itmsURL:&store]) {
325 NSLog(@"itms#%@#%u#%@", url, store, itms);
327 store == 1 && [capability containsObject:@"com.apple.MobileStore"] ||
328 store == 2 && [capability containsObject:@"com.apple.AppStore"]
335 NSString *scheme([[url scheme] lowercaseString]);
337 if ([scheme isEqualToString:@"tel"]) {
342 if ([scheme isEqualToString:@"mailto"]) {
343 [delegate_ openMailToURL:url];
347 if ([self getSpecial:url])
349 else if ([WebView _canHandleRequest:request])
351 else if ([url isSpringboardHandledURL])
357 - (void) webView:(WebView *)sender setStatusText:(NSString *)text {
358 //lprintf("Status:%s\n", [text UTF8String]);
365 [book_ pushPage:self];
368 - (void) alertSheet:(UIActionSheet *)sheet buttonClicked:(int)button {
369 NSString *context([sheet context]);
371 if ([context isEqualToString:@"alert"])
373 else if ([context isEqualToString:@"confirm"]) {
376 confirm_ = [NSNumber numberWithBool:YES];
380 confirm_ = [NSNumber numberWithBool:NO];
385 } else if ([context isEqualToString:@"challenge"]) {
386 id<NSURLAuthenticationChallengeSender> sender([challenge_ sender]);
390 NSString *username([[sheet textFieldAtIndex:0] text]);
391 NSString *password([[sheet textFieldAtIndex:1] text]);
393 NSURLCredential *credential([NSURLCredential credentialWithUser:username password:password persistence:NSURLCredentialPersistenceForSession]);
395 [sender useCredential:credential forAuthenticationChallenge:challenge_];
399 [sender cancelAuthenticationChallenge:challenge_];
406 [challenge_ release];
410 } else if ([context isEqualToString:@"submit"]) {
417 [webview_ loadRequest:request_];
428 - (void) webView:(WebView *)sender resource:(id)identifier didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge fromDataSource:(WebDataSource *)source {
429 challenge_ = [challenge retain];
431 NSURLProtectionSpace *space([challenge protectionSpace]);
432 NSString *realm([space realm]);
436 UIActionSheet *sheet = [[[UIActionSheet alloc]
438 buttons:[NSArray arrayWithObjects:@"Login", @"Cancel", nil]
444 [sheet setNumberOfRows:1];
446 [sheet addTextFieldWithValue:@"" label:@"username"];
447 [sheet addTextFieldWithValue:@"" label:@"password"];
449 UITextField *username([sheet textFieldAtIndex:0]); {
450 UITextInputTraits *traits([username textInputTraits]);
451 [traits setAutocapitalizationType:UITextAutocapitalizationTypeNone];
452 [traits setAutocorrectionType:UITextAutocorrectionTypeNo];
453 [traits setKeyboardType:UIKeyboardTypeASCIICapable];
454 [traits setReturnKeyType:UIReturnKeyNext];
457 UITextField *password([sheet textFieldAtIndex:1]); {
458 UITextInputTraits *traits([password textInputTraits]);
459 [traits setAutocapitalizationType:UITextAutocapitalizationTypeNone];
460 [traits setAutocorrectionType:UITextAutocorrectionTypeNo];
461 [traits setKeyboardType:UIKeyboardTypeASCIICapable];
462 // XXX: UIReturnKeyDone
463 [traits setReturnKeyType:UIReturnKeyNext];
464 [traits setSecureTextEntry:YES];
467 [sheet popupAlertAnimated:YES];
470 - (NSURLRequest *) webView:(WebView *)sender resource:(id)identifier willSendRequest:(NSURLRequest *)request redirectResponse:(NSURLResponse *)redirectResponse fromDataSource:(WebDataSource *)source {
471 NSURL *url = [request URL];
472 if ([self getSpecial:url])
475 return [self _addHeadersToRequest:request];
478 - (WebView *) _createWebViewWithRequest:(NSURLRequest *)request pushed:(BOOL)pushed {
479 [self setBackButtonTitle:title_];
481 BrowserView *browser = [[[BrowserView alloc] initWithBook:book_] autorelease];
482 [browser setDelegate:delegate_];
485 [browser loadRequest:request];
486 [book_ pushPage:browser];
489 return [browser webView];
492 - (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request {
493 return [self _createWebViewWithRequest:request pushed:(request != nil)];
496 - (WebView *) webView:(WebView *)sender createWebViewWithRequest:(NSURLRequest *)request windowFeatures:(NSDictionary *)features {
497 return [self _createWebViewWithRequest:request pushed:YES];
500 - (void) webView:(WebView *)sender didReceiveTitle:(NSString *)title forFrame:(WebFrame *)frame {
501 if ([frame parentFrame] != nil)
504 title_ = [title retain];
505 [book_ reloadTitleForPage:self];
508 - (void) webView:(WebView *)sender didStartProvisionalLoadForFrame:(WebFrame *)frame {
509 if ([frame parentFrame] != nil)
514 [self reloadButtons];
521 if (button_ != nil) {
531 if (function_ != nil) {
536 [book_ reloadTitleForPage:self];
538 [scroller_ scrollPointVisibleAtTopLeft:CGPointZero];
540 CGRect webrect = [scroller_ bounds];
541 webrect.size.height = 0;
542 [webview_ setFrame:webrect];
545 - (void) _finishLoading {
548 [self reloadButtons];
556 - (void) reloadButtons {
558 [indicator_ startAnimation];
560 [indicator_ stopAnimation];
561 [super reloadButtons];
564 - (BOOL) webView:(WebView *)sender shouldScrollToPoint:(struct CGPoint)point forFrame:(WebFrame *)frame {
565 return [webview_ webView:sender shouldScrollToPoint:point forFrame:frame];
568 - (void) webView:(WebView *)sender didReceiveViewportArguments:(id)arguments forFrame:(WebFrame *)frame {
569 return [webview_ webView:sender didReceiveViewportArguments:arguments forFrame:frame];
572 - (void) webView:(WebView *)sender needsScrollNotifications:(id)notifications forFrame:(WebFrame *)frame {
573 return [webview_ webView:sender needsScrollNotifications:notifications forFrame:frame];
576 - (void) webView:(WebView *)sender didCommitLoadForFrame:(WebFrame *)frame {
577 return [webview_ webView:sender didCommitLoadForFrame:frame];
580 - (void) webView:(WebView *)sender didReceiveDocTypeForFrame:(WebFrame *)frame {
581 return [webview_ webView:sender didReceiveDocTypeForFrame:frame];
584 - (void) webView:(WebView *)sender didFinishLoadForFrame:(WebFrame *)frame {
585 if ([frame parentFrame] == nil) {
586 [self _finishLoading];
588 if (DOMDocument *document = [frame DOMDocument])
589 if (DOMNodeList<NSFastEnumeration> *bodies = [document getElementsByTagName:@"body"])
590 for (DOMHTMLBodyElement *body in bodies) {
591 DOMCSSStyleDeclaration *style([document getComputedStyle:body pseudoElement:nil]);
595 if (DOMCSSPrimitiveValue *color = static_cast<DOMCSSPrimitiveValue *>([style getPropertyCSSValue:@"background-color"])) {
596 if ([color primitiveType] == DOM_CSS_RGBCOLOR) {
597 DOMRGBColor *rgb([color getRGBColorValue]);
599 float red([[rgb red] getFloatValue:DOM_CSS_NUMBER]);
600 float green([[rgb green] getFloatValue:DOM_CSS_NUMBER]);
601 float blue([[rgb blue] getFloatValue:DOM_CSS_NUMBER]);
602 float alpha([[rgb alpha] getFloatValue:DOM_CSS_NUMBER]);
606 if (red == 0xc7 && green == 0xce && blue == 0xd5)
607 uic = [UIColor pinStripeColor];
610 colorWithRed:(red / 255)
618 [scroller_ setBackgroundColor:uic];
624 [scroller_ setBackgroundColor:[UIColor pinStripeColor]];
629 return [webview_ webView:sender didFinishLoadForFrame:frame];
632 - (void) webView:(WebView *)sender didFailProvisionalLoadWithError:(NSError *)error forFrame:(WebFrame *)frame {
633 if ([frame parentFrame] != nil)
635 [self _finishLoading];
637 [self loadURL:[NSURL URLWithString:[NSString stringWithFormat:@"%@?%@",
638 [[NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"error" ofType:@"html"]] absoluteString],
639 [[error localizedDescription] stringByAddingPercentEscapes]
645 - (void) webView:(WebView *)sender addMessageToConsole:(NSDictionary *)dictionary {
647 lprintf("Console:%s\n", [[dictionary description] UTF8String]);
651 - (id) initWithBook:(RVBook *)book {
652 if ((self = [super initWithBook:book]) != nil) {
655 struct CGRect bounds = [self bounds];
657 scroller_ = [[UIScroller alloc] initWithFrame:bounds];
658 [self addSubview:scroller_];
660 [scroller_ setShowBackgroundShadow:NO];
661 [scroller_ setFixedBackgroundPattern:YES];
662 [scroller_ setBackgroundColor:[UIColor pinStripeColor]];
664 [scroller_ setScrollingEnabled:YES];
665 [scroller_ setAdjustForContentSizeChange:YES];
666 [scroller_ setClipsSubviews:YES];
667 [scroller_ setAllowsRubberBanding:YES];
668 [scroller_ setScrollDecelerationFactor:0.99];
669 [scroller_ setDelegate:self];
671 CGRect webrect = [scroller_ bounds];
672 webrect.size.height = 0;
677 webview_ = [Documents_ lastObject];
678 if (webview_ != nil) {
679 webview_ = [webview_ retain];
680 webview = [webview_ webView];
681 [Documents_ removeLastObject];
682 [webview_ setFrame:webrect];
687 webview_ = [[UIWebDocumentView alloc] initWithFrame:webrect];
688 webview = [webview_ webView];
690 // XXX: this is terribly (too?) expensive
691 //[webview_ setDrawsBackground:NO];
692 [webview setPreferencesIdentifier:@"Cydia"];
694 [webview_ setTileSize:CGSizeMake(webrect.size.width, 500)];
696 [webview_ setAllowsMessaging:YES];
698 [webview_ setTilingEnabled:YES];
699 [webview_ setDrawsGrid:NO];
700 [webview_ setLogsTilingChanges:NO];
701 [webview_ setTileMinificationFilter:kCAFilterNearest];
702 [webview_ setDetectsPhoneNumbers:NO];
703 [webview_ setAutoresizes:YES];
705 [webview_ setMinimumScale:0.25f forDocumentTypes:0x10];
706 [webview_ setInitialScale:UIWebViewScalesToFitScale forDocumentTypes:0x10];
707 [webview_ setViewportSize:CGSizeMake(980, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x10];
709 [webview_ setViewportSize:CGSizeMake(320, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x2];
711 [webview_ setMinimumScale:1.0f forDocumentTypes:0x8];
712 [webview_ setInitialScale:UIWebViewScalesToFitScale forDocumentTypes:0x8];
713 [webview_ setViewportSize:CGSizeMake(320, UIWebViewGrowsAndShrinksToFitHeight) forDocumentTypes:0x8];
715 [webview_ _setDocumentType:0x4];
717 [webview_ setZoomsFocusedFormControl:YES];
718 [webview_ setContentsPosition:7];
719 [webview_ setEnabledGestures:0xa];
720 [webview_ setValue:[NSNumber numberWithBool:YES] forGestureAttribute:UIGestureAttributeIsZoomRubberBandEnabled];
721 [webview_ setValue:[NSNumber numberWithBool:YES] forGestureAttribute:UIGestureAttributeUpdatesScroller];
723 [webview_ setSmoothsFonts:YES];
725 [webview _setUsesLoaderCache:YES];
726 [webview setGroupName:@"Cydia"];
727 [webview _setLayoutInterval:0];
730 [webview_ setDelegate:self];
731 [webview_ setGestureDelegate:self];
732 [scroller_ addSubview:webview_];
734 //NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
736 CGSize indsize = [UIProgressIndicator defaultSizeForStyle:UIProgressIndicatorStyleMediumWhite];
737 indicator_ = [[UIProgressIndicator alloc] initWithFrame:CGRectMake(281, 12, indsize.width, indsize.height)];
738 [indicator_ setStyle:UIProgressIndicatorStyleMediumWhite];
740 Package *package([[Database sharedInstance] packageWithName:@"cydia"]);
741 NSString *application = package == nil ? @"Cydia" : [NSString
742 stringWithFormat:@"Cydia/%@",
744 ]; [webview setApplicationNameForUserAgent:application];
746 indirect_ = [[IndirectDelegate alloc] initWithDelegate:self];
748 [webview setFrameLoadDelegate:self];
749 [webview setResourceLoadDelegate:indirect_];
750 [webview setUIDelegate:self];
751 [webview setScriptDebugDelegate:self];
752 [webview setPolicyDelegate:self];
754 [self setAutoresizingMask:UIViewAutoresizingFlexibleHeight];
755 [scroller_ setAutoresizingMask:UIViewAutoresizingFlexibleHeight];
759 - (void) didFinishGesturesInView:(UIView *)view forEvent:(id)event {
760 [webview_ redrawScaledDocument];
763 - (void) _rightButtonClicked {
764 if (function_ == nil) {
768 WebView *webview([webview_ webView]);
769 WebFrame *frame([webview mainFrame]);
771 id _private(MSHookIvar<id>(webview, "_private"));
772 WebCore::Page *page(_private == nil ? NULL : MSHookIvar<WebCore::Page *>(_private, "page"));
773 WebCore::Settings *settings(page == NULL ? NULL : page->settings());
776 if (settings == NULL)
779 no = settings->JavaScriptCanOpenWindowsAutomatically();
780 settings->setJavaScriptCanOpenWindowsAutomatically(true);
783 [delegate_ clearFirstResponder];
784 JSObjectRef function([function_ JSObject]);
785 JSGlobalContextRef context([frame globalContext]);
786 JSObjectCallAsFunction(context, function, NULL, 0, NULL, NULL);
788 if (settings != NULL)
789 settings->setJavaScriptCanOpenWindowsAutomatically(no);
793 - (id) _rightButtonTitle {
794 return button_ != nil ? button_ : @"Reload";
797 - (id) rightButtonTitle {
798 return [self _loading] ? @"" : [self _rightButtonTitle];
801 - (UINavigationButtonStyle) rightButtonStyle {
802 if (style_ == nil) normal:
803 return UINavigationButtonStyleNormal;
804 else if ([style_ isEqualToString:@"Normal"])
805 return UINavigationButtonStyleNormal;
806 else if ([style_ isEqualToString:@"Back"])
807 return UINavigationButtonStyleBack;
808 else if ([style_ isEqualToString:@"Highlighted"])
809 return UINavigationButtonStyleHighlighted;
810 else if ([style_ isEqualToString:@"Destructive"])
811 return UINavigationButtonStyleDestructive;
815 - (NSString *) title {
816 return title_ == nil ? @"Loading" : title_;
819 - (NSString *) backButtonTitle {
823 - (void) setPageActive:(BOOL)active {
825 [indicator_ removeFromSuperview];
827 [[book_ navigationBar] addSubview:indicator_];
830 - (void) resetViewAnimated:(BOOL)animated {
833 - (void) setPushed:(bool)pushed {