使用 Parse 从一个用户向另一个用户发送推送通知

Posted

技术标签:

【中文标题】使用 Parse 从一个用户向另一个用户发送推送通知【英文标题】:Sending a Push Notification from one user to another user with Parse 【发布时间】:2013-12-17 05:22:56 【问题描述】:

我已经构建了一个类似于 Snapchat 的消息应用程序 - 一个用户可以发送另一个用户的图片。我正在尝试向应用程序添加推送通知,以便当用户A 向用户B 发送消息时,用户B 会收到“来自用户A 的新消息”的推送通知。

我已经研究了好几个小时了,我觉得我已经很接近了。

我正在尝试使用 Parse 发送推送通知。我希望它像这样工作:当 UserA 向 UserB 发送消息时,UserB 也会收到一条推送通知,上面写着“来自 UserA 的新消息”。我成功地使用 Parse 网站向使用该应用程序的设备发送推送通知,但无法从应用程序内成功发送推送通知(当用户发送消息时)到接收用户的设备。

推送通知显然已成功发送,因为我的 Parse 帐户显示了我已发送的消息。但是,实际上没有消息到达预期的设备,并且推送通知列表显示每个推送通知有 0 个订阅者。

我可以单击其中一个来查看详细信息。

另外,我正在使用分发/生产配置文件和证书。

这是我在 UserA 向 UserB 发送消息后使用发送推送通知的代码 - message 对象是已上传到 Parse 的消息,messageRecipients 是消息正在发送的用户发送至:

// Send Push Notification to recipients

NSArray *messageRecipients = [message objectForKey:@"recipientIds"];

PFQuery *pushQuery = [PFInstallation query];
                    [pushQuery whereKey:@"owner" containedIn:messageRecipients];

PFPush *push = [[PFPush alloc] init];
[push setQuery:pushQuery];
[push setMessage:[NSString stringWithFormat: @"New Message from %@!",  [PFUser currentUser].username]];
[push sendPushInBackground];

这是我的 AppDelegate.m 相关方法:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions

    [Parse setApplicationId:@"[This is where my app Id is]"
                  clientKey:@"[This is where client Id is]"];
    [self customizeUserInterface];
    [application registerForRemoteNotificationTypes:UIRemoteNotificationTypeAlert|UIRemoteNotificationTypeBadge|UIRemoteNotificationTypeSound];

    return YES;


- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken

    [PFPush storeDeviceToken:deviceToken];
    [PFPush subscribeToChannelInBackground:@""];


- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error

    NSLog(@"Did fail to register for push, %@", error);


- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo

    [PFPush handlePush:userInfo];

我还在 Parse.com 论坛上提交了一个帖子:https://parse.com/questions/sending-a-push-notification-from-one-user-to-another-user

我有什么遗漏或做错了吗?

编辑:我现在可以在我的 Parse 帐户中看到订阅者,但我实际上并没有在我的设备上收到推送通知。当我尝试从 Parse 网站发送推送通知测试时也是如此。

【问题讨论】:

“所有者”是否存储在您的安装对象中?你确定 [message objectForKey:@"recipientIds"] 返回正确的数组吗? 是的,我认为是。我记录了数组,它返回了正确的数组。 而“owner”其实就在Parse网站你的数据浏览器的安装表中? 肯定在安装表中,类型是字符串。 @tagabek 你有想过这个吗?我遇到了完全相同的问题 【参考方案1】:

我的搜索不断返回到这里,但这里没有真正为我说明。所以这就是我的工作方式:

在我的 AppDelegate.m 中有:

- (void)applicationDidBecomeActive:(UIApplication *)application


    PFUser *currentUser = [PFUser currentUser];
    if (currentUser) 
        //save the installation
        PFInstallation *currentInstallation = [PFInstallation currentInstallation];
        currentInstallation[@"installationUser"] = [[PFUser currentUser]objectId];
        // here we add a column to the installation table and store the current user’s ID
        // this way we can target specific users later

        // while we’re at it, this is a good place to reset our app’s badge count
        // you have to do this locally as well as on the parse server by updating
        // the PFInstallation object
        if (currentInstallation.badge != 0) 
            currentInstallation.badge = 0;
            [currentInstallation saveInBackgroundWithBlock:^(BOOL succeeded, NSError *error) 
                if (error) 
                    // Handle error here with an alert…
                
                else 
                    // only update locally if the remote update succeeded so they always match
                    [[UIApplication sharedApplication] setApplicationIconBadgeNumber:0];
                    NSLog(@"updated badge");
                
            ];
        
     else 

        [PFUser logOut];
        // show the signup screen here....
    

在我发送推送的 viewController 中:

myViewController.h

@property (nonatomic, strong) NSMutableArray *recipients; // declare the array we'll use to store our recipients

myViewController.m

- (void)viewDidLoad

    [super viewDidLoad];
    self.recipients = [[NSMutableArray alloc] init]; // initialize the array we'll use to hold our recipients



// in another part of the code (not shown here) we set up a tableView with all of the current user's friends in it
// when the user taps a row in that tableView we add or remove the selected friend from our recipients list
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath

    [self.tableView deselectRowAtIndexPath:indexPath animated:NO];

    UITableViewCell *cell = [self.tableView cellForRowAtIndexPath:indexPath];
    PFUser *user = [self.friends objectAtIndex:indexPath.row];

    if (cell.accessoryType == UITableViewCellAccessoryNone) 
        cell.accessoryType = UITableViewCellAccessoryCheckmark;
        [self.recipients addObject:user.objectId]; //  user selected a recipient, add them to the array
    
    else 
        cell.accessoryType = UITableViewCellAccessoryNone;
        [self.recipients removeObject:user.objectId]; //  user de-selected a recipient, remove them from the array
    






- (void)uploadMessage 

    UIImage *newImage = [self resizeImage:self.image toWidth:640.0f andHeight:960.0f];
    NSData *fileData= UIImageJPEGRepresentation(newImage, 1.0);
    NSString *fileName= @"image.jpg";;
    NSString *fileType= @"image";

    PFFile *file = [PFFile fileWithName:fileName data:fileData];

    [file saveInBackgroundWithBlock:^(BOOL succeeded, NSError *error) 
        if (error) 
                // Handle error here with an alert…
        
        else 
            PFObject *message = [PFObject objectWithClassName:@"Messages"];
            [message setObject:file forKey:@"file"];
            [message setObject:fileType forKey:@"fileType"];
            [message setObject:self.recipients forKey:@"recipientIds"];
                  // self.recipients is an NSMutableArray of the objectIds for each 
                  // user the message will go to 
            [message setObject:[[PFUser currentUser] objectId] forKey:@"senderId"];
            [message setObject:[[PFUser currentUser] username] forKey:@"senderName"];
            [message saveInBackgroundWithBlock:^(BOOL succeeded, NSError *error) 
                if (error) 
                    // Handle error here with an alert…
                
                else 
                    // Everything was successful! Reset UI… do other stuff
                    // Here’s where we will send the push
                    //set our options
                    NSDictionary *data = [NSDictionary dictionaryWithObjectsAndKeys:
                                          @“Ne messages available!!”, @"alert",
                                          @"Increment", @"badge",
                                          nil];

                    // Now we’ll need to query all saved installations to find those of our recipients 
                    // Create our Installation query using the self.recipients array we already have
                    PFQuery *pushQuery = [PFInstallation query];
                    [pushQuery whereKey:@"installationUser" containedIn:self.recipients];

                    // Send push notification to our query
                    PFPush *push = [[PFPush alloc] init];
                    [push setQuery:pushQuery];
                    [push setData:data];
                    [push sendPushInBackground];
                
            ];
        
    ];

【讨论】:

如果您在登录时创建新用户,应用程序将崩溃。您必须检查 applicationDidBecomeActive 中的 [PFUser currentUser] 是否不为零,然后在 didLoginUser 中设置 currentInstallation[@"installationUser"] @NikolayDerkach 不错,修复了我检查 currentUser 的答案 @DelightedD0D [[PFUser currentUser]objectId] 获得空值。任何想法:( @BhaveshNai 你确定有用户登录吗?你打电话给[PFUser logInWithUsernameInBackground:@"myname" password:@"mypass"......] @DelightedD0D 感谢您的回复。还有一件事,如何发送多个用户通知,如小组讨论。在此先感谢:)【参考方案2】:

我想我有答案。我遇到了同样的问题,我刚刚解决了。

当您为所有者设置值时,您应该使用“[PFUser currentUser].objectId”,而不是 [PFUser currentUser]。后者为您提供了指向所有者的指针,但在这种情况下我们需要一个字符串来设置 Push 查询。

当我们第一次将所有者设置为安装时,我们应该将 objectId 作为字符串设置为所有者,而不是像下面的 [PFUser currentUser]。

[currentInstallation setObject:[PFUser currentUser].objectId forKey:@"owner"];

稍后我们可以将所有者(字符串)设置为我们的 pushQuery。

【讨论】:

嗨,尼克,谢谢。当我将其更改为该值时,我收到此错误:“错误:密钥所有者的类型无效,应为 *_User,但得到字符串”。 您遇到该错误是因为您尝试将字符串分配给非字符串列“所有者”;您可以在解析中删除“所有者”列并创建另一个列,如“userId”,最重要的是,给它一个字符串类型。并确保您的设备使用新的 userId 字符串保存 PFInstallation。 我也卡在这里了:如果创建的字段具有错误的数据类型,则需要先删除该列,然后才能尝试插入新数据。 我也被困在这里,我将用户 ID 作为字符串存储在我的安装表中,并将其传递给我的 Push 查询。它似乎出现在在线推送浏览器中,但就像在原始问题中显示发送了 0 一样。有什么方法可以测试 Push 查询吗? @user379468 您是否将确切的 userId 分配给 pushQuery 或只是 [PFUser currentUser]?这是不同的。还有另一种测试方法。尝试从 Parse 推送控制台推送。选择段并从您的安装中选择 userId 列,并“等于”您的 userId(您要发送的用户)。【参考方案3】:

鉴于您迄今为止发布的内容,我不确定为什么您的代码不适合您。似乎您的安装表不包含“所有者”列,或者它包含一个“所有者”列,其中的值/类型与您的 [message objectForKey:@"recipientIds"] 数组中包含的值/类型不同;但是,以防万一您需要/想要尝试备份方法,这是另一种使用 Parse 将推送通知从一台设备发送到另一台设备的方法:

首先将接收推送的用户/用户组订阅到他们自己的唯一频道(除了主频道),例如

[PFPush subscribeToChannelInBackground:taylors_channel];

然后从发送推送的设备,将频道设置为接收者的频道:

PFPush *push = [[PFPush alloc] init];
[push setChannel:taylors_channel];
[push setMessage:[NSString stringWithFormat: @"New Message from %@!",  [PFUser currentUser].username]];
[push sendPushInBackground];

此外,您还可以使用[push setChannels:channel_array]为多个peer设置同一个频道或一次向多个频道发送消息;

【讨论】:

【参考方案4】:

为了详细说明 djshiow 的响应,一旦您保存了当前安装。

PFQuery * pushQuery = [PFInstallation query];
PFUser * userReceivingPush; 
[pushQuery whereKey:@"owner" equalTo:userReceivingPush]; 

NSString * alert = [NSString stringWithFormat:@"You have a new message from %@!", [PFUser currentUser].username];
NSDictionary *data = [NSDictionary dictionaryWithObjectsAndKeys:
                                  alert, @"alert",
                                  @"default", @"sound",
                                  @"Increment", @"badge",
                                  nil];
[PFPush sendPushDataToQueryInBackground:pushQuery withData:data block:^(BOOL succeeded, NSError *error) 
                if (!error) 

                
                else 
                
            ];

【讨论】:

【参考方案5】:

您是否尝试将所有者存储在安装中。即。

- (void)application:(UIApplication *)application
        didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken


    PFInstallation *currentInstallation = [PFInstallation currentInstallation];
    [currentInstallation setDeviceTokenFromData:deviceToken];
    [currentInstallation setObject:[PFUser currentUser] forKey:@"owner"];
    [currentInstallation saveInBackground];

发件人:click here

【讨论】:

以上是关于使用 Parse 从一个用户向另一个用户发送推送通知的主要内容,如果未能解决你的问题,请参考以下文章

使用 Parse 和 Swift 从用户向用户发送推送通知时遇到问题

Swift 从用户向用户发送推送通知 Firebase

Parse.com - 从 Parse.com 向 iOS 应用程序发送消息,不像推送通知

是否可以根据用户偏好(时间)从解析中发送推送通知?

如何使用firebase在android studio中从一个用户向另一个用户发送通知[重复]

使用 Parse Sdk 从应用推送通知