ConversationVC.mm 8.46 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
 *
 *  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>
26
#import <QuartzCore/QuartzCore.h>
27

28
// LRC
29 30

#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"
Kateryna Kostiuk's avatar
Kateryna Kostiuk committed
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
@interface ConversationVC () {
41 42

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

47
    __unsafe_unretained IBOutlet NSButton* sentContactRequestButton;
Kateryna Kostiuk's avatar
Kateryna Kostiuk committed
48
    IBOutlet MessagesVC* messagesViewVC;
49

50
    IBOutlet NSLayoutConstraint *titleCenteredConstraint;
51
    IBOutlet NSLayoutConstraint* titleTopConstraint;
52 53 54 55 56

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

57 58
    RingWindowController* delegate;

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

@end

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

70 71
@implementation ConversationVC

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

Andreas Traczyk's avatar
Andreas Traczyk committed
81 82 83 84 85 86 87 88 89 90 91 92
-(void) clearData {
    cachedConv_ = nil;
    convUid_ = "";
    convModel_ = nil;

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

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

    if (cachedConv_ != nil)
        return cachedConv_;

101
    auto convQueue = convModel_->allFilteredConversations();
102

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

    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;
114
    [self clearData];
115 116 117 118 119 120 121 122 123 124 125
    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.
126 127 128 129 130
    QObject::disconnect(modelSortedConnection_);
    QObject::disconnect(filterChangedConnection_);
    QObject::disconnect(newConversationConnection_);
    QObject::disconnect(conversationRemovedConnection_);
    modelSortedConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::modelSorted,
131 132 133
                                          [self](){
                                              cachedConv_ = nil;
                                          });
134
    filterChangedConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::filterChanged,
135 136 137
                                            [self](){
                                                cachedConv_ = nil;
                                            });
138
    newConversationConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::newConversation,
139 140 141
                                                [self](){
                                                    cachedConv_ = nil;
                                                });
142
    conversationRemovedConnection_ = QObject::connect(convModel_, &lrc::api::ConversationModel::conversationRemoved,
143 144 145
                                                [self](){
                                                    cachedConv_ = nil;
                                                });
146 147 148 149 150 151 152 153

    auto* conv = [self getCurrentConversation];

    if (conv == nil)
        return;

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

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

160 161 162
    [conversationID setHidden:hideBestId];
    [titleCenteredConstraint setActive:hideBestId];
    [titleTopConstraint setActive:!hideBestId];
163 164
    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)];
165 166
}

167 168 169 170 171 172 173 174 175
- (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
{
176 177
    auto* conv = [self getCurrentConversation];
    convModel_->placeCall(conv->uid);
178 179
}

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

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

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

197 198
# pragma mark private IN/OUT animations

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

206 207 208 209 210 211 212 213 214 215 216 217 218 219
    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];
}

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

Andreas Traczyk's avatar
Andreas Traczyk committed
226 227
    [self clearData];

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

233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251
    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