Source code

Revision control

Copy as Markdown

Other Tools

/* -*- Mode: c++; tab-width: 2; indent-tabs-mode: nil; -*- */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#import <Cocoa/Cocoa.h>
#include "nsMacSharingService.h"
#include "jsapi.h"
#include "js/Array.h" // JS::NewArrayObject
#include "js/PropertyAndElement.h" // JS_SetElement, JS_SetProperty
#include "nsCocoaUtils.h"
#include "mozilla/MacStringHelpers.h"
NS_IMPL_ISUPPORTS(nsMacSharingService, nsIMacSharingService)
NSString* const remindersServiceName =
@"com.apple.reminders.RemindersShareExtension";
// These are some undocumented constants also used by Safari
// to let us open the preferences window
NSString* const extensionPrefPanePath =
@"/System/Library/PreferencePanes/Extensions.prefPane";
const UInt32 openSharingSubpaneDescriptorType = 'ptru';
NSString* const openSharingSubpaneActionKey = @"action";
NSString* const openSharingSubpaneActionValue = @"revealExtensionPoint";
NSString* const openSharingSubpaneProtocolKey = @"protocol";
NSString* const openSharingSubpaneProtocolValue = @"com.apple.share-services";
// Expose the id so we can pass reference through to JS and back
@interface NSSharingService (ExposeName)
- (id)name;
@end
// Filter providers that we do not want to expose to the user, because they are
// duplicates or do not work correctly within the context
static bool ShouldIgnoreProvider(NSString* aProviderName) {
return [aProviderName
isEqualToString:@"com.apple.share.System.add-to-safari-reading-list"];
}
// Clean up the activity once the share is complete
@interface SharingServiceDelegate : NSObject <NSSharingServiceDelegate> {
NSUserActivity* mShareActivity;
}
- (void)cleanup;
@end
@implementation SharingServiceDelegate
- (id)initWithActivity:(NSUserActivity*)activity {
self = [super init];
mShareActivity = [activity retain];
return self;
}
- (void)cleanup {
[mShareActivity resignCurrent];
[mShareActivity invalidate];
[mShareActivity release];
mShareActivity = nil;
}
- (void)sharingService:(NSSharingService*)sharingService
didShareItems:(NSArray*)items {
[self cleanup];
}
- (void)sharingService:(NSSharingService*)service
didFailToShareItems:(NSArray*)items
error:(NSError*)error {
[self cleanup];
}
- (void)dealloc {
[mShareActivity release];
[super dealloc];
}
@end
static NSString* NSImageToBase64(const NSImage* aImage) {
CGImageRef cgRef = [aImage CGImageForProposedRect:nil context:nil hints:nil];
NSBitmapImageRep* bitmapRep =
[[NSBitmapImageRep alloc] initWithCGImage:cgRef];
[bitmapRep setSize:[aImage size]];
NSData* imageData =
[bitmapRep representationUsingType:NSBitmapImageFileTypePNG
properties:@{}];
NSString* base64Encoded = [imageData base64EncodedStringWithOptions:0];
[bitmapRep release];
return [NSString stringWithFormat:@"data:image/png;base64,%@", base64Encoded];
}
static void SetStrAttribute(JSContext* aCx, JS::Rooted<JSObject*>& aObj,
const char* aKey, NSString* aVal) {
nsAutoString strVal;
mozilla::CopyNSStringToXPCOMString(aVal, strVal);
JS::Rooted<JSString*> title(aCx, JS_NewUCStringCopyZ(aCx, strVal.get()));
JS::Rooted<JS::Value> attVal(aCx, JS::StringValue(title));
JS_SetProperty(aCx, aObj, aKey, attVal);
}
nsresult nsMacSharingService::GetSharingProviders(
const nsAString& aPageUrl, JSContext* aCx,
JS::MutableHandle<JS::Value> aResult) {
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
NSURL* url = nsCocoaUtils::ToNSURL(aPageUrl);
if (!url) {
// aPageUrl is not a valid URL.
return NS_ERROR_FAILURE;
}
NSArray* sharingService = [NSSharingService sharingServicesForItems:@[ url ]];
int32_t serviceCount = 0;
JS::Rooted<JSObject*> array(aCx, JS::NewArrayObject(aCx, 0));
for (NSSharingService* currentService in sharingService) {
if (ShouldIgnoreProvider([currentService name])) {
continue;
}
JS::Rooted<JSObject*> obj(aCx, JS_NewPlainObject(aCx));
SetStrAttribute(aCx, obj, "name", [currentService name]);
SetStrAttribute(aCx, obj, "menuItemTitle", currentService.menuItemTitle);
SetStrAttribute(aCx, obj, "image", NSImageToBase64(currentService.image));
JS::Rooted<JS::Value> element(aCx, JS::ObjectValue(*obj));
JS_SetElement(aCx, array, serviceCount++, element);
}
aResult.setObject(*array);
return NS_OK;
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
}
NS_IMETHODIMP
nsMacSharingService::OpenSharingPreferences() {
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
NSURL* prefPaneURL = [NSURL fileURLWithPath:extensionPrefPanePath
isDirectory:YES];
NSDictionary* args = @{
openSharingSubpaneActionKey : openSharingSubpaneActionValue,
openSharingSubpaneProtocolKey : openSharingSubpaneProtocolValue
};
NSData* data = [NSPropertyListSerialization
dataWithPropertyList:args
format:NSPropertyListXMLFormat_v1_0
options:0
error:nil];
NSAppleEventDescriptor* descriptor = [[NSAppleEventDescriptor alloc]
initWithDescriptorType:openSharingSubpaneDescriptorType
data:data];
[[NSWorkspace sharedWorkspace] openURLs:@[ prefPaneURL ]
withAppBundleIdentifier:nil
options:NSWorkspaceLaunchAsync
additionalEventParamDescriptor:descriptor
launchIdentifiers:NULL];
[descriptor release];
return NS_OK;
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
}
NS_IMETHODIMP
nsMacSharingService::ShareUrl(const nsAString& aServiceName,
const nsAString& aPageUrl,
const nsAString& aPageTitle) {
NS_OBJC_BEGIN_TRY_BLOCK_RETURN;
NSString* serviceName = nsCocoaUtils::ToNSString(aServiceName);
NSURL* pageUrl = nsCocoaUtils::ToNSURL(aPageUrl);
NSString* pageTitle = nsCocoaUtils::ToNSString(aPageTitle);
NSSharingService* service =
[NSSharingService sharingServiceNamed:serviceName];
// Reminders fetch its data from an activity, not the share data
if ([[service name] isEqual:remindersServiceName]) {
NSUserActivity* shareActivity = [[NSUserActivity alloc]
initWithActivityType:NSUserActivityTypeBrowsingWeb];
if ([pageUrl.scheme hasPrefix:@"http"]) {
[shareActivity setWebpageURL:pageUrl];
}
[shareActivity setEligibleForHandoff:NO];
[shareActivity setTitle:pageTitle];
[shareActivity becomeCurrent];
// Pass ownership of shareActivity to shareDelegate, which will release the
// activity once sharing has completed.
SharingServiceDelegate* shareDelegate =
[[SharingServiceDelegate alloc] initWithActivity:shareActivity];
[shareActivity release];
[service setDelegate:shareDelegate];
[shareDelegate release];
}
// Twitter likes the the title as an additional share item
NSArray* toShare = [[service name] isEqual:NSSharingServiceNamePostOnTwitter]
? @[ pageUrl, pageTitle ]
: @[ pageUrl ];
[service setSubject:pageTitle];
[service performWithItems:toShare];
return NS_OK;
NS_OBJC_END_TRY_BLOCK_RETURN(NS_ERROR_FAILURE);
}