-
判断是否需要启动一个程序
if(nil == [[NSRunningApplication runningApplicationsWithBundleIdentifier:HELPER_ID] firstObject]) { dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ //start startApplication([NSURL fileURLWithPath:loginItem], NSWorkspaceLaunchWithoutActivation); }); }
判断启动附带参数
if(YES == [[[NSProcessInfo processInfo] arguments] containsObject:CMDLINE_FLAG_WELCOME])
激活自己的app:
[NSApp activateIgnoringOtherApps:YES];
-
显示主窗口
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];
安装文件时需要提升权限,使用帮助任务,servicemanagement.framework xpc通信的方式来完成root文件安装
-
托盘
statusBarItem = NSStatusBar.system.statusItem(withLength: NSStatusItem.squareLength) statusBarItem.button?.image = NSImage(named: "settings")
7.不在Dock中显示,infor.plist
添加Application is agent (UIElement)
并设置为YES
NSWindow 能够自定义修改,隐藏几乎所有功能。都采用自定义提供相似功能
-
进程间通信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; }
-
安装程序设计思路。
- X-installer.app 分两个部分构成。1个普通窗口应用程序,一个高权限帮助程序。
主程序通过smjobbless吊起帮助程序,帮助程序接下来复制相应的运行程序到缓存空间,修改用户组所有者,然后运行这个程序。特殊情况下帮助程序复制主程序然后运行,通过传参区分是命令行运行还是直接运行。如果需要清晰一般会复制另外一个程序或是直接在帮助程序运行。因为帮助程序与主程序分别运行在不同进程,这里需要进程间通信,简单的采用XPC,用于触发并获得反馈。
- X-installer.app 分两个部分构成。1个普通窗口应用程序,一个高权限帮助程序。
多进程程序就是多个项目的集合,把所有需要的项目都添加到一个工作空间中,安装程序的scheme build 中添加其他需要使用的其他项目的产品。编译时就会都编译到一个项目里面了。安装的帮助程序会放到Contents/Library下,其他的在Contents/Resources下。