ConversationVC.mm 8.49 KB
Newer Older
1
/*
2
 *  Copyright (C) 2016-2018 Savoir-faire Linux Inc.
3
 *  Author: Alexandre Lision <alexandre.lision@savoirfairelinux.com>
4
 *  Author: Anthony Léonard <anthony.leonard@savoirfairelinux.com>
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to the Free Software
 *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301 USA.
 */

#import "ConversationVC.h"

#import <qstring.h>
#import <QPixmap>
#import <QtMacExtras/qmacfunctions.h>

27
// LRC
28 29 30
#import <globalinstances.h>

#import "views/IconButton.h"
31
#import "views/HoverButton.h"
32 33 34
#import "views/IMTableCellView.h"
#import "views/NSColor+RingTheme.h"
#import "delegates/ImageManipulationDelegate.h"
35
#import "MessagesVC.h"
36
#import "utils.h"
37
#import "RingWindowController.h"
Kateryna Kostiuk's avatar
Kateryna Kostiuk committed
38
#import "NSString+Extensions.h"
39 40 41

#import <QuartzCore/QuartzCore.h>

42
@interface ConversationVC () {
43 44

    __unsafe_unretained IBOutlet NSTextField* conversationTitle;
45
    __unsafe_unretained IBOutlet NSTextField *conversationID;
46
    __unsafe_unretained IBOutlet HoverButton *addContactButton;
47
    __unsafe_unretained IBOutlet NSLayoutConstraint* sentContactRequestWidth;
Kateryna Kostiuk's avatar
Kateryna Kostiuk committed
48

49
    __unsafe_unretained IBOutlet NSButton* sentContactRequestButton;
50
    IBOutlet MessagesVC* messagesViewVC;
51

52
    IBOutlet NSLayoutConstraint *titleCenteredConstraint;
53
    IBOutlet NSLayoutConstraint* titleTopConstraint;
54 55 56 57 58

    std::string convUid_;
    const lrc::api::conversation::Info* cachedConv_;
    lrc::api::ConversationModel* convModel_;

59 60
    RingWindowController* delegate;

61
    // All those connections are needed to invalidate cached conversation as pointer
62
    // may not be referencing the same conversation anymore
63
    QMetaObject::Connection modelSortedConnection_, filterChangedConnection_, newConversationConnection_, conversationRemovedConnection_;
64 65 66 67
}

@end

Kateryna Kostiuk's avatar
Kateryna Kostiuk committed
68 69 70 71
NSInteger const MEESAGE_MARGIN = 21;
NSInteger const SEND_PANEL_DEFAULT_HEIGHT = 60;
NSInteger const SEND_PANEL_MAX_HEIGHT = 120;

72 73
@implementation ConversationVC

74 75 76 77 78 79 80 81 82
- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil delegate:(RingWindowController*) mainWindow
{
    if (self = [self initWithNibName:nibNameOrNil bundle:nibBundleOrNil])
    {
        delegate = mainWindow;
    }
    return self;
}

83 84 85 86 87 88 89 90 91 92 93 94
-(void) clearData {
    cachedConv_ = nil;
    convUid_ = "";
    convModel_ = nil;

    [messagesViewVC clearData];
    QObject::disconnect(modelSortedConnection_);
    QObject::disconnect(filterChangedConnection_);
    QObject::disconnect(newConversationConnection_);
    QObject::disconnect(conversationRemovedConnection_);
}

95 96 97 98 99 100 101 102
-(const lrc::api::conversation::Info*) getCurrentConversation
{
    if (convModel_ == nil || convUid_.empty())
        return nil;

    if (cachedConv_ != nil)
        return cachedConv_;

103
    auto convQueue = convModel_->allFilteredConversations();
104

105
    auto it = getConversationFromUid(convUid_, *convModel_);
106 107 108 109 110 111 112 113 114 115

    if (it != convQueue.end())
        cachedConv_ = &(*it);

    return cachedConv_;
}

-(void) setConversationUid:(const std::string)convUid model:(lrc::api::ConversationModel *)model {
    if (convUid_ == convUid && convModel_ == model)
        return;
116
    [self clearData];
117 118 119 120 121 122 123 124 125 126 127
    cachedConv_ = nil;
    convUid_ = convUid;
    convModel_ = model;

    [messagesViewVC setConversationUid:convUid_ model:convModel_];

    if (convUid_.empty() || convModel_ == nil)
        return;

    // Signals tracking changes in conversation list, we need them as cached conversation can be invalid
    // after a reordering.
128 129 130 131 132
    QObject::disconnect(modelSortedConnection_);
    QObject::disconnect(filterChangedConnection_);
    QObject::disconnect(newConversationConnection_);
    QObject::disconnect(conversationRemovedConnection_);
    modelSortedConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::modelSorted,
133 134 135
                                          [self](){
                                              cachedConv_ = nil;
                                          });
136
    filterChangedConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::filterChanged,
137 138 139
                                            [self](){
                                                cachedConv_ = nil;
                                            });
140
    newConversationConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::newConversation,
141 142 143
                                                [self](){
                                                    cachedConv_ = nil;
                                                });
144
    conversationRemovedConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::conversationRemoved,
145 146 147
                                                [self](){
                                                    cachedConv_ = nil;
                                                });
148 149 150 151 152 153 154 155

    auto* conv = [self getCurrentConversation];

    if (conv == nil)
        return;

    // Setup UI elements according to new conversation
    NSString* bestName = bestNameForConversation(*conv, *convModel_);
156
    NSString* bestId = bestIDForConversation(*conv, *convModel_);
157
    [conversationTitle setStringValue: bestName];
158
    [conversationID setStringValue: bestId];
159

160
    BOOL hideBestId = [bestNameForConversation(*conv, *convModel_) isEqualTo:bestIDForConversation(*conv, *convModel_)];
161

162 163 164
    [conversationID setHidden:hideBestId];
    [titleCenteredConstraint setActive:hideBestId];
    [titleTopConstraint setActive:!hideBestId];
165 166
    auto accountType = convModel_->owner.profileInfo.type;
    [addContactButton setHidden:((convModel_->owner.contactModel->getContact(conv->participants[0]).profileInfo.type != lrc::api::profile::Type::TEMPORARY) || accountType == lrc::api::profile::Type::SIP)];
167 168
}

169 170 171 172 173 174 175 176 177
- (void) initFrame
{
    [self.view setFrame:self.view.superview.bounds];
    [self.view setHidden:YES];
    self.view.layer.position = self.view.frame.origin;
}

- (IBAction)placeCall:(id)sender
{
178 179
    auto* conv = [self getCurrentConversation];
    convModel_->placeCall(conv->uid);
180 181
}

182 183 184 185 186 187 188 189 190 191 192 193
- (IBAction)placeAudioCall:(id)sender
{
    auto* conv = [self getCurrentConversation];
    convModel_->placeAudioOnlyCall(conv->uid);
}

- (IBAction)addContact:(id)sender
{
    auto* conv = [self getCurrentConversation];
    convModel_->makePermanent(conv->uid);
}

194
- (IBAction)backPressed:(id)sender {
195
    [delegate rightPanelClosed];
196
    [self hideWithAnimation:false];
197 198
}

199 200
# pragma mark private IN/OUT animations

201
-(void) showWithAnimation:(BOOL)animate
202
{
203 204 205 206 207
    if (!animate) {
        [self.view setHidden:NO];
        return;
    }

208 209 210 211 212 213 214 215 216 217 218 219 220 221
    CGRect frame = CGRectOffset(self.view.superview.bounds, -self.view.superview.bounds.size.width, 0);
    [self.view setHidden:NO];

    [CATransaction begin];
    CABasicAnimation *animation = [CABasicAnimation animationWithKeyPath:@"position"];
    [animation setFromValue:[NSValue valueWithPoint:frame.origin]];
    [animation setToValue:[NSValue valueWithPoint:self.view.superview.bounds.origin]];
    [animation setDuration:0.2f];
    [animation setTimingFunction:[CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseIn]];
    [self.view.layer addAnimation:animation forKey:animation.keyPath];

    [CATransaction commit];
}

222
-(void) hideWithAnimation:(BOOL)animate
223 224 225 226 227
{
    if(self.view.frame.origin.x < 0) {
        return;
    }

228 229
    [self clearData];

230 231 232 233 234
    if (!animate) {
        [self.view setHidden:YES];
        return;
    }

235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253
    CGRect frame = CGRectOffset(self.view.frame, -self.view.frame.size.width, 0);
    [CATransaction begin];
    CABasicAnimation *animation = [CABasicAnimation animationWithKeyPath:@"position"];
    [animation setFromValue:[NSValue valueWithPoint:self.view.frame.origin]];
    [animation setToValue:[NSValue valueWithPoint:frame.origin]];
    [animation setDuration:0.2f];
    [animation setTimingFunction:[CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseOut]];

    [CATransaction setCompletionBlock:^{
        [self.view setHidden:YES];
    }];
    [self.view.layer addAnimation:animation forKey:animation.keyPath];

    [self.view.layer setPosition:frame.origin];
    [CATransaction commit];
}


@end