MacOS常用技术点

  1. 判断是否需要启动一个程序

     if(nil == [[NSRunningApplication runningApplicationsWithBundleIdentifier:HELPER_ID] firstObject])
     {
         dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
         ^{
            //start
            startApplication([NSURL fileURLWithPath:loginItem], NSWorkspaceLaunchWithoutActivation);
         });
     }
    
  2. 判断启动附带参数
    if(YES == [[[NSProcessInfo processInfo] arguments] containsObject:CMDLINE_FLAG_WELCOME])

  3. 激活自己的app:[NSApp activateIgnoringOtherApps:YES];

  4. 显示主窗口

     if(nil == self.rulesWindowController)
     {
         //alloc
         rulesWindowController = [[RulesWindowController alloc] initWithWindowNibName:@"Rules"];
     }
    
     //center
     [self.rulesWindowController.window center];
     
     //show it
     [self.rulesWindowController showWindow:self];
     
     //make it key window
     [[self.rulesWindowController window] makeKeyAndOrderFront:self];
    
  5. 安装文件时需要提升权限,使用帮助任务,servicemanagement.framework xpc通信的方式来完成root文件安装

  6. 托盘

     statusBarItem = NSStatusBar.system.statusItem(withLength: NSStatusItem.squareLength)
     statusBarItem.button?.image = NSImage(named: "settings")
    

7.不在Dock中显示,infor.plist添加Application is agent (UIElement) 并设置为YES

  1. NSWindow 能够自定义修改,隐藏几乎所有功能。都采用自定义提供相似功能

  2. 进程间通信CFMessagePortRef

      CFDataRef onRecvMsgCallBack(CFMessagePortRef local,SInt32 msgid,CFDataRef cfData,void *info)
     {
     NSLog(@"local = %@",local);
     NSLog(@"msgid = %d",msgid);
     NSString *strData = nil;
     if (cfData)
     {
             const UInt8  * recvedMsg = CFDataGetBytePtr(cfData);
             strData = [NSString stringWithCString:(char *)recvedMsg encoding:NSUTF8StringEncoding];
             NSLog(@"receive message:%@",strData);
     }
     
     //生成返回数据
     NSString *returnString = [NSString stringWithFormat:@"i have receive:%@",strData];
     const char* cStr = [returnString UTF8String];
     NSUInteger ulen = [returnString lengthOfBytesUsingEncoding:NSUTF8StringEncoding];
     CFDataRef sgReturn = CFDataCreate(NULL, (UInt8 *)cStr, ulen);
     
     return sgReturn;
     }
     
     - (void)start
     {
     if (0 != mMsgPortListenner && CFMessagePortIsValid(mMsgPortListenner)) {
             CFMessagePortInvalidate(mMsgPortListenner);
     }
     
     mMsgPortListenner = CFMessagePortCreateLocal(kCFAllocatorDefault, CFSTR(LOCAL_MACH_PORT_NAME), onRecvMsgCallBack, NULL, NULL);
     CFRunLoopSourceRef source = CFMessagePortCreateRunLoopSource(kCFAllocatorDefault, mMsgPortListenner, 0);
     CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopCommonModes);
     NSLog(@"start");
     }
    

    发送代码

     - (NSString *)sendMessageToDameonWith:(id)msgInfo msgID:(SInt32)msgid
     {
     CFMessagePortRef bRemote = CFMessagePortCreateRemote(kCFAllocatorDefault, CFSTR(MSG_PORT));
     if (nil == bRemote) {
             NSLog(@"bRemote create failed");
             return nil;
     }
     
     NSString *msg = [NSString stringWithFormat:@"%@",msgInfo];
     NSLog(@"send msg is :%@",msg);
     const char *message = [msg UTF8String];
     CFDataRef data,recvData = nil;
     data = CFDataCreate(NULL, (UInt8 *)message, strlen(message));
     
     /* 发送消息 */
     CFMessagePortSendRequest(bRemote, msgid, data, 0, 100, kCFRunLoopDefaultMode, &recvData);
     if (nil == recvData) {
             NSLog(@"recvData data is nil.");
             CFRelease(data);
             CFMessagePortInvalidate(bRemote);
             CFRelease(bRemote);
    
             return nil;
     }
     
     const UInt8 *recvedMsg = CFDataGetBytePtr(recvData);
     if (nil == recvedMsg) {
             NSLog(@"receive data err.");
             CFRelease(data);
             CFMessagePortInvalidate(bRemote);
             CFRelease(bRemote);
             return nil;
     }
     
     NSString *strMsg = [NSString stringWithCString:(char *)recvedMsg encoding:NSUTF8StringEncoding];
     NSLog(@"%@",strMsg);
     
     
     
     CFRelease(data);
     CFMessagePortInvalidate(bRemote);
     CFRelease(bRemote);
     CFRelease(recvData);
     return nil;
     }
    
  3. 安装程序设计思路。

    1. X-installer.app 分两个部分构成。1个普通窗口应用程序,一个高权限帮助程序。
      主程序通过smjobbless吊起帮助程序,帮助程序接下来复制相应的运行程序到缓存空间,修改用户组所有者,然后运行这个程序。特殊情况下帮助程序复制主程序然后运行,通过传参区分是命令行运行还是直接运行。如果需要清晰一般会复制另外一个程序或是直接在帮助程序运行。因为帮助程序与主程序分别运行在不同进程,这里需要进程间通信,简单的采用XPC,用于触发并获得反馈。
  4. 多进程程序就是多个项目的集合,把所有需要的项目都添加到一个工作空间中,安装程序的scheme build 中添加其他需要使用的其他项目的产品。编译时就会都编译到一个项目里面了。安装的帮助程序会放到Contents/Library下,其他的在Contents/Resources下。

你可能感兴趣的:(MacOS常用技术点)