| /* |
| * Copyright (C) 2015-2016 Savoir-faire Linux Inc. |
| * Author: Alexandre Lision <alexandre.lision@savoirfairelinux.com> |
| * |
| * 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 "SmartViewVC.h" |
| |
| //Qt |
| #import <QtMacExtras/qmacfunctions.h> |
| #import <QPixmap> |
| #import <QIdentityProxyModel> |
| #import <QItemSelectionModel> |
| |
| //LRC |
| #import <recentmodel.h> |
| #import <callmodel.h> |
| #import <call.h> |
| #import <uri.h> |
| #import <itemdataroles.h> |
| #import <namedirectory.h> |
| #import <accountmodel.h> |
| #import <account.h> |
| #import <person.h> |
| #import <contactmethod.h> |
| #import <globalinstances.h> |
| #import <phonedirectorymodel.h> |
| #import <AvailableAccountModel.h> |
| |
| #import "QNSTreeController.h" |
| #import "delegates/ImageManipulationDelegate.h" |
| #import "views/HoverTableRowView.h" |
| #import "PersonLinkerVC.h" |
| #import "views/IconButton.h" |
| #import "views/RingOutlineView.h" |
| #import "views/ContextualTableCellView.h" |
| |
| @interface SmartViewVC () <NSOutlineViewDelegate, NSPopoverDelegate, ContextMenuDelegate, ContactLinkedDelegate, KeyboardShortcutDelegate> { |
| |
| QNSTreeController *treeController; |
| NSPopover* addToContactPopover; |
| |
| //UI elements |
| __unsafe_unretained IBOutlet RingOutlineView* smartView; |
| __unsafe_unretained IBOutlet NSSearchField* searchField; |
| |
| /* Pending ring usernames lookup for the search entry */ |
| QMetaObject::Connection usernameLookupConnection; |
| } |
| |
| @end |
| |
| @implementation SmartViewVC |
| |
| @synthesize tabbar; |
| |
| // Tags for views |
| NSInteger const IMAGE_TAG = 100; |
| NSInteger const DISPLAYNAME_TAG = 200; |
| NSInteger const DETAILS_TAG = 300; |
| NSInteger const CALL_BUTTON_TAG = 400; |
| NSInteger const TXT_BUTTON_TAG = 500; |
| NSInteger const CANCEL_BUTTON_TAG = 600; |
| NSInteger const RING_ID_LABEL = 700; |
| NSInteger const PRESENCE_TAG = 800; |
| |
| - (void)awakeFromNib |
| { |
| NSLog(@"INIT SmartView VC"); |
| |
| treeController = [[QNSTreeController alloc] initWithQModel:RecentModel::instance().peopleProxy()]; |
| [treeController setAvoidsEmptySelection:NO]; |
| [treeController setChildrenKeyPath:@"children"]; |
| |
| [smartView bind:@"content" toObject:treeController withKeyPath:@"arrangedObjects" options:nil]; |
| [smartView bind:@"sortDescriptors" toObject:treeController withKeyPath:@"sortDescriptors" options:nil]; |
| [smartView bind:@"selectionIndexPaths" toObject:treeController withKeyPath:@"selectionIndexPaths" options:nil]; |
| [smartView setTarget:self]; |
| [smartView setAction:@selector(selectRow:)]; |
| [smartView setDoubleAction:@selector(placeCall:)]; |
| |
| [smartView setContextMenuDelegate:self]; |
| [smartView setShortcutsDelegate:self]; |
| |
| QObject::connect(RecentModel::instance().peopleProxy(), |
| &QAbstractItemModel::dataChanged, |
| [self](const QModelIndex &topLeft, const QModelIndex &bottomRight) { |
| for(int row = topLeft.row() ; row <= bottomRight.row() ; ++row) |
| { |
| [smartView reloadDataForRowIndexes:[NSIndexSet indexSetWithIndex:row] |
| columnIndexes:[NSIndexSet indexSetWithIndex:0]]; |
| } |
| }); |
| |
| QObject::connect(RecentModel::instance().selectionModel(), |
| &QItemSelectionModel::currentChanged, |
| [=](const QModelIndex ¤t, const QModelIndex &previous) { |
| if(!current.isValid()) { |
| [smartView deselectAll:nil]; |
| return; |
| } |
| |
| auto proxyIdx = RecentModel::instance().peopleProxy()->mapFromSource(current); |
| if (proxyIdx.isValid()) { |
| [treeController setSelectionQModelIndex:proxyIdx]; |
| [tabbar selectTabViewItemAtIndex:0]; |
| [smartView scrollRowToVisible:proxyIdx.row()]; |
| } |
| }); |
| |
| QObject::connect(RecentModel::instance().peopleProxy(), |
| &QAbstractItemModel::rowsInserted, |
| [=](const QModelIndex &parent, int first, int last) { |
| Q_UNUSED(parent) |
| Q_UNUSED(first) |
| Q_UNUSED(last) |
| [smartView scrollRowToVisible:0]; |
| }); |
| |
| QObject::connect(AvailableAccountModel::instance().selectionModel(), |
| &QItemSelectionModel::currentChanged, |
| [self](const QModelIndex& idx){ |
| [self clearSearchField]; |
| }); |
| |
| [self.view setWantsLayer:YES]; |
| [self.view setLayer:[CALayer layer]]; |
| [self.view.layer setBackgroundColor:[NSColor whiteColor].CGColor]; |
| |
| [searchField setWantsLayer:YES]; |
| [searchField setLayer:[CALayer layer]]; |
| [searchField.layer setBackgroundColor:[NSColor colorWithCalibratedRed:0.949 green:0.949 blue:0.949 alpha:0.9].CGColor]; |
| } |
| |
| -(void) selectRow:(id)sender |
| { |
| if ([treeController selectedNodes].count == 0) { |
| RecentModel::instance().selectionModel()->clearCurrentIndex(); |
| return; |
| } |
| auto qIdx = [treeController toQIdx:[treeController selectedNodes][0]]; |
| auto proxyIdx = RecentModel::instance().peopleProxy()->mapToSource(qIdx); |
| RecentModel::instance().selectionModel()->setCurrentIndex(proxyIdx, QItemSelectionModel::ClearAndSelect); |
| } |
| |
| - (void)placeCall:(id)sender |
| { |
| QModelIndex qIdx = [treeController toQIdx:[treeController selectedNodes][0]]; |
| ContactMethod* m = nil; |
| |
| // Double click on an ongoing call |
| if (qIdx.parent().isValid()) { |
| return; |
| } |
| |
| if([[treeController selectedNodes] count] > 0) { |
| QVariant var = qIdx.data((int)Call::Role::ContactMethod); |
| m = qvariant_cast<ContactMethod*>(var); |
| if (!m) { |
| // test if it is a person |
| QVariant var = qIdx.data((int)Person::Role::Object); |
| if (var.isValid()) { |
| Person *c = var.value<Person*>(); |
| if (c->phoneNumbers().size() > 0) { |
| m = c->phoneNumbers().first(); |
| } |
| } |
| } |
| } |
| |
| // Before calling check if we properly extracted a contact method and that |
| // there is NOT already an ongoing call for this index (e.g: no children for this node) |
| if(m && !RecentModel::instance().peopleProxy()->index(0, 0, qIdx).isValid()){ |
| auto c = CallModel::instance().dialingCall(); |
| c->setPeerContactMethod(m); |
| c << Call::Action::ACCEPT; |
| CallModel::instance().selectCall(c); |
| } |
| } |
| |
| - (void)showHistory |
| { |
| [tabbar selectTabViewItemAtIndex:1]; |
| } |
| |
| - (void)showContacts |
| { |
| [tabbar selectTabViewItemAtIndex:2]; |
| } |
| |
| - (void)showSmartlist |
| { |
| [tabbar selectTabViewItemAtIndex:0]; |
| } |
| |
| #pragma mark - NSOutlineViewDelegate methods |
| |
| - (BOOL)outlineView:(NSOutlineView *)outlineView shouldSelectItem:(id)item; |
| { |
| return YES; |
| } |
| |
| - (BOOL)outlineView:(NSOutlineView *)outlineView shouldEditTableColumn:(NSTableColumn *)tableColumn item:(id)item |
| { |
| return NO; |
| } |
| |
| - (void)outlineViewSelectionDidChange:(NSNotification *)notification |
| { |
| if ([treeController selectedNodes].count <= 0) { |
| RecentModel::instance().selectionModel()->clearCurrentIndex(); |
| return; |
| } |
| } |
| |
| - (NSView *)outlineView:(NSOutlineView *)outlineView viewForTableColumn:(NSTableColumn *)tableColumn item:(id)item |
| { |
| QModelIndex proxyIdx = [treeController toQIdx:((NSTreeNode*)item)]; |
| QModelIndex qIdx = RecentModel::instance().peopleProxy()->mapToSource(proxyIdx); |
| |
| NSTableCellView* result; |
| if (!qIdx.parent().isValid()) { |
| result = [outlineView makeViewWithIdentifier:@"MainCell" owner:outlineView]; |
| NSTextField* details = [result viewWithTag:DETAILS_TAG]; |
| |
| NSMutableArray* controls = [NSMutableArray arrayWithObject:[result viewWithTag:CALL_BUTTON_TAG]]; |
| [((ContextualTableCellView*) result) setContextualsControls:controls]; |
| [((ContextualTableCellView*) result) setShouldBlurParentView:YES]; |
| |
| if (auto call = RecentModel::instance().getActiveCall(qIdx)) { |
| [details setStringValue:call->roleData((int)Ring::Role::FormattedState).toString().toNSString()]; |
| [((ContextualTableCellView*) result) setActiveState:YES]; |
| } else { |
| [details setStringValue:qIdx.data((int)Ring::Role::FormattedLastUsed).toString().toNSString()]; |
| [((ContextualTableCellView*) result) setActiveState:NO]; |
| } |
| |
| NSTextField* unreadCount = [result viewWithTag:TXT_BUTTON_TAG]; |
| int unread = qIdx.data((int)Ring::Role::UnreadTextMessageCount).toInt(); |
| [unreadCount setHidden:(unread == 0)]; |
| [unreadCount setStringValue:qIdx.data((int)Ring::Role::UnreadTextMessageCount).toString().toNSString()]; |
| |
| } else { |
| result = [outlineView makeViewWithIdentifier:@"CallCell" owner:outlineView]; |
| NSMutableArray* controls = [NSMutableArray arrayWithObject:[result viewWithTag:CANCEL_BUTTON_TAG]]; |
| [((ContextualTableCellView*) result) setContextualsControls:controls]; |
| [((ContextualTableCellView*) result) setShouldBlurParentView:YES]; |
| [((ContextualTableCellView*) result) setActiveState:NO]; |
| NSTextField* details = [result viewWithTag:DETAILS_TAG]; |
| |
| [details setStringValue:qIdx.data((int)Call::Role::HumanStateName).toString().toNSString()]; |
| } |
| |
| NSTextField* displayName = [result viewWithTag:DISPLAYNAME_TAG]; |
| NSString* displayNameString = qIdx.data((int)Ring::Role::Name).toString().toNSString(); |
| NSString* displayIDString = qIdx.data((int)Person::Role::IdOfLastCMUsed).toString().toNSString(); |
| if(displayNameString.length == 0 || [displayNameString isEqualToString:displayIDString]) { |
| [displayName setStringValue:displayIDString]; |
| } |
| else { |
| NSTextField* displayRingID = [result viewWithTag:RING_ID_LABEL]; |
| [displayName setStringValue:displayNameString]; |
| [displayRingID setStringValue:displayIDString]; |
| } |
| NSImageView* photoView = [result viewWithTag:IMAGE_TAG]; |
| |
| [photoView setImage:QtMac::toNSImage(qvariant_cast<QPixmap>(qIdx.data(Qt::DecorationRole)))]; |
| |
| NSView* presenceView = [result viewWithTag:PRESENCE_TAG]; |
| if (qIdx.data(static_cast<int>(Ring::Role::IsPresent)).value<bool>()) { |
| [presenceView setHidden:NO]; |
| } else { |
| [presenceView setHidden:YES]; |
| } |
| return result; |
| } |
| |
| - (NSTableRowView *)outlineView:(NSOutlineView *)outlineView rowViewForItem:(id)item |
| { |
| return [outlineView makeViewWithIdentifier:@"HoverRowView" owner:nil]; |
| } |
| |
| - (void)startCallForRow:(id)sender { |
| NSInteger row = [smartView rowForView:sender]; |
| [smartView selectRowIndexes:[NSIndexSet indexSetWithIndex:row] byExtendingSelection:NO]; |
| [self placeCall:nil]; |
| } |
| |
| - (IBAction)hangUpClickedAtRow:(id)sender { |
| NSInteger row = [smartView rowForView:sender]; |
| id callNode = [smartView itemAtRow:row]; |
| auto callIdx = [treeController toQIdx:((NSTreeNode*)callNode)]; |
| |
| if (callIdx.isValid()) { |
| auto call = RecentModel::instance().getActiveCall(RecentModel::instance().peopleProxy()->mapToSource(callIdx)); |
| call << Call::Action::REFUSE; |
| } |
| } |
| |
| - (CGFloat)outlineView:(NSOutlineView *)outlineView heightOfRowByItem:(id)item |
| { |
| QModelIndex qIdx = [treeController toQIdx:((NSTreeNode*)item)]; |
| return (((NSTreeNode*)item).indexPath.length == 1) ? 60.0 : 50.0; |
| } |
| |
| - (IBAction)placeCallFromSearchField:(id)sender |
| { |
| if ([searchField stringValue].length == 0) { |
| return; |
| } |
| [self processSearchFieldInputAndStartCall:YES]; |
| } |
| |
| - (void) startCallFromURI:(const URI&) uri |
| { |
| auto cm = PhoneDirectoryModel::instance().getNumber(uri, [self chosenAccount]); |
| if(!cm->account() && [self chosenAccount]) { |
| cm->setAccount([self chosenAccount]); |
| } |
| auto c = CallModel::instance().dialingCall(); |
| c->setPeerContactMethod(cm); |
| c << Call::Action::ACCEPT; |
| CallModel::instance().selectCall(c); |
| } |
| |
| - (void) startConversationFromURI:(const URI&) uri |
| { |
| auto cm = PhoneDirectoryModel::instance().getNumber(uri, [self chosenAccount]); |
| if(!cm->account() && [self chosenAccount]) { |
| cm->setAccount([self chosenAccount]); |
| } |
| time_t currentTime; |
| ::time(¤tTime); |
| cm->setLastUsed(currentTime); |
| auto proxyIdx = RecentModel::instance().peopleProxy()->mapToSource(RecentModel::instance().peopleProxy()->index(0, 0)); |
| RecentModel::instance().selectionModel()->setCurrentIndex(proxyIdx, QItemSelectionModel::ClearAndSelect); |
| } |
| |
| - (void) displayErrorModalWithTitle:(NSString*) title WithMessage:(NSString*) message |
| { |
| NSAlert* alert = [NSAlert alertWithMessageText:title |
| defaultButton:@"Ok" |
| alternateButton:nil |
| otherButton:nil |
| informativeTextWithFormat:message]; |
| |
| [alert beginSheetModalForWindow:self.view.window modalDelegate:nil didEndSelector:NULL contextInfo:NULL]; |
| } |
| |
| - (void) processSearchFieldInputAndStartCall:(BOOL) shouldCall |
| { |
| NSString* noValidAccountTitle = NSLocalizedString(@"No valid account available", |
| @"Alert dialog title"); |
| NSString* noValidAccountMessage = NSLocalizedString(@"Make sure you have at least one valid account", |
| @"Alert dialo message"); |
| |
| const auto* numberEntered = [searchField stringValue]; |
| URI uri = URI(numberEntered.UTF8String); |
| [self clearSearchField]; |
| |
| if ([self chosenAccount] && [self chosenAccount]->protocol() == Account::Protocol::RING) { |
| if (uri.protocolHint() == URI::ProtocolHint::RING) { |
| // If it is a RingID start the conversation or the call |
| if (shouldCall) { |
| [self startCallFromURI:uri]; |
| } else { |
| [self startConversationFromURI:uri]; |
| } |
| } else { |
| // If it's not a ringID and the user choosen account is a Ring account do a search on the blockchain |
| QString usernameToLookup = uri.userinfo(); |
| QObject::disconnect(usernameLookupConnection); |
| usernameLookupConnection = QObject::connect(&NameDirectory::instance(), |
| &NameDirectory::registeredNameFound, |
| [self,usernameToLookup,shouldCall] (const Account* account, NameDirectory::LookupStatus status, const QString& address, const QString& name) { |
| if (usernameToLookup.compare(name) != 0) { |
| //That is not our lookup. |
| return; |
| } |
| switch(status) { |
| case NameDirectory::LookupStatus::SUCCESS: { |
| URI uri = URI("ring:" + address); |
| if (shouldCall) { |
| [self startCallFromURI:uri]; |
| } else { |
| [self startConversationFromURI:uri]; |
| } |
| break; |
| } |
| case NameDirectory::LookupStatus::INVALID_NAME: |
| case NameDirectory::LookupStatus::ERROR: |
| case NameDirectory::LookupStatus::NOT_FOUND: { |
| [self displayErrorModalWithTitle:NSLocalizedString(@"Entered name not found", |
| @"Alert dialog title") |
| WithMessage:NSLocalizedString(@"The username you entered do not match a RingID on the network", |
| @"Alert dialog title")]; |
| } |
| break; |
| } |
| }); |
| |
| NameDirectory::instance().lookupName([self chosenAccount], QString(), usernameToLookup); |
| } |
| } else if ([self chosenAccount] && [self chosenAccount]->protocol() == Account::Protocol::SIP) { |
| if (uri.protocolHint() == URI::ProtocolHint::RING) { |
| // If it is a RingID and no valid account is available, present error |
| [self displayErrorModalWithTitle:noValidAccountTitle |
| WithMessage:noValidAccountMessage]; |
| return; |
| } |
| if (shouldCall) { |
| [self startCallFromURI:uri]; |
| } else { |
| [self startConversationFromURI:uri]; |
| } |
| } else { |
| [self displayErrorModalWithTitle:noValidAccountTitle |
| WithMessage:noValidAccountMessage]; |
| } |
| } |
| |
| -(Account* ) chosenAccount |
| { |
| auto idx = AvailableAccountModel::instance().selectionModel()->currentIndex(); |
| if (idx.isValid()) { |
| return idx.data(static_cast<int>(Ring::Role::Object)).value<Account*>(); |
| } |
| return nullptr; |
| } |
| |
| - (void) clearSearchField |
| { |
| [searchField setStringValue:@""]; |
| RecentModel::instance().peopleProxy()->setFilterWildcard(QString::fromNSString([searchField stringValue])); |
| } |
| |
| - (void) addToContact |
| { |
| if ([treeController selectedNodes].count == 0) |
| return; |
| |
| auto qIdx = [treeController toQIdx:[treeController selectedNodes][0]]; |
| auto originIdx = RecentModel::instance().peopleProxy()->mapToSource(qIdx); |
| auto contactmethod = RecentModel::instance().getContactMethods(originIdx); |
| if (contactmethod.isEmpty()) |
| return; |
| |
| if (addToContactPopover != nullptr) { |
| [addToContactPopover performClose:self]; |
| addToContactPopover = NULL; |
| } else if (contactmethod.first()) { |
| auto* editorVC = [[PersonLinkerVC alloc] initWithNibName:@"PersonLinker" bundle:nil]; |
| [editorVC setMethodToLink:contactmethod.first()]; |
| [editorVC setContactLinkedDelegate:self]; |
| addToContactPopover = [[NSPopover alloc] init]; |
| [addToContactPopover setContentSize:editorVC.view.frame.size]; |
| [addToContactPopover setContentViewController:editorVC]; |
| [addToContactPopover setAnimates:YES]; |
| [addToContactPopover setBehavior:NSPopoverBehaviorTransient]; |
| [addToContactPopover setDelegate:self]; |
| |
| [addToContactPopover showRelativeToRect:[smartView frameOfCellAtColumn:0 row:[smartView selectedRow]] |
| ofView:smartView preferredEdge:NSMaxXEdge]; |
| } |
| } |
| |
| - (void) addContactForRow:(id) sender |
| { |
| NSInteger row = [smartView rowForItem:[sender representedObject]]; |
| if(row < 0) { |
| return; |
| } |
| [smartView selectRowIndexes:[NSIndexSet indexSetWithIndex:row] byExtendingSelection:NO]; |
| [self addToContact]; |
| } |
| |
| /** |
| Copy a NSString in the general Pasteboard |
| |
| @param sender the NSObject containing the represented object to copy |
| */ |
| - (void) copyStringToPasteboard:(id) sender |
| { |
| NSPasteboard *pasteBoard = [NSPasteboard generalPasteboard]; |
| [pasteBoard declareTypes:[NSArray arrayWithObject:NSStringPboardType] owner:nil]; |
| [pasteBoard setString:[sender representedObject] forType:NSStringPboardType]; |
| } |
| |
| - (void) callNumber:(id) sender |
| { |
| Call* c = CallModel::instance().dialingCall(); |
| c->setDialNumber(QString::fromNSString([sender representedObject])); |
| c << Call::Action::ACCEPT; |
| } |
| |
| #pragma NSTextFieldDelegate |
| |
| - (BOOL)control:(NSControl *)control textView:(NSTextView *)fieldEditor doCommandBySelector:(SEL)commandSelector |
| { |
| if (commandSelector == @selector(insertNewline:)) { |
| if([[searchField stringValue] isNotEqualTo:@""]) { |
| [self processSearchFieldInputAndStartCall:NO]; |
| return YES; |
| } |
| } |
| return NO; |
| } |
| |
| - (void)controlTextDidChange:(NSNotification *) notification |
| { |
| RecentModel::instance().peopleProxy()->setFilterWildcard(QString::fromNSString([searchField stringValue])); |
| } |
| |
| #pragma mark - NSPopOverDelegate |
| |
| - (void)popoverDidClose:(NSNotification *)notification |
| { |
| if (addToContactPopover != nullptr) { |
| [addToContactPopover performClose:self]; |
| addToContactPopover = NULL; |
| } |
| } |
| |
| |
| #pragma mark - ContactLinkedDelegate |
| |
| - (void)contactLinked |
| { |
| if (addToContactPopover != nullptr) { |
| [addToContactPopover performClose:self]; |
| addToContactPopover = NULL; |
| } |
| } |
| |
| #pragma mark - KeyboardShortcutDelegate |
| |
| - (void) onAddShortcut |
| { |
| auto qIdx = [treeController toQIdx:[treeController selectedNodes][0]]; |
| auto originIdx = RecentModel::instance().peopleProxy()->mapToSource(qIdx); |
| auto contactmethods = RecentModel::instance().getContactMethods(originIdx); |
| if (contactmethods.isEmpty()) |
| return; |
| |
| auto contactmethod = contactmethods.first(); |
| if (contactmethod && (!contactmethod->contact() || contactmethod->contact()->isPlaceHolder())) { |
| [self addToContact]; |
| } |
| } |
| |
| #pragma mark - ContextMenuDelegate |
| |
| - (NSMenu*) contextualMenuForIndex:(NSTreeNode*) item |
| { |
| auto qIdx = [treeController toQIdx:item]; |
| |
| if (!qIdx.isValid()) { |
| return nil; |
| } |
| |
| auto originIdx = RecentModel::instance().peopleProxy()->mapToSource(qIdx); |
| auto contactmethods = RecentModel::instance().getContactMethods(originIdx); |
| if (contactmethods.isEmpty()) |
| return nil; |
| |
| NSMenu *theMenu = [[NSMenu alloc] initWithTitle:@""]; |
| |
| if (contactmethods.size() == 1 |
| && !contactmethods.first()->contact() |
| || contactmethods.first()->contact()->isPlaceHolder()) { |
| |
| NSMenuItem* addContactItem = [[NSMenuItem alloc] initWithTitle:NSLocalizedString(@"Add to contacts", @"Contextual menu action") |
| action:@selector(addContactForRow:) |
| keyEquivalent:@""]; |
| [addContactItem setRepresentedObject:item]; |
| [theMenu addItem:addContactItem]; |
| } else if (auto person = contactmethods.first()->contact()) { |
| NSMenuItem* copyNameItem = [[NSMenuItem alloc] initWithTitle:NSLocalizedString(@"Copy name", @"Contextual menu action") |
| action:@selector(copyStringToPasteboard:) |
| keyEquivalent:@""]; |
| |
| [copyNameItem setRepresentedObject:person->formattedName().toNSString()]; |
| [theMenu addItem:copyNameItem]; |
| } |
| |
| NSMenu* copySubmenu = [[NSMenu alloc] init]; |
| NSMenu* callSubmenu = [[NSMenu alloc] init]; |
| |
| for (auto cm : contactmethods) { |
| NSMenuItem* tmpCopyItem = [[NSMenuItem alloc] initWithTitle:cm->uri().toNSString() |
| action:@selector(copyStringToPasteboard:) |
| keyEquivalent:@""]; |
| |
| [tmpCopyItem setRepresentedObject:cm->uri().toNSString()]; |
| [copySubmenu addItem:tmpCopyItem]; |
| |
| NSMenuItem* tmpCallItem = [[NSMenuItem alloc] initWithTitle:cm->uri().toNSString() |
| action:@selector(callNumber:) |
| keyEquivalent:@""]; |
| [tmpCallItem setRepresentedObject:cm->uri().toNSString()]; |
| [callSubmenu addItem:tmpCallItem]; |
| } |
| |
| NSMenuItem* copyNumberItem = [[NSMenuItem alloc] init]; |
| [copyNumberItem setTitle:NSLocalizedString(@"Copy number", @"Contextual menu action")]; |
| [copyNumberItem setSubmenu:copySubmenu]; |
| |
| NSMenuItem* callItems = [[NSMenuItem alloc] init]; |
| [callItems setTitle:NSLocalizedString(@"Call number", @"Contextual menu action")]; |
| [callItems setSubmenu:callSubmenu]; |
| |
| [theMenu insertItem:copyNumberItem atIndex:theMenu.itemArray.count]; |
| [theMenu insertItem:[NSMenuItem separatorItem] atIndex:theMenu.itemArray.count]; |
| [theMenu insertItem:callItems atIndex:theMenu.itemArray.count]; |
| |
| return theMenu; |
| } |
| |
| @end |