3 * @copyright Copyright (C) 2010-2021, the Friendica project
5 * @license GNU AGPL version 3 or any later version
7 * This program is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU Affero General Public License as
9 * published by the Free Software Foundation, either version 3 of the
10 * License, or (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU Affero General Public License for more details.
17 * You should have received a copy of the GNU Affero General Public License
18 * along with this program. If not, see <https://www.gnu.org/licenses/>.
22 namespace Friendica\Module;
24 use Friendica\BaseModule;
25 use Friendica\Content\ContactSelector;
26 use Friendica\Content\Nav;
27 use Friendica\Content\Pager;
28 use Friendica\Content\Text\BBCode;
29 use Friendica\Content\Widget;
30 use Friendica\Core\ACL;
31 use Friendica\Core\Hook;
32 use Friendica\Core\Protocol;
33 use Friendica\Core\Renderer;
34 use Friendica\Core\Theme;
35 use Friendica\Core\Worker;
36 use Friendica\Database\DBA;
39 use Friendica\Model\User;
40 use Friendica\Module\Security\Login;
41 use Friendica\Network\HTTPException\BadRequestException;
42 use Friendica\Network\HTTPException\NotFoundException;
43 use Friendica\Util\DateTimeFormat;
44 use Friendica\Util\Strings;
47 * Manages and show Contacts and their content
49 class Contact extends BaseModule
51 const TAB_CONVERSATIONS = 1;
53 const TAB_PROFILE = 3;
54 const TAB_CONTACTS = 4;
55 const TAB_ADVANCED = 5;
57 private static function batchActions()
59 if (empty($_POST['contact_batch']) || !is_array($_POST['contact_batch'])) {
63 $contacts_id = $_POST['contact_batch'];
65 $stmt = DBA::select('contact', ['id', 'archive'], ['id' => $contacts_id, 'uid' => local_user(), 'self' => false, 'deleted' => false]);
66 $orig_records = DBA::toArray($stmt);
69 foreach ($orig_records as $orig_record) {
70 $contact_id = $orig_record['id'];
71 if (!empty($_POST['contacts_batch_update'])) {
72 self::updateContactFromPoll($contact_id);
75 if (!empty($_POST['contacts_batch_block'])) {
76 self::blockContact($contact_id);
79 if (!empty($_POST['contacts_batch_ignore'])) {
80 self::ignoreContact($contact_id);
83 if (!empty($_POST['contacts_batch_archive'])
84 && self::archiveContact($contact_id, $orig_record)
88 if (!empty($_POST['contacts_batch_drop'])) {
89 self::dropContact($orig_record);
93 if ($count_actions > 0) {
94 info(DI::l10n()->tt('%d contact edited.', '%d contacts edited.', $count_actions));
97 DI::baseUrl()->redirect('contact');
100 public static function post(array $parameters = [])
108 // @TODO: Replace with parameter from router
109 if ($a->argv[1] === 'batch') {
110 self::batchActions();
114 // @TODO: Replace with parameter from router
115 $contact_id = intval($a->argv[1]);
120 if (!DBA::exists('contact', ['id' => $contact_id, 'uid' => local_user(), 'deleted' => false])) {
121 notice(DI::l10n()->t('Could not access contact record.'));
122 DI::baseUrl()->redirect('contact');
123 return; // NOTREACHED
126 Hook::callAll('contact_edit_post', $_POST);
128 $hidden = !empty($_POST['hidden']);
130 $notify = !empty($_POST['notify']);
132 $fetch_further_information = intval($_POST['fetch_further_information'] ?? 0);
134 $remote_self = $_POST['remote_self'] ?? false;
136 $ffi_keyword_denylist = Strings::escapeHtml(trim($_POST['ffi_keyword_denylist'] ?? ''));
138 $priority = intval($_POST['poll'] ?? 0);
139 if ($priority > 5 || $priority < 0) {
143 $info = Strings::escapeHtml(trim($_POST['info'] ?? ''));
145 $r = DBA::update('contact', [
146 'priority' => $priority,
149 'notify_new_posts' => $notify,
150 'fetch_further_information' => $fetch_further_information,
151 'remote_self' => $remote_self,
152 'ffi_keyword_denylist' => $ffi_keyword_denylist],
153 ['id' => $contact_id, 'uid' => local_user()]
156 if (!DBA::isResult($r)) {
157 notice(DI::l10n()->t('Failed to update contact record.'));
162 /* contact actions */
164 private static function updateContactFromPoll($contact_id)
166 $contact = DBA::selectFirst('contact', ['uid', 'url', 'network'], ['id' => $contact_id, 'uid' => local_user(), 'deleted' => false]);
167 if (!DBA::isResult($contact)) {
171 if ($contact['network'] == Protocol::OSTATUS) {
172 $user = Model\User::getById($contact['uid']);
173 $result = Model\Contact::createFromProbe($user, $contact['url'], false, $contact['network']);
175 if ($result['success']) {
176 DBA::update('contact', ['subhub' => 1], ['id' => $contact_id]);
179 // pull feed and consume it, which should subscribe to the hub.
180 Worker::add(PRIORITY_HIGH, 'OnePoll', $contact_id, 'force');
182 Worker::add(PRIORITY_HIGH, 'UpdateContact', $contact_id);
186 private static function updateContactFromProbe($contact_id)
188 $contact = DBA::selectFirst('contact', ['url'], ['id' => $contact_id, 'uid' => [0, local_user()], 'deleted' => false]);
189 if (!DBA::isResult($contact)) {
193 // Update the entry in the contact table
194 Model\Contact::updateFromProbe($contact_id);
198 * Toggles the blocked status of a contact identified by id.
203 private static function blockContact($contact_id)
205 $blocked = !Model\Contact\User::isBlocked($contact_id, local_user());
206 Model\Contact\User::setBlocked($contact_id, local_user(), $blocked);
210 * Toggles the ignored status of a contact identified by id.
215 private static function ignoreContact($contact_id)
217 $ignored = !Model\Contact\User::isIgnored($contact_id, local_user());
218 Model\Contact\User::setIgnored($contact_id, local_user(), $ignored);
222 * Toggles the archived status of a contact identified by id.
223 * If the current status isn't provided, this will always archive the contact.
226 * @param $orig_record
230 private static function archiveContact($contact_id, $orig_record)
232 $archived = empty($orig_record['archive']);
233 $r = DBA::update('contact', ['archive' => $archived], ['id' => $contact_id, 'uid' => local_user()]);
235 return DBA::isResult($r);
238 private static function dropContact($orig_record)
240 $owner = Model\User::getOwnerDataById(local_user());
241 if (!DBA::isResult($owner)) {
245 Model\Contact::terminateFriendship($owner, $orig_record, true);
246 Model\Contact::remove($orig_record['id']);
249 public static function content(array $parameters = [], $update = 0)
252 return Login::form($_SERVER['REQUEST_URI']);
257 $search = Strings::escapeTags(trim($_GET['search'] ?? ''));
258 $nets = Strings::escapeTags(trim($_GET['nets'] ?? ''));
259 $rel = Strings::escapeTags(trim($_GET['rel'] ?? ''));
260 $group = Strings::escapeTags(trim($_GET['group'] ?? ''));
262 $accounttype = $_GET['accounttype'] ?? '';
263 $accounttypeid = User::getAccountTypeByString($accounttype);
267 $page->registerFooterScript(Theme::getPathForFile('asset/typeahead.js/dist/typeahead.bundle.js'));
268 $page->registerFooterScript(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.js'));
269 $page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.css'));
270 $page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput-typeahead.css'));
273 // @TODO: Replace with parameter from router
274 if ($a->argc == 2 && intval($a->argv[1])
275 || $a->argc == 3 && intval($a->argv[1]) && in_array($a->argv[2], ['posts', 'conversations'])
277 $contact_id = intval($a->argv[1]);
279 // Ensure to use the user contact when the public contact was provided
280 $data = Model\Contact::getPublicAndUserContactID($contact_id, local_user());
281 if (!empty($data['user']) && ($contact_id == $data['public'])) {
282 $contact_id = $data['user'];
285 $contact = DBA::selectFirst('contact', [], ['id' => $contact_id, 'uid' => [0, local_user()], 'deleted' => false]);
287 // Don't display contacts that are about to be deleted
288 if ($contact['network'] == Protocol::PHANTOM) {
293 if (DBA::isResult($contact)) {
294 if ($contact['self']) {
295 // @TODO: Replace with parameter from router
296 if (($a->argc == 3) && intval($a->argv[1]) && in_array($a->argv[2], ['posts', 'conversations'])) {
297 DI::baseUrl()->redirect('profile/' . $contact['nick']);
299 DI::baseUrl()->redirect('profile/' . $contact['nick'] . '/profile');
303 $vcard_widget = Widget\VCard::getHTML($contact);
305 $findpeople_widget = '';
307 $account_widget = '';
308 $networks_widget = '';
311 if ($contact['uid'] != 0) {
312 $groups_widget = Model\Group::sidebarWidget('contact', 'group', 'full', 'everyone', $contact_id);
318 $findpeople_widget = Widget::findPeople();
319 if (isset($_GET['add'])) {
320 $follow_widget = Widget::follow($_GET['add']);
322 $follow_widget = Widget::follow();
325 $account_widget = Widget::accounttypes($_SERVER['REQUEST_URI'], $accounttype);
326 $networks_widget = Widget::networks($_SERVER['REQUEST_URI'], $nets);
327 $rel_widget = Widget::contactRels($_SERVER['REQUEST_URI'], $rel);
328 $groups_widget = Widget::groups($_SERVER['REQUEST_URI'], $group);
331 DI::page()['aside'] .= $vcard_widget . $findpeople_widget . $follow_widget . $account_widget . $groups_widget . $networks_widget . $rel_widget;
333 $tpl = Renderer::getMarkupTemplate('contacts-head.tpl');
334 DI::page()['htmlhead'] .= Renderer::replaceMacros($tpl, [
335 '$baseurl' => DI::baseUrl()->get(true),
339 Nav::setSelected('contact');
342 notice(DI::l10n()->t('Permission denied.'));
343 return Login::form();
347 $contact_id = intval($a->argv[1]);
349 throw new BadRequestException();
352 // @TODO: Replace with parameter from router
355 $orig_record = DBA::selectFirst('contact', [], ['id' => $contact_id, 'uid' => [0, local_user()], 'self' => false, 'deleted' => false]);
356 if (!DBA::isResult($orig_record)) {
357 throw new NotFoundException(DI::l10n()->t('Contact not found'));
360 if ($cmd === 'update' && ($orig_record['uid'] != 0)) {
361 self::updateContactFromPoll($contact_id);
362 DI::baseUrl()->redirect('contact/' . $contact_id);
366 if ($cmd === 'updateprofile') {
367 self::updateContactFromProbe($contact_id);
368 DI::baseUrl()->redirect('contact/' . $contact_id);
372 if ($cmd === 'block') {
373 if (public_contact() === $contact_id) {
374 throw new BadRequestException(DI::l10n()->t('You can\'t block yourself'));
377 self::blockContact($contact_id);
379 $blocked = Model\Contact\User::isBlocked($contact_id, local_user());
380 info(($blocked ? DI::l10n()->t('Contact has been blocked') : DI::l10n()->t('Contact has been unblocked')));
382 DI::baseUrl()->redirect('contact/' . $contact_id);
386 if ($cmd === 'ignore') {
387 if (public_contact() === $contact_id) {
388 throw new BadRequestException(DI::l10n()->t('You can\'t ignore yourself'));
391 self::ignoreContact($contact_id);
393 $ignored = Model\Contact\User::isIgnored($contact_id, local_user());
394 info(($ignored ? DI::l10n()->t('Contact has been ignored') : DI::l10n()->t('Contact has been unignored')));
396 DI::baseUrl()->redirect('contact/' . $contact_id);
400 if ($cmd === 'archive' && ($orig_record['uid'] != 0)) {
401 $r = self::archiveContact($contact_id, $orig_record);
403 $archived = (($orig_record['archive']) ? 0 : 1);
404 info((($archived) ? DI::l10n()->t('Contact has been archived') : DI::l10n()->t('Contact has been unarchived')));
407 DI::baseUrl()->redirect('contact/' . $contact_id);
411 if ($cmd === 'drop' && ($orig_record['uid'] != 0)) {
412 // Check if we should do HTML-based delete confirmation
413 if (!empty($_REQUEST['confirm'])) {
414 DI::page()['aside'] = '';
416 return Renderer::replaceMacros(Renderer::getMarkupTemplate('contact_drop_confirm.tpl'), [
417 '$header' => DI::l10n()->t('Drop contact'),
418 '$contact' => self::getContactTemplateVars($orig_record),
420 '$message' => DI::l10n()->t('Do you really want to delete this contact?'),
421 '$confirm' => DI::l10n()->t('Yes'),
422 '$confirm_url' => DI::args()->getCommand(),
423 '$confirm_name' => 'confirmed',
424 '$cancel' => DI::l10n()->t('Cancel'),
427 // Now check how the user responded to the confirmation query
428 if (!empty($_REQUEST['canceled'])) {
429 DI::baseUrl()->redirect('contact');
432 self::dropContact($orig_record);
433 info(DI::l10n()->t('Contact has been removed.'));
435 DI::baseUrl()->redirect('contact');
438 if ($cmd === 'posts') {
439 return self::getPostsHTML($a, $contact_id);
441 if ($cmd === 'conversations') {
442 return self::getConversationsHMTL($a, $contact_id, $update);
446 $_SESSION['return_path'] = DI::args()->getQueryString();
448 if (!empty($contact)) {
449 DI::page()['htmlhead'] .= Renderer::replaceMacros(Renderer::getMarkupTemplate('contact_head.tpl'), [
450 '$baseurl' => DI::baseUrl()->get(true),
453 $contact['blocked'] = Model\Contact\User::isBlocked($contact['id'], local_user());
454 $contact['readonly'] = Model\Contact\User::isIgnored($contact['id'], local_user());
457 switch ($contact['rel']) {
458 case Model\Contact::FRIEND:
459 $relation_text = DI::l10n()->t('You are mutual friends with %s');
462 case Model\Contact::FOLLOWER;
463 $relation_text = DI::l10n()->t('You are sharing with %s');
466 case Model\Contact::SHARING;
467 $relation_text = DI::l10n()->t('%s is sharing with you');
474 if ($contact['uid'] == 0) {
478 if (!in_array($contact['network'], array_merge(Protocol::FEDERATED, [Protocol::TWITTER]))) {
482 $relation_text = sprintf($relation_text, $contact['name']);
484 $url = Model\Contact::magicLinkByContact($contact);
485 if (strpos($url, 'redir/') === 0) {
486 $sparkle = ' class="sparkle" ';
491 $insecure = DI::l10n()->t('Private communications are not available for this contact.');
493 $last_update = (($contact['last-update'] <= DBA::NULL_DATETIME) ? DI::l10n()->t('Never') : DateTimeFormat::local($contact['last-update'], 'D, j M Y, g:i A'));
495 if ($contact['last-update'] > DBA::NULL_DATETIME) {
496 $last_update .= ' ' . ($contact['failed'] ? DI::l10n()->t('(Update was not successful)') : DI::l10n()->t('(Update was successful)'));
498 $lblsuggest = (($contact['network'] === Protocol::DFRN) ? DI::l10n()->t('Suggest friends') : '');
500 $poll_enabled = in_array($contact['network'], [Protocol::DFRN, Protocol::OSTATUS, Protocol::FEED, Protocol::MAIL]);
502 $nettype = DI::l10n()->t('Network type: %s', ContactSelector::networkToName($contact['network'], $contact['url'], $contact['protocol'], $contact['gsid']));
505 $tab_str = self::getTabsHTML($contact, self::TAB_PROFILE);
507 $lost_contact = (($contact['archive'] && $contact['term-date'] > DBA::NULL_DATETIME && $contact['term-date'] < DateTimeFormat::utcNow()) ? DI::l10n()->t('Communications lost with this contact!') : '');
509 $fetch_further_information = null;
510 if ($contact['network'] == Protocol::FEED) {
511 $fetch_further_information = [
512 'fetch_further_information',
513 DI::l10n()->t('Fetch further information for feeds'),
514 $contact['fetch_further_information'],
515 DI::l10n()->t('Fetch information like preview pictures, title and teaser from the feed item. You can activate this if the feed doesn\'t contain much text. Keywords are taken from the meta header in the feed item and are posted as hash tags.'),
517 '0' => DI::l10n()->t('Disabled'),
518 '1' => DI::l10n()->t('Fetch information'),
519 '3' => DI::l10n()->t('Fetch keywords'),
520 '2' => DI::l10n()->t('Fetch information and keywords')
525 // Disable remote self for everything except feeds.
526 // There is an issue when you repeat an item from maybe twitter and you got comments from friendica and twitter
527 // Problem is, you couldn't reply to both networks.
528 $allow_remote_self = in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::FEED, Protocol::DFRN, Protocol::DIASPORA, Protocol::TWITTER])
529 && DI::config()->get('system', 'allow_users_remote_self');
531 if ($contact['network'] == Protocol::FEED) {
532 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
533 Model\Contact::MIRROR_FORWARDED => DI::l10n()->t('Mirror as forwarded posting'),
534 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting')];
535 } elseif (in_array($contact['network'], [Protocol::ACTIVITYPUB])) {
536 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
537 Model\Contact::MIRROR_NATIVE_RESHARE => DI::l10n()->t('Native reshare')];
538 } elseif (in_array($contact['network'], [Protocol::DFRN])) {
539 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
540 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting'),
541 Model\Contact::MIRROR_NATIVE_RESHARE => DI::l10n()->t('Native reshare')];
543 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
544 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting')];
547 $poll_interval = null;
548 if ((($contact['network'] == Protocol::FEED) && !DI::config()->get('system', 'adjust_poll_frequency')) || ($contact['network']== Protocol::MAIL)) {
549 $poll_interval = ContactSelector::pollInterval($contact['priority'], !$poll_enabled);
552 // Load contactact related actions like hide, suggest, delete and others
553 $contact_actions = self::getContactActions($contact);
555 if ($contact['uid'] != 0) {
556 $lbl_info1 = DI::l10n()->t('Contact Information / Notes');
557 $contact_settings_label = DI::l10n()->t('Contact Settings');
560 $contact_settings_label = null;
563 $tpl = Renderer::getMarkupTemplate('contact_edit.tpl');
564 $o .= Renderer::replaceMacros($tpl, [
565 '$header' => DI::l10n()->t('Contact'),
566 '$tab_str' => $tab_str,
567 '$submit' => DI::l10n()->t('Submit'),
568 '$lbl_info1' => $lbl_info1,
569 '$lbl_info2' => DI::l10n()->t('Their personal note'),
570 '$reason' => trim(Strings::escapeTags($contact['reason'])),
571 '$infedit' => DI::l10n()->t('Edit contact notes'),
572 '$common_link' => 'contact/' . $contact['id'] . '/contacts/common',
573 '$relation_text' => $relation_text,
574 '$visit' => DI::l10n()->t('Visit %s\'s profile [%s]', $contact['name'], $contact['url']),
575 '$blockunblock' => DI::l10n()->t('Block/Unblock contact'),
576 '$ignorecont' => DI::l10n()->t('Ignore contact'),
577 '$lblrecent' => DI::l10n()->t('View conversations'),
578 '$lblsuggest' => $lblsuggest,
579 '$nettype' => $nettype,
580 '$poll_interval' => $poll_interval,
581 '$poll_enabled' => $poll_enabled,
582 '$lastupdtext' => DI::l10n()->t('Last update:'),
583 '$lost_contact' => $lost_contact,
584 '$updpub' => DI::l10n()->t('Update public posts'),
585 '$last_update' => $last_update,
586 '$udnow' => DI::l10n()->t('Update now'),
587 '$contact_id' => $contact['id'],
588 '$block_text' => ($contact['blocked'] ? DI::l10n()->t('Unblock') : DI::l10n()->t('Block')),
589 '$ignore_text' => ($contact['readonly'] ? DI::l10n()->t('Unignore') : DI::l10n()->t('Ignore')),
590 '$insecure' => (in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::DFRN, Protocol::MAIL, Protocol::DIASPORA]) ? '' : $insecure),
591 '$info' => $contact['info'],
592 '$cinfo' => ['info', '', $contact['info'], ''],
593 '$blocked' => ($contact['blocked'] ? DI::l10n()->t('Currently blocked') : ''),
594 '$ignored' => ($contact['readonly'] ? DI::l10n()->t('Currently ignored') : ''),
595 '$archived' => ($contact['archive'] ? DI::l10n()->t('Currently archived') : ''),
596 '$pending' => ($contact['pending'] ? DI::l10n()->t('Awaiting connection acknowledge') : ''),
597 '$hidden' => ['hidden', DI::l10n()->t('Hide this contact from others'), ($contact['hidden'] == 1), DI::l10n()->t('Replies/likes to your public posts <strong>may</strong> still be visible')],
598 '$notify' => ['notify', DI::l10n()->t('Notification for new posts'), ($contact['notify_new_posts'] == 1), DI::l10n()->t('Send a notification of every new post of this contact')],
599 '$fetch_further_information' => $fetch_further_information,
600 '$ffi_keyword_denylist' => ['ffi_keyword_denylist', DI::l10n()->t('Keyword Deny List'), $contact['ffi_keyword_denylist'], DI::l10n()->t('Comma separated list of keywords that should not be converted to hashtags, when "Fetch information and keywords" is selected')],
601 '$photo' => Model\Contact::getPhoto($contact),
602 '$name' => $contact['name'],
603 '$sparkle' => $sparkle,
605 '$profileurllabel'=> DI::l10n()->t('Profile URL'),
606 '$profileurl' => $contact['url'],
607 '$account_type' => Model\Contact::getAccountType($contact),
608 '$location' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['location']),
609 '$location_label' => DI::l10n()->t('Location:'),
610 '$xmpp' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['xmpp']),
611 '$xmpp_label' => DI::l10n()->t('XMPP:'),
612 '$about' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['about'], BBCode::EXTERNAL),
613 '$about_label' => DI::l10n()->t('About:'),
614 '$keywords' => $contact['keywords'],
615 '$keywords_label' => DI::l10n()->t('Tags:'),
616 '$contact_action_button' => DI::l10n()->t('Actions'),
617 '$contact_actions'=> $contact_actions,
618 '$contact_status' => DI::l10n()->t('Status'),
619 '$contact_settings_label' => $contact_settings_label,
620 '$contact_profile_label' => DI::l10n()->t('Profile'),
621 '$allow_remote_self' => $allow_remote_self,
622 '$remote_self' => ['remote_self',
623 DI::l10n()->t('Mirror postings from this contact'),
624 $contact['remote_self'],
625 DI::l10n()->t('Mark this contact as remote_self, this will cause friendica to repost new entries from this contact.'),
630 $arr = ['contact' => $contact, 'output' => $o];
632 Hook::callAll('contact_edit', $arr);
634 return $arr['output'];
637 $sql_values = [local_user()];
639 // @TODO: Replace with parameter from router
640 $type = $a->argv[1] ?? '';
644 $sql_extra = " AND EXISTS(SELECT `id` from `user-contact` WHERE `contact`.`id` = `user-contact`.`cid` and `user-contact`.`uid` = ? and `user-contact`.`blocked`)";
645 // This makes the query look for contact.uid = 0
646 array_unshift($sql_values, 0);
649 $sql_extra = " AND `hidden` AND NOT `blocked` AND NOT `pending`";
652 $sql_extra = " AND EXISTS(SELECT `id` from `user-contact` WHERE `contact`.`id` = `user-contact`.`cid` and `user-contact`.`uid` = ? and `user-contact`.`ignored`)";
653 // This makes the query look for contact.uid = 0
654 array_unshift($sql_values, 0);
657 $sql_extra = " AND `archive` AND NOT `blocked` AND NOT `pending`";
660 $sql_extra = " AND `pending` AND NOT `archive` AND NOT `failed` AND ((`rel` = ?)
661 OR EXISTS (SELECT `id` FROM `intro` WHERE `contact-id` = `contact`.`id` AND NOT `ignore`))";
662 $sql_values[] = Model\Contact::SHARING;
665 $sql_extra = " AND NOT `archive` AND NOT `blocked` AND NOT `pending`";
669 if (isset($accounttypeid)) {
670 $sql_extra .= " AND `contact-type` = ?";
671 $sql_values[] = $accounttypeid;
678 $search_hdr = $search;
679 $search_txt = preg_quote($search);
680 $sql_extra .= " AND (name REGEXP ? OR url REGEXP ? OR nick REGEXP ?)";
681 $sql_values[] = $search_txt;
682 $sql_values[] = $search_txt;
683 $sql_values[] = $search_txt;
687 $sql_extra .= " AND network = ? ";
688 $sql_values[] = $nets;
693 $sql_extra .= " AND `rel` IN (?, ?)";
694 $sql_values[] = Model\Contact::FOLLOWER;
695 $sql_values[] = Model\Contact::FRIEND;
698 $sql_extra .= " AND `rel` IN (?, ?)";
699 $sql_values[] = Model\Contact::SHARING;
700 $sql_values[] = Model\Contact::FRIEND;
703 $sql_extra .= " AND `rel` = ?";
704 $sql_values[] = Model\Contact::FRIEND;
709 $sql_extra = " AND EXISTS(SELECT `id` FROM `group_member` WHERE `gid` = ? AND `contact`.`id` = `contact-id`)";
710 $sql_values[] = $group;
714 $stmt = DBA::p("SELECT COUNT(*) AS `total`
720 " . Widget::unavailableNetworks(),
723 if (DBA::isResult($stmt)) {
724 $total = DBA::fetch($stmt)['total'];
728 $pager = new Pager(DI::l10n(), DI::args()->getQueryString());
730 $sql_values[] = $pager->getStart();
731 $sql_values[] = $pager->getItemsPerPage();
735 $stmt = DBA::p("SELECT *
745 while ($contact = DBA::fetch($stmt)) {
746 $contact['blocked'] = Model\Contact\User::isBlocked($contact['id'], local_user());
747 $contact['readonly'] = Model\Contact\User::isIgnored($contact['id'], local_user());
748 $contacts[] = self::getContactTemplateVars($contact);
754 'label' => DI::l10n()->t('All Contacts'),
756 'sel' => !$type ? 'active' : '',
757 'title' => DI::l10n()->t('Show all contacts'),
758 'id' => 'showall-tab',
762 'label' => DI::l10n()->t('Pending'),
763 'url' => 'contact/pending',
764 'sel' => $type == 'pending' ? 'active' : '',
765 'title' => DI::l10n()->t('Only show pending contacts'),
766 'id' => 'showpending-tab',
770 'label' => DI::l10n()->t('Blocked'),
771 'url' => 'contact/blocked',
772 'sel' => $type == 'blocked' ? 'active' : '',
773 'title' => DI::l10n()->t('Only show blocked contacts'),
774 'id' => 'showblocked-tab',
778 'label' => DI::l10n()->t('Ignored'),
779 'url' => 'contact/ignored',
780 'sel' => $type == 'ignored' ? 'active' : '',
781 'title' => DI::l10n()->t('Only show ignored contacts'),
782 'id' => 'showignored-tab',
786 'label' => DI::l10n()->t('Archived'),
787 'url' => 'contact/archived',
788 'sel' => $type == 'archived' ? 'active' : '',
789 'title' => DI::l10n()->t('Only show archived contacts'),
790 'id' => 'showarchived-tab',
794 'label' => DI::l10n()->t('Hidden'),
795 'url' => 'contact/hidden',
796 'sel' => $type == 'hidden' ? 'active' : '',
797 'title' => DI::l10n()->t('Only show hidden contacts'),
798 'id' => 'showhidden-tab',
802 'label' => DI::l10n()->t('Groups'),
805 'title' => DI::l10n()->t('Organize your contact groups'),
806 'id' => 'contactgroups-tab',
811 $tabs_tpl = Renderer::getMarkupTemplate('common_tabs.tpl');
812 $tabs_html = Renderer::replaceMacros($tabs_tpl, ['$tabs' => $tabs]);
815 case 'followers': $header = DI::l10n()->t('Followers'); break;
816 case 'following': $header = DI::l10n()->t('Following'); break;
817 case 'mutuals': $header = DI::l10n()->t('Mutual friends'); break;
818 default: $header = DI::l10n()->t('Contacts');
822 case 'pending': $header .= ' - ' . DI::l10n()->t('Pending'); break;
823 case 'blocked': $header .= ' - ' . DI::l10n()->t('Blocked'); break;
824 case 'hidden': $header .= ' - ' . DI::l10n()->t('Hidden'); break;
825 case 'ignored': $header .= ' - ' . DI::l10n()->t('Ignored'); break;
826 case 'archived': $header .= ' - ' . DI::l10n()->t('Archived'); break;
829 $header .= $nets ? ' - ' . ContactSelector::networkToName($nets) : '';
831 $tpl = Renderer::getMarkupTemplate('contacts-template.tpl');
832 $o .= Renderer::replaceMacros($tpl, [
833 '$header' => $header,
834 '$tabs' => $tabs_html,
836 '$search' => $search_hdr,
837 '$desc' => DI::l10n()->t('Search your contacts'),
838 '$finding' => $searching ? DI::l10n()->t('Results for: %s', $search) : '',
839 '$submit' => DI::l10n()->t('Find'),
840 '$cmd' => DI::args()->getCommand(),
841 '$contacts' => $contacts,
842 '$contact_drop_confirm' => DI::l10n()->t('Do you really want to delete this contact?'),
844 '$batch_actions' => [
845 'contacts_batch_update' => DI::l10n()->t('Update'),
846 'contacts_batch_block' => DI::l10n()->t('Block') . '/' . DI::l10n()->t('Unblock'),
847 'contacts_batch_ignore' => DI::l10n()->t('Ignore') . '/' . DI::l10n()->t('Unignore'),
848 'contacts_batch_archive' => DI::l10n()->t('Archive') . '/' . DI::l10n()->t('Unarchive'),
849 'contacts_batch_drop' => DI::l10n()->t('Delete'),
851 '$h_batch_actions' => DI::l10n()->t('Batch Actions'),
852 '$paginate' => $pager->renderFull($total),
859 * List of pages for the Contact TabBar
861 * Available Pages are 'Status', 'Profile', 'Contacts' and 'Common Friends'
863 * @param array $contact The contact array
864 * @param int $active_tab 1 if tab should be marked as active
866 * @return string HTML string of the contact page tabs buttons.
867 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
868 * @throws \ImagickException
870 public static function getTabsHTML(array $contact, int $active_tab)
872 $cid = $pcid = $contact['id'];
873 $data = Model\Contact::getPublicAndUserContactID($contact['id'], local_user());
874 if (!empty($data['user']) && ($contact['id'] == $data['public'])) {
875 $cid = $data['user'];
876 } elseif (!empty($data['public'])) {
877 $pcid = $data['public'];
883 'label' => DI::l10n()->t('Status'),
884 'url' => 'contact/' . $pcid . '/conversations',
885 'sel' => (($active_tab == self::TAB_CONVERSATIONS) ? 'active' : ''),
886 'title' => DI::l10n()->t('Conversations started by this contact'),
887 'id' => 'status-tab',
891 'label' => DI::l10n()->t('Posts and Comments'),
892 'url' => 'contact/' . $pcid . '/posts',
893 'sel' => (($active_tab == self::TAB_POSTS) ? 'active' : ''),
894 'title' => DI::l10n()->t('Status Messages and Posts'),
899 'label' => DI::l10n()->t('Profile'),
900 'url' => 'contact/' . $cid,
901 'sel' => (($active_tab == self::TAB_PROFILE) ? 'active' : ''),
902 'title' => DI::l10n()->t('Profile Details'),
903 'id' => 'profile-tab',
906 ['label' => DI::l10n()->t('Contacts'),
907 'url' => 'contact/' . $pcid . '/contacts',
908 'sel' => (($active_tab == self::TAB_CONTACTS) ? 'active' : ''),
909 'title' => DI::l10n()->t('View all known contacts'),
910 'id' => 'contacts-tab',
915 if (!empty($contact['network']) && in_array($contact['network'], [Protocol::FEED, Protocol::MAIL]) && ($cid != $pcid)) {
916 $tabs[] = ['label' => DI::l10n()->t('Advanced'),
917 'url' => 'contact/' . $cid . '/advanced/',
918 'sel' => (($active_tab == self::TAB_ADVANCED) ? 'active' : ''),
919 'title' => DI::l10n()->t('Advanced Contact Settings'),
920 'id' => 'advanced-tab',
925 $tab_tpl = Renderer::getMarkupTemplate('common_tabs.tpl');
926 $tab_str = Renderer::replaceMacros($tab_tpl, ['$tabs' => $tabs]);
931 public static function getConversationsHMTL($a, $contact_id, $update, $parent = 0)
936 // We need the editor here to be able to reshare an item.
940 'allow_location' => $a->user['allow_location'],
941 'default_location' => $a->user['default-location'],
942 'nickname' => $a->user['nickname'],
943 'lockstate' => (is_array($a->user) && (strlen($a->user['allow_cid']) || strlen($a->user['allow_gid']) || strlen($a->user['deny_cid']) || strlen($a->user['deny_gid'])) ? 'lock' : 'unlock'),
944 'acl' => ACL::getFullSelectorHTML(DI::page(), $a->user, true),
946 'visitor' => 'block',
947 'profile_uid' => local_user(),
949 $o = status_editor($a, $x, 0, true);
953 $contact = DBA::selectFirst('contact', ['uid', 'url', 'id'], ['id' => $contact_id, 'deleted' => false]);
956 $o .= self::getTabsHTML($contact, self::TAB_CONVERSATIONS);
959 if (DBA::isResult($contact)) {
961 $profiledata = Model\Contact::getByURLForUser($contact['url'], local_user());
962 DI::page()['aside'] = Widget\VCard::getHTML($profiledata);
964 DI::page()['aside'] = '';
967 if ($contact['uid'] == 0) {
968 $o .= Model\Contact::getPostsFromId($contact['id'], true, $update, $parent);
970 $o .= Model\Contact::getPostsFromUrl($contact['url'], true, $update, $parent);
977 private static function getPostsHTML($a, $contact_id)
979 $contact = DBA::selectFirst('contact', ['uid', 'url', 'id'], ['id' => $contact_id, 'deleted' => false]);
981 $o = self::getTabsHTML($contact, self::TAB_POSTS);
983 if (DBA::isResult($contact)) {
984 $profiledata = Model\Contact::getByURLForUser($contact['url'], local_user());
986 if (local_user() && in_array($profiledata['network'], Protocol::FEDERATED)) {
987 $profiledata['remoteconnect'] = DI::baseUrl() . '/follow?url=' . urlencode($profiledata['url']);
990 DI::page()['aside'] = Widget\VCard::getHTML($profiledata);
992 if ($contact['uid'] == 0) {
993 $o .= Model\Contact::getPostsFromId($contact['id']);
995 $o .= Model\Contact::getPostsFromUrl($contact['url']);
1003 * Return the fields for the contact template
1005 * @param array $contact Contact array
1006 * @return array Template fields
1008 public static function getContactTemplateVars(array $contact)
1012 if (!empty($contact['url']) && isset($contact['uid']) && ($contact['uid'] == 0) && local_user()) {
1013 $personal = Model\Contact::getByURL($contact['url'], false, ['uid', 'rel', 'self'], local_user());
1014 if (!empty($personal)) {
1015 $contact['uid'] = $personal['uid'];
1016 $contact['rel'] = $personal['rel'];
1017 $contact['self'] = $personal['self'];
1021 if (!empty($contact['uid']) && !empty($contact['rel']) && local_user() == $contact['uid']) {
1022 switch ($contact['rel']) {
1023 case Model\Contact::FRIEND:
1024 $alt_text = DI::l10n()->t('Mutual Friendship');
1027 case Model\Contact::FOLLOWER;
1028 $alt_text = DI::l10n()->t('is a fan of yours');
1031 case Model\Contact::SHARING;
1032 $alt_text = DI::l10n()->t('you are a fan of');
1040 $url = Model\Contact::magicLinkByContact($contact);
1042 if (strpos($url, 'redir/') === 0) {
1043 $sparkle = ' class="sparkle" ';
1048 if ($contact['pending']) {
1049 if (in_array($contact['rel'], [Model\Contact::FRIEND, Model\Contact::SHARING])) {
1050 $alt_text = DI::l10n()->t('Pending outgoing contact request');
1052 $alt_text = DI::l10n()->t('Pending incoming contact request');
1056 if ($contact['self']) {
1057 $alt_text = DI::l10n()->t('This is you');
1058 $url = $contact['url'];
1063 'id' => $contact['id'],
1065 'img_hover' => DI::l10n()->t('Visit %s\'s profile [%s]', $contact['name'], $contact['url']),
1066 'photo_menu' => Model\Contact::photoMenu($contact),
1067 'thumb' => Model\Contact::getThumb($contact, true),
1068 'alt_text' => $alt_text,
1069 'name' => $contact['name'],
1070 'nick' => $contact['nick'],
1071 'details' => $contact['location'],
1072 'tags' => $contact['keywords'],
1073 'about' => $contact['about'],
1074 'account_type' => Model\Contact::getAccountType($contact),
1075 'sparkle' => $sparkle,
1076 'itemurl' => ($contact['addr'] ?? '') ?: $contact['url'],
1077 'network' => ContactSelector::networkToName($contact['network'], $contact['url'], $contact['protocol'], $contact['gsid']),
1082 * Gives a array with actions which can performed to a given contact
1084 * This includes actions like e.g. 'block', 'hide', 'archive', 'delete' and others
1086 * @param array $contact Data about the Contact
1087 * @return array with contact related actions
1089 private static function getContactActions($contact)
1091 $poll_enabled = in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::DFRN, Protocol::OSTATUS, Protocol::FEED, Protocol::MAIL]);
1092 $contact_actions = [];
1094 // Provide friend suggestion only for Friendica contacts
1095 if ($contact['network'] === Protocol::DFRN) {
1096 $contact_actions['suggest'] = [
1097 'label' => DI::l10n()->t('Suggest friends'),
1098 'url' => 'fsuggest/' . $contact['id'],
1105 if ($poll_enabled) {
1106 $contact_actions['update'] = [
1107 'label' => DI::l10n()->t('Update now'),
1108 'url' => 'contact/' . $contact['id'] . '/update',
1115 if (in_array($contact['network'], Protocol::NATIVE_SUPPORT)) {
1116 $contact_actions['updateprofile'] = [
1117 'label' => DI::l10n()->t('Refetch contact data'),
1118 'url' => 'contact/' . $contact['id'] . '/updateprofile',
1121 'id' => 'updateprofile',
1125 $contact_actions['block'] = [
1126 'label' => (intval($contact['blocked']) ? DI::l10n()->t('Unblock') : DI::l10n()->t('Block')),
1127 'url' => 'contact/' . $contact['id'] . '/block',
1128 'title' => DI::l10n()->t('Toggle Blocked status'),
1129 'sel' => (intval($contact['blocked']) ? 'active' : ''),
1130 'id' => 'toggle-block',
1133 $contact_actions['ignore'] = [
1134 'label' => (intval($contact['readonly']) ? DI::l10n()->t('Unignore') : DI::l10n()->t('Ignore')),
1135 'url' => 'contact/' . $contact['id'] . '/ignore',
1136 'title' => DI::l10n()->t('Toggle Ignored status'),
1137 'sel' => (intval($contact['readonly']) ? 'active' : ''),
1138 'id' => 'toggle-ignore',
1141 if ($contact['uid'] != 0) {
1142 $contact_actions['archive'] = [
1143 'label' => (intval($contact['archive']) ? DI::l10n()->t('Unarchive') : DI::l10n()->t('Archive')),
1144 'url' => 'contact/' . $contact['id'] . '/archive',
1145 'title' => DI::l10n()->t('Toggle Archive status'),
1146 'sel' => (intval($contact['archive']) ? 'active' : ''),
1147 'id' => 'toggle-archive',
1150 $contact_actions['delete'] = [
1151 'label' => DI::l10n()->t('Delete'),
1152 'url' => 'contact/' . $contact['id'] . '/drop',
1153 'title' => DI::l10n()->t('Delete contact'),
1159 return $contact_actions;