然後通過File > New > Target菜單給給工程添加壹個Target,選擇Action Extension:
在
創建Action擴展時需要指定壹個Aciton類型,Apple提供了兩種Action擴展的類型模板。壹種是有用戶界面的類型,包含壹個
UIViewController和壹個Storeboard文件,可以自定義顯示界面和行為。另壹種是不帶用戶界面的類型,這種類型只允許我們處理來自
Host應用的請求。
現在我們在工程中就可以看到剛才創建的Action擴展NoteAppExtension,它包含兩個主要的文件,壹個是Action.js,另壹個是ActionRequestHandler.swift:
我們來看看這兩個文件的作用。Action.js文件用來實現和處理瀏覽器中請求的邏輯,在本文的例子中,它主要實現用戶在瀏覽器中選中文本並發送到我們的應用中。ActionRequestHandler.swift用來處理Host應用發送的請求和參數。
在實現邏輯之前我們需要設置壹下擴展的屬性,打開Info.plist文件將NSExtensionActivationSupportsWebURLWithMaxCount屬性設置為1,該設置讓擴展知道我們需要請求壹個URL。
我們Action.js文件中有如下內容:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
var Action = function() {};
Action.prototype = {
run: function(arguments) {
// Here, you can run code that modifies the document and/or prepares
// things to pass to your action's native code.
// We will not modify anything, but will pass the body's background
// style to the native code.
var selected = "No Text Selected";
if (window.getSelection) {
selected = window.getSelection().getRangeAt(0).toString();
} else {
selected = document.getSelection().getRangeAt(0).toString();
}
arguments.completionFunction({"args" : selected});
},
finalize: function(arguments) {
// This method is run after the native code completes.
// We'll see if the native code has passed us a new background style,
// and set it on the body.
alert(arguments["message"])
}
};
var ExtensionPreprocessingJS = new Action
Safari
與Action擴展的交互就是通過Action.js文件中的run和finalize這兩個方法實現的。當我們在Safari中使用Action擴展時
就會調用run方法,它能讓我們在該方法中操作當前Safari顯示頁面的DOM元素。當Action擴展處理完邏輯向Safari返回信息時會調用
finalize方法,在我們的例子中,我們通過
self.extensionContext!.completeRequestReturningItems(nil,
completionHandler:
nil)這段代碼向Safari返回信息。該方法的第壹個參數就是要返回的信息,它會將信息傳給Action.js文件,然後通過js代碼操作HTML。
如果第壹個參數傳入nil,那就意味著不會調用Action.js文件中的finalize方法。run和finalize這兩個方法的參數
arguments都包含著壹些信息,只不過壹個是來自與HTML,壹個來自ActionRequestHandler文件。
壹定要記住:我們必須要實例化ExtensionPreprocessingJS這個全局變量,因為它是Safari和Action擴展之間的橋梁。
我們的ActionRequestHandler文件內容如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
class ActionRequestHandler: NSObject, NSExtensionRequestHandling {
var extensionContext: NSExtensionContext?
func beginRequestWithExtensionContext(context: NSExtensionContext!) {
self.extensionContext = context
let identifierType = NSString(format: kUTTypePropertyList, NSUTF8StringEncoding)
for (item: NSExtensionItem) in context.inputItems as [NSExtensionItem] {
for (itemProvider: NSItemProvider) in item.attachments as [NSItemProvider] {
if itemProvider.hasItemConformingToTypeIdentifier(identifierType) {
itemProvider.loadItemForTypeIdentifier(identifierType, options: nil, completionHandler: {(item, error) in
let dictionary = item as NSDictionary
dispatch_async(dispatch_get_main_queue(), {
self.itemLoadCompletedWithPreprocessingResults(dictionary[NSExtensionJavaScriptPreprocessingResultsKey] as NSDictionary)
})
})
}
}
}
}
func itemLoadCompletedWithPreprocessingResults(javaScriptPreprocessingResults: NSDictionary) {
if let text = javaScriptPreprocessingResults["args"] as? String {
let userDefaults = NSUserDefaults(suiteName: "group.name")
userDefaults.setValue(text, forKey: "note")
userDefaults.synchronize()
self.doneWithResults(["message": "Successfully added to the note app"])
}
}
func doneWithResults(resultsForJavaScriptFinalizeArg: NSDictionary?) {
if let resultsForJavaScriptFinalize = resultsForJavaScriptFinalizeArg {
let identifierType = NSString(format: kUTTypePropertyList, NSUTF8StringEncoding)
// Construct an NSExtensionItem of the appropriate type to return our
// results dictionary in.
// These will be used as the arguments to the JavaScript finalize()
// method.
var resultsDictionary = [NSExtensionJavaScriptFinalizeArgumentKey: resultsForJavaScriptFinalize]
var resultsProvider = NSItemProvider(item: resultsDictionary, typeIdentifier: identifierType)
var resultsItem = NSExtensionItem()
resultsItem.attachments = [resultsProvider]
// Signal that we're complete, returning our results.
self.extensionContext!.completeRequestReturningItems([resultsItem], completionHandler: nil)
} else {
// We still need to signal that we're done even if we have nothing to
// pass back.
self.extensionContext!.completeRequestReturningItems(nil, completionHandler: nil)
}
// Don't hold on to this after we finished with it.
self.extensionContext = nil
}
}
現在我們可以運行壹下我們的應用,然後打開Safari,在Action選項中開啟我們的Action擴展:
然後我們就可以在Action欄中看到我們的擴展了:
我們使用Safari隨便打開壹個含有文字的頁面,選中壹段文字,然後打開Action欄,點擊NoteApp擴展,此時我們的擴展就會將選中的這段文字發送給我們的應用,形成壹條新的重要信息。
以上只是壹個簡單的Aciton擴展的例子,但我們可以由此延伸出更多有用、有創意的功能,讓我們的生活更加美好。