Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #import "chrome/browser/ui/cocoa/applescript/tab_applescript.h" | 5 #import "chrome/browser/ui/cocoa/applescript/tab_applescript.h" |
| 6 | 6 |
| 7 #import <Carbon/Carbon.h> | 7 #import <Carbon/Carbon.h> |
| 8 #import <Foundation/NSAppleEventDescriptor.h> | 8 #import <Foundation/NSAppleEventDescriptor.h> |
| 9 | 9 |
| 10 #include "base/bind.h" | |
| 10 #include "base/file_path.h" | 11 #include "base/file_path.h" |
| 11 #include "base/logging.h" | 12 #include "base/logging.h" |
| 12 #import "base/memory/scoped_nsobject.h" | 13 #import "base/memory/scoped_nsobject.h" |
| 13 #include "base/sys_string_conversions.h" | 14 #include "base/sys_string_conversions.h" |
| 14 #include "base/utf_string_conversions.h" | 15 #include "base/utf_string_conversions.h" |
| 15 #include "chrome/browser/printing/print_view_manager.h" | 16 #include "chrome/browser/printing/print_view_manager.h" |
| 16 #include "chrome/browser/sessions/session_id.h" | 17 #include "chrome/browser/sessions/session_id.h" |
| 17 #include "chrome/browser/sessions/session_tab_helper.h" | 18 #include "chrome/browser/sessions/session_tab_helper.h" |
| 18 #include "chrome/browser/ui/cocoa/applescript/error_applescript.h" | 19 #include "chrome/browser/ui/cocoa/applescript/error_applescript.h" |
| 19 #include "chrome/common/url_constants.h" | 20 #include "chrome/common/url_constants.h" |
| 20 #include "content/public/browser/navigation_controller.h" | 21 #include "content/public/browser/navigation_controller.h" |
| 21 #include "content/public/browser/navigation_entry.h" | 22 #include "content/public/browser/navigation_entry.h" |
| 22 #include "content/public/browser/render_view_host.h" | 23 #include "content/public/browser/render_view_host.h" |
| 23 #include "content/public/browser/save_page_type.h" | 24 #include "content/public/browser/save_page_type.h" |
| 24 #include "content/public/browser/web_contents.h" | 25 #include "content/public/browser/web_contents.h" |
| 25 #include "content/public/browser/web_contents_delegate.h" | 26 #include "content/public/browser/web_contents_delegate.h" |
| 26 #include "googleurl/src/gurl.h" | 27 #include "googleurl/src/gurl.h" |
| 27 | 28 |
| 28 using content::NavigationController; | 29 using content::NavigationController; |
| 29 using content::NavigationEntry; | 30 using content::NavigationEntry; |
| 30 using content::OpenURLParams; | 31 using content::OpenURLParams; |
| 31 using content::RenderViewHost; | 32 using content::RenderViewHost; |
| 32 using content::Referrer; | 33 using content::Referrer; |
| 33 using content::WebContents; | 34 using content::WebContents; |
| 34 | 35 |
| 35 @interface AnyResultValue : NSObject { | 36 namespace { |
| 36 @private | |
| 37 scoped_nsobject<NSAppleEventDescriptor> descriptor; | |
| 38 } | |
| 39 - (id)initWithDescriptor:(NSAppleEventDescriptor*)desc; | |
| 40 - (NSAppleEventDescriptor *)scriptingAnyDescriptor; | |
| 41 @end | |
| 42 | 37 |
| 43 @implementation AnyResultValue | 38 NSAppleEventDescriptor* valueToDescriptor(const base::Value* value) { |
|
Avi (use Gerrit)
2013/01/03 22:31:43
A future CL (ETA a few days) will bring this funct
| |
| 44 | |
| 45 - (id)initWithDescriptor:(NSAppleEventDescriptor*)desc { | |
| 46 if (self = [super init]) { | |
| 47 descriptor.reset([desc retain]); | |
| 48 } | |
| 49 return self; | |
| 50 } | |
| 51 | |
| 52 - (NSAppleEventDescriptor *)scriptingAnyDescriptor { | |
| 53 return descriptor.get(); | |
| 54 } | |
| 55 | |
| 56 @end | |
| 57 | |
| 58 static NSAppleEventDescriptor* valueToDescriptor(Value* value) { | |
| 59 NSAppleEventDescriptor* descriptor = nil; | 39 NSAppleEventDescriptor* descriptor = nil; |
| 60 switch (value->GetType()) { | 40 switch (value->GetType()) { |
| 61 case Value::TYPE_NULL: | 41 case base::Value::TYPE_NULL: |
| 62 descriptor = [NSAppleEventDescriptor | 42 descriptor = [NSAppleEventDescriptor |
| 63 descriptorWithTypeCode:cMissingValue]; | 43 descriptorWithTypeCode:cMissingValue]; |
| 64 break; | 44 break; |
| 65 case Value::TYPE_BOOLEAN: { | 45 case base::Value::TYPE_BOOLEAN: { |
| 66 bool bool_value; | 46 bool bool_value; |
| 67 value->GetAsBoolean(&bool_value); | 47 value->GetAsBoolean(&bool_value); |
| 68 descriptor = [NSAppleEventDescriptor descriptorWithBoolean:bool_value]; | 48 descriptor = [NSAppleEventDescriptor descriptorWithBoolean:bool_value]; |
| 69 break; | 49 break; |
| 70 } | 50 } |
| 71 case Value::TYPE_INTEGER: { | 51 case base::Value::TYPE_INTEGER: { |
| 72 int int_value; | 52 int int_value; |
| 73 value->GetAsInteger(&int_value); | 53 value->GetAsInteger(&int_value); |
| 74 descriptor = [NSAppleEventDescriptor descriptorWithInt32:int_value]; | 54 descriptor = [NSAppleEventDescriptor descriptorWithInt32:int_value]; |
| 75 break; | 55 break; |
| 76 } | 56 } |
| 77 case Value::TYPE_DOUBLE: { | 57 case base::Value::TYPE_DOUBLE: { |
| 78 double double_value; | 58 double double_value; |
| 79 value->GetAsDouble(&double_value); | 59 value->GetAsDouble(&double_value); |
| 80 descriptor = [NSAppleEventDescriptor | 60 descriptor = [NSAppleEventDescriptor |
| 81 descriptorWithDescriptorType:typeIEEE64BitFloatingPoint | 61 descriptorWithDescriptorType:typeIEEE64BitFloatingPoint |
| 82 bytes:&double_value | 62 bytes:&double_value |
| 83 length:sizeof(double_value)]; | 63 length:sizeof(double_value)]; |
| 84 break; | 64 break; |
| 85 } | 65 } |
| 86 case Value::TYPE_STRING: { | 66 case base::Value::TYPE_STRING: { |
| 87 std::string string_value; | 67 std::string string_value; |
| 88 value->GetAsString(&string_value); | 68 value->GetAsString(&string_value); |
| 89 descriptor = [NSAppleEventDescriptor descriptorWithString: | 69 descriptor = [NSAppleEventDescriptor descriptorWithString: |
| 90 base::SysUTF8ToNSString(string_value)]; | 70 base::SysUTF8ToNSString(string_value)]; |
| 91 break; | 71 break; |
| 92 } | 72 } |
| 93 case Value::TYPE_BINARY: | 73 case base::Value::TYPE_BINARY: |
| 94 NOTREACHED(); | 74 NOTREACHED(); |
| 95 break; | 75 break; |
| 96 case Value::TYPE_DICTIONARY: { | 76 case base::Value::TYPE_DICTIONARY: { |
| 97 DictionaryValue* dictionary_value = static_cast<DictionaryValue*>(value); | 77 const base::DictionaryValue* dictionary_value = |
| 78 static_cast<const base::DictionaryValue*>(value); | |
| 98 descriptor = [NSAppleEventDescriptor recordDescriptor]; | 79 descriptor = [NSAppleEventDescriptor recordDescriptor]; |
| 99 NSAppleEventDescriptor* userRecord = [NSAppleEventDescriptor | 80 NSAppleEventDescriptor* userRecord = [NSAppleEventDescriptor |
| 100 listDescriptor]; | 81 listDescriptor]; |
| 101 for (DictionaryValue::key_iterator iter(dictionary_value->begin_keys()); | 82 for (DictionaryValue::key_iterator iter(dictionary_value->begin_keys()); |
| 102 iter != dictionary_value->end_keys(); ++iter) { | 83 iter != dictionary_value->end_keys(); ++iter) { |
| 103 Value* item; | 84 const base::Value* item; |
| 104 if (dictionary_value->Get(*iter, &item)) { | 85 if (dictionary_value->Get(*iter, &item)) { |
| 105 [userRecord insertDescriptor:[NSAppleEventDescriptor | 86 [userRecord insertDescriptor:[NSAppleEventDescriptor |
| 106 descriptorWithString:base::SysUTF8ToNSString(*iter)] atIndex:0]; | 87 descriptorWithString:base::SysUTF8ToNSString(*iter)] atIndex:0]; |
| 107 [userRecord insertDescriptor:valueToDescriptor(item) atIndex:0]; | 88 [userRecord insertDescriptor:valueToDescriptor(item) atIndex:0]; |
| 108 } | 89 } |
| 109 } | 90 } |
| 110 // Description of what keyASUserRecordFields does. | 91 // Description of what keyASUserRecordFields does. |
| 111 // http://www.mail-archive.com/cocoa-dev%40lists.apple.com/msg40149.html | 92 // http://www.mail-archive.com/cocoa-dev%40lists.apple.com/msg40149.html |
| 112 [descriptor setDescriptor:userRecord forKeyword:keyASUserRecordFields]; | 93 [descriptor setDescriptor:userRecord forKeyword:keyASUserRecordFields]; |
| 113 break; | 94 break; |
| 114 } | 95 } |
| 115 case Value::TYPE_LIST: { | 96 case base::Value::TYPE_LIST: { |
| 116 ListValue* list_value; | 97 const base::ListValue* list_value; |
| 117 value->GetAsList(&list_value); | 98 value->GetAsList(&list_value); |
| 118 descriptor = [NSAppleEventDescriptor listDescriptor]; | 99 descriptor = [NSAppleEventDescriptor listDescriptor]; |
| 119 for (unsigned i = 0; i < list_value->GetSize(); ++i) { | 100 for (unsigned i = 0; i < list_value->GetSize(); ++i) { |
| 120 Value* item; | 101 const base::Value* item; |
| 121 list_value->Get(i, &item); | 102 list_value->Get(i, &item); |
| 122 [descriptor insertDescriptor:valueToDescriptor(item) atIndex:0]; | 103 [descriptor insertDescriptor:valueToDescriptor(item) atIndex:0]; |
| 123 } | 104 } |
| 124 break; | 105 break; |
| 125 } | 106 } |
| 126 } | 107 } |
| 127 return descriptor; | 108 return descriptor; |
| 128 } | 109 } |
| 129 | 110 |
| 111 void ResumeAppleEventAndSendReply(NSAppleEventManagerSuspensionID suspension_id, | |
| 112 const base::Value* result_value) { | |
| 113 NSAppleEventDescriptor* result_descriptor = valueToDescriptor(result_value); | |
| 114 | |
| 115 NSAppleEventManager* manager = [NSAppleEventManager sharedAppleEventManager]; | |
| 116 NSAppleEventDescriptor* reply_event = | |
| 117 [manager replyAppleEventForSuspensionID:suspension_id]; | |
| 118 [reply_event setParamDescriptor:result_descriptor | |
| 119 forKeyword:keyDirectObject]; | |
| 120 [manager resumeWithSuspensionID:suspension_id]; | |
| 121 } | |
| 122 | |
| 123 } // namespace | |
| 124 | |
| 130 @interface TabAppleScript() | 125 @interface TabAppleScript() |
| 131 @property (nonatomic, copy) NSString* tempURL; | 126 @property (nonatomic, copy) NSString* tempURL; |
| 132 @end | 127 @end |
| 133 | 128 |
| 134 @implementation TabAppleScript | 129 @implementation TabAppleScript |
| 135 | 130 |
| 136 @synthesize tempURL = tempURL_; | 131 @synthesize tempURL = tempURL_; |
| 137 | 132 |
| 138 - (id)init { | 133 - (id)init { |
| 139 if ((self = [super init])) { | 134 if ((self = [super init])) { |
| 140 SessionID session; | 135 SessionID session; |
| 141 SessionID::id_type futureSessionIDOfTab = session.id() + 1; | 136 SessionID::id_type futureSessionIDOfTab = session.id() + 1; |
| 142 // Holds the SessionID that the new tab is going to get. | 137 // Holds the SessionID that the new tab is going to get. |
| 143 scoped_nsobject<NSNumber> numID( | 138 scoped_nsobject<NSNumber> numID( |
| 144 [[NSNumber alloc] | 139 [[NSNumber alloc] initWithInt:futureSessionIDOfTab]); |
| 145 initWithInt:futureSessionIDOfTab]); | |
| 146 [self setUniqueID:numID]; | 140 [self setUniqueID:numID]; |
| 147 } | 141 } |
| 148 return self; | 142 return self; |
| 149 } | 143 } |
| 150 | 144 |
| 151 - (void)dealloc { | 145 - (void)dealloc { |
| 152 [tempURL_ release]; | 146 [tempURL_ release]; |
| 153 [super dealloc]; | 147 [super dealloc]; |
| 154 } | 148 } |
| 155 | 149 |
| 156 - (id)initWithWebContents:(content::WebContents*)webContents { | 150 - (id)initWithWebContents:(content::WebContents*)webContents { |
| 157 if (!webContents) { | 151 if (!webContents) { |
| 158 [self release]; | 152 [self release]; |
| 159 return nil; | 153 return nil; |
| 160 } | 154 } |
| 161 | 155 |
| 162 if ((self = [super init])) { | 156 if ((self = [super init])) { |
| 163 // It is safe to be weak, if a tab goes away (eg user closing a tab) | 157 // It is safe to be weak; if a tab goes away (e.g. the user closes a tab) |
| 164 // the applescript runtime calls tabs in AppleScriptWindow and this | 158 // the AppleScript runtime calls tabs in AppleScriptWindow and this |
| 165 // particular tab is never returned. | 159 // particular tab is never returned. |
| 166 webContents_ = webContents; | 160 webContents_ = webContents; |
| 167 SessionTabHelper* session_tab_helper = | 161 SessionTabHelper* session_tab_helper = |
| 168 SessionTabHelper::FromWebContents(webContents); | 162 SessionTabHelper::FromWebContents(webContents); |
| 169 scoped_nsobject<NSNumber> numID( | 163 scoped_nsobject<NSNumber> numID( |
| 170 [[NSNumber alloc] initWithInt:session_tab_helper->session_id().id()]); | 164 [[NSNumber alloc] initWithInt:session_tab_helper->session_id().id()]); |
| 171 [self setUniqueID:numID]; | 165 [self setUniqueID:numID]; |
| 172 } | 166 } |
| 173 return self; | 167 return self; |
| 174 } | 168 } |
| 175 | 169 |
| 176 - (void)setWebContents:(content::WebContents*)webContents { | 170 - (void)setWebContents:(content::WebContents*)webContents { |
| 177 DCHECK(webContents); | 171 DCHECK(webContents); |
| 178 // It is safe to be weak, if a tab goes away (eg user closing a tab) | 172 // It is safe to be weak; if a tab goes away (e.g. the user closes a tab) |
| 179 // the applescript runtime calls tabs in AppleScriptWindow and this | 173 // the AppleScript runtime calls tabs in AppleScriptWindow and this |
| 180 // particular tab is never returned. | 174 // particular tab is never returned. |
| 181 webContents_ = webContents; | 175 webContents_ = webContents; |
| 182 SessionTabHelper* session_tab_helper = | 176 SessionTabHelper* session_tab_helper = |
| 183 SessionTabHelper::FromWebContents(webContents); | 177 SessionTabHelper::FromWebContents(webContents); |
| 184 scoped_nsobject<NSNumber> numID( | 178 scoped_nsobject<NSNumber> numID( |
| 185 [[NSNumber alloc] initWithInt:session_tab_helper->session_id().id()]); | 179 [[NSNumber alloc] initWithInt:session_tab_helper->session_id().id()]); |
| 186 [self setUniqueID:numID]; | 180 [self setUniqueID:numID]; |
| 187 | 181 |
| 188 if ([self tempURL]) | 182 if ([self tempURL]) |
| 189 [self setURL:[self tempURL]]; | 183 [self setURL:[self tempURL]]; |
| (...skipping 205 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 395 } | 389 } |
| 396 } | 390 } |
| 397 | 391 |
| 398 - (id)handlesExecuteJavascriptScriptCommand:(NSScriptCommand*)command { | 392 - (id)handlesExecuteJavascriptScriptCommand:(NSScriptCommand*)command { |
| 399 RenderViewHost* view = webContents_->GetRenderViewHost(); | 393 RenderViewHost* view = webContents_->GetRenderViewHost(); |
| 400 if (!view) { | 394 if (!view) { |
| 401 NOTREACHED(); | 395 NOTREACHED(); |
| 402 return nil; | 396 return nil; |
| 403 } | 397 } |
| 404 | 398 |
| 399 NSAppleEventManager* manager = [NSAppleEventManager sharedAppleEventManager]; | |
| 400 NSAppleEventManagerSuspensionID suspensionID = | |
| 401 [manager suspendCurrentAppleEvent]; | |
| 402 content::RenderViewHost::JavascriptResultCallback callback = | |
| 403 base::Bind(&ResumeAppleEventAndSendReply, suspensionID); | |
| 404 | |
| 405 string16 script = base::SysNSStringToUTF16( | 405 string16 script = base::SysNSStringToUTF16( |
| 406 [[command evaluatedArguments] objectForKey:@"javascript"]); | 406 [[command evaluatedArguments] objectForKey:@"javascript"]); |
| 407 Value* value = view->ExecuteJavascriptAndGetValue(string16(), script); | 407 view->ExecuteJavascriptInWebFrameCallbackResult(string16(), // frame_xpath |
| 408 NSAppleEventDescriptor* descriptor = valueToDescriptor(value); | 408 script, |
| 409 return [[[AnyResultValue alloc] initWithDescriptor:descriptor] autorelease]; | 409 callback); |
| 410 | |
| 411 return nil; | |
| 410 } | 412 } |
| 411 | 413 |
| 412 @end | 414 @end |
| OLD | NEW |