iOS ShareExtension

使用系统分享。将Safari中的网页分享给微信中的好友。


0.gif

1.新建ShareExtension。

1.png
2.png
4.png
5.png
6.png
3.png

2.配置Share Extension,允许发送的数据类型,url,image,mp3,mp4,pdf,word,excel,ppt。


7.png

3.处理Share Extension中的数据。
Share Extension中默认都会有一个数据展现的UI界面。该界面继承SLComposeServiceViewController这个类型,如:

@interface ShareViewController : SLComposeServiceViewController

@end
10.gif

一般采用自定义控制器:

@interface ShareViewController : SLComposeServiceViewController

@end
8.png
11.gif

4.从inputItems中获取数据。

 [self.extensionContext.inputItems enumerateObjectsUsingBlock:^(NSExtensionItem *  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
       
       if (obj.attributedContentText.string.length > 0)
       {
           self.contentText = obj.attributedContentText.string;
       }
       
       [obj.attachments enumerateObjectsUsingBlock:^(NSItemProvider *  _Nonnull itemProvider, NSUInteger idx, BOOL * _Nonnull stop) {
            if ([itemProvider hasItemConformingToTypeIdentifier:@"public.url"])
            {
                [itemProvider loadItemForTypeIdentifier:@"public.url" options:nil completionHandler:^(id<NSSecureCoding>  _Nullable item, NSError * _Null_unspecified error) {
                    
                    if ([(NSObject *)item isKindOfClass:[NSURL class]])
                    {
                        self.url = (NSURL *)item;

                        NSInteger preValue = self.flag.integerValue;

                        if ([self.url isFileURL])
                        {
                            self.flag = [NSNumber numberWithInteger:(preValue|url_file)];
                        }
                        else
                        {
                            self.flag = [NSNumber numberWithInteger:(preValue|url_flag)];
                        }
                    }
                    
                    [self refreshView];
                }];
            }
           
           if ([itemProvider hasItemConformingToTypeIdentifier:@"public.image"])
            {
                [itemProvider loadItemForTypeIdentifier:@"public.image" options:nil completionHandler:^(id item, NSError *error) {
                    
                    if ([item isKindOfClass:[NSURL class]])
                    {
                        [self.arrImagePath addObject:item];
                        NSInteger preValue = self.flag.integerValue;
                        self.flag = [NSNumber numberWithInteger:(preValue | url_image)];
                        [self refreshView];
                        count ++;
                    }
                    else if ([item isKindOfClass:[UIImage class]] && !self.thumb)
                    {
                        self.thumb = (UIImage *)item;
                        NSInteger preValue = self.flag.integerValue;
                        self.flag = [NSNumber numberWithInteger:(preValue | url_image)];
                        [self refreshView];
                        count ++;
                    }
                }];
            }
           
            if ([itemProvider hasItemConformingToTypeIdentifier:(__bridge NSString *)kUTTypeText])
            {
                NSInteger preValue = self.flag.integerValue;
                self.flag = [NSNumber numberWithInteger:(preValue|url_text)];
                
                [itemProvider loadItemForTypeIdentifier:(__bridge NSString *)kUTTypeText options:nil completionHandler:^(id item, NSError *error) {
                    
                    if ([item isKindOfClass:[NSString class]])
                    {
                        NSString *str = (NSString *)item;
                        
                        if ([str containsString:@"http://"] || [str containsString:@"https://"] || [str containsString:@"file:///"])
                        {
                            if (!self.url)
                            {
                                self.url = [NSURL URLWithString:str];
                                
                                if ([self.url isFileURL])
                                {
                                    self.flag = [NSNumber numberWithInteger:(preValue|url_file)];
                                }
                                else
                                {
                                    self.flag = [NSNumber numberWithInteger:(preValue|url_flag)];
                                }
                            }
                        }
                        else
                        {
                            [self.text appendString:str];
                            [self.text appendString:@"\n"];
                        }
                    }

                    [self refreshView];
                }];
            }
           
           if ([itemProvider hasItemConformingToTypeIdentifier:@"public.movie"])
           {
               [itemProvider loadItemForTypeIdentifier:@"public.movie" options:nil completionHandler:^(id<NSSecureCoding>  _Nullable item, NSError * _Null_unspecified error)
               {
                   NSInteger preValue = self.flag.integerValue;
                   NSURL *fileurl = (NSURL *)item;
                   if ([fileurl isFileURL])
                   {
                       self.flag = [NSNumber numberWithInteger:(preValue | url_file)];
                       self.url = fileurl;
                       [self refreshView];
                   }
               }];
           }
           
        }];
       
    }];

上面的例子中遍历了extensionContext的inputItems数组中所有NSExtensionItem对象,然后从这些对象中遍历attachments数组中的所有NSItemProvider对象。匹配第一个包含public.url标识的附件(具体要匹配什么资源,数量是多少皆有自己的业务所决定)。注意:[self.extensionContext completeRequestReturningItems:@[] completionHandler:nil];这行代码,主要是使到视图控制器不被关闭,等到实现相应的处理后再进行调用该方法,对分享视图进行关闭。调用该方法则回到宿主App。

5.传递Share Extension中的数据。有个App Groups功能可以据此传递数据。


WX20190210-223758.png

WX20190210-224011.png

1.依据写文件传递数据。例如:要分享的App储存登录信息。

 //获取分组的共享目录
      NSURL *groupURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.cn.com.mengniu.oa.sharextension"];
      NSURL *fileURL = [groupURL URLByAppendingPathComponent:@"login.txt"];
      if (success) {
        [@"isLogin" writeToURL:fileURL atomically:YES encoding:NSUTF8StringEncoding error:nil];
      } else {
        [@"isNotLogin" writeToURL:fileURL atomically:YES encoding:NSUTF8StringEncoding error:Nil];
      }
      
//获取储存在App Groups中的登录信息。
  NSURL *groupURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.cn.com.mengniu.oa.sharextension"];
  NSURL *fileURL = [groupURL URLByAppendingPathComponent:@"login.txt"];
  NSString *isLoginStatus = [NSString stringWithContentsOfURL:fileURL encoding:NSUTF8StringEncoding error:nil];

2.依据NSUserDefaults,储存数据,试了几次没有取成功过。

   NSUserDefaults *userDefaults = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.xxx.sharextension"];
  if (![userDefaults objectForKey:@"isLogin"])
  {
   UIAlertController* alert = [UIAlertController alertControllerWithTitle:@"温馨提示"
                                                                   message:@"请登录"
                                                            preferredStyle:UIAlertControllerStyleAlert];
    
    UIAlertAction* defaultAction = [UIAlertAction actionWithTitle:@"确定"
                                                            style:UIAlertActionStyleDefault
                                                          handler:^(UIAlertAction * action) {
                                                            UIResponder* responder = self;
                                                            while ((responder = [responder nextResponder]) != nil)
                                                            {
                                                              if([responder respondsToSelector:@selector(openURL:)] == YES)
                                                              {
                                                                [responder performSelector:@selector(openURL:) withObject:[NSURL URLWithString:@"sharefile://"]];
                                                              }
                                                            }
                                                          }];
    UIAlertAction* cancelAction = [UIAlertAction actionWithTitle:@"取消"
                                                           style:UIAlertActionStyleCancel
                                                         handler:^(UIAlertAction * action) {}];
    [alert addAction:cancelAction];
    [alert addAction:defaultAction];
    [self presentViewController:alert animated:YES completion:nil];
  }

会报错:

[User Defaults] Couldn't read values in CFPrefsPlistSource<0x1c010e340> (Domain:
 group.cn.com.mengniu.oa.sharextension, User: kCFPreferencesAnyUser, ByHost: 
 Yes, Container: (null), Contents Need Refresh: Yes): Using kCFPreferencesAnyUser with a container is only allowed for System Containers, 
 detaching from cfprefsd

网上说在储存App Groups时要添加Team ID。之后取值时虽然不会再报错,但取出来的值为nil。

6.其实苹果官方除了Today Extension外,其他Extension是不提供跳转接口的。所以这里总结的是两种非正常的方式。

1.在Share Extension中无法获取到UIApplication对象,则通过拼接字符串获取。

    NSURL *destinationURL = [NSURL URLWithString:[NSString stringWithFormat:@"sharefile://%@",saveFilePath]];
//     Get "UIApplication" class name through ASCII Character codes.
    NSString *className = [[NSString alloc] initWithData:[NSData dataWithBytes:(unsigned char []){0x55, 0x49, 0x41, 0x70, 0x70, 0x6C, 0x69, 0x63, 0x61, 0x74, 0x69, 0x6F, 0x6E} length:13] encoding:NSASCIIStringEncoding];
    if (NSClassFromString(className)) {
      id object = [NSClassFromString(className) performSelector:@selector(sharedApplication)];
      [object performSelector:@selector(openURL:) withObject:destinationURL];
    }
    

2.这种方式主要实现原理是通过响应链找到Host App的UIApplication对象,通过该对象调用openURL方法返回自己的应用。

    UIResponder *responder = self;
    while ((responder = [responder nextResponder]) != nil) {
    if ([responder respondsToSelector:@selector(openURL:)] == YES) {
        [responder performSelector:@selector(openURL:) withObject:[NSURL URLWithString:@"sharefile://"]];
      }
    }
12.gif

7.未登录的处理。登录成功后先写文件储存登录信息到App Groups中,退出登录后,删除储存在App Groups中的登录信息。到ShareViewController中先判断是否登录,若未登录,则弹窗提示登录不再弹起发送框。

1.登录成功,则保存登录信息。

      //获取分组的共享目录
      NSURL *groupURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.cn.com.mengniu.oa.sharextension"];
      NSURL *fileURL = [groupURL URLByAppendingPathComponent:@"login.txt"];
      if (success) {
        [@"isLogin" writeToURL:fileURL atomically:YES encoding:NSUTF8StringEncoding error:nil];
      } else {
        [@"isNotLogin" writeToURL:fileURL atomically:YES encoding:NSUTF8StringEncoding error:Nil];
      }
    });

2.退出登录或未登录,则清空已经保存的登录信息。

  //获取分组的共享目录
  NSURL *groupURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.cn.com.mengniu.oa.sharextension"];
  NSURL *fileURL = [groupURL URLByAppendingPathComponent:@"login.txt"];
  [@"isNotLogin" writeToURL:fileURL atomically:YES encoding:NSUTF8StringEncoding error:nil];
  
  
  NSURL *groupURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.cn.com.mengniu.oa.sharextension"];
  NSURL *fileURL = [groupURL URLByAppendingPathComponent:@"login.txt"];
  NSString *isLoginStatus = [NSString stringWithContentsOfURL:fileURL encoding:NSUTF8StringEncoding error:nil];
 //如果未登录提示登录
  if (isLoginStatus && [isLoginStatus isEqualToString:@"isNotLogin"]) {
    UIAlertController* alert = [UIAlertController alertControllerWithTitle:@"请先登录办随,再分享"
                                                                   message:nil
                                                            preferredStyle:UIAlertControllerStyleAlert];
    
    UIAlertAction* defaultAction = [UIAlertAction actionWithTitle:@"确定"
                                                            style:UIAlertActionStyleDefault
                                                          handler:^(UIAlertAction * action) {
                                                            UIResponder* responder = self;
                                                            while ((responder = [responder nextResponder]) != nil)
                                                            {
                                                              if([responder respondsToSelector:@selector(openURL:)] == YES)
                                                              {
                                                                [responder performSelector:@selector(openURL:) withObject:[NSURL URLWithString:@"sharefile://"]];
                                                              }
                                                              [self.extensionContext completeRequestReturningItems:@[] completionHandler:nil];
                                                            }
                                                          }];
    UIAlertAction* cancelAction = [UIAlertAction actionWithTitle:@"取消"
                                                           style:UIAlertActionStyleCancel
                                                         handler:^(UIAlertAction * action) {
                                                           [self.extensionContext completeRequestReturningItems:@[] completionHandler:nil];
                                                         }];
    [alert addAction:cancelAction];
    [alert addAction:defaultAction];
    [self presentViewController:alert animated:YES completion:nil];
  }
  else//已登录加载发送框
  {
    [self.view addSubview:container];
  }

9.gif

8.在ShareExtension中处理逻辑代码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容