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.'));
160 $contact = DBA::selectFirst('contact', [], ['id' => $contact_id, 'uid' => local_user(), 'deleted' => false]);
161 if (DBA::isResult($contact)) {
162 $a->data['contact'] = $contact;
168 /* contact actions */
170 private static function updateContactFromPoll($contact_id)
172 $contact = DBA::selectFirst('contact', ['uid', 'url', 'network'], ['id' => $contact_id, 'uid' => local_user(), 'deleted' => false]);
173 if (!DBA::isResult($contact)) {
177 if ($contact['network'] == Protocol::OSTATUS) {
178 $user = Model\User::getById($contact['uid']);
179 $result = Model\Contact::createFromProbe($user, $contact['url'], false, $contact['network']);
181 if ($result['success']) {
182 DBA::update('contact', ['subhub' => 1], ['id' => $contact_id]);
185 // pull feed and consume it, which should subscribe to the hub.
186 Worker::add(PRIORITY_HIGH, 'OnePoll', $contact_id, 'force');
188 Worker::add(PRIORITY_HIGH, 'UpdateContact', $contact_id);
192 private static function updateContactFromProbe($contact_id)
194 $contact = DBA::selectFirst('contact', ['url'], ['id' => $contact_id, 'uid' => [0, local_user()], 'deleted' => false]);
195 if (!DBA::isResult($contact)) {
199 // Update the entry in the contact table
200 Model\Contact::updateFromProbe($contact_id);
204 * Toggles the blocked status of a contact identified by id.
209 private static function blockContact($contact_id)
211 $blocked = !Model\Contact\User::isBlocked($contact_id, local_user());
212 Model\Contact\User::setBlocked($contact_id, local_user(), $blocked);
216 * Toggles the ignored status of a contact identified by id.
221 private static function ignoreContact($contact_id)
223 $ignored = !Model\Contact\User::isIgnored($contact_id, local_user());
224 Model\Contact\User::setIgnored($contact_id, local_user(), $ignored);
228 * Toggles the archived status of a contact identified by id.
229 * If the current status isn't provided, this will always archive the contact.
232 * @param $orig_record
236 private static function archiveContact($contact_id, $orig_record)
238 $archived = empty($orig_record['archive']);
239 $r = DBA::update('contact', ['archive' => $archived], ['id' => $contact_id, 'uid' => local_user()]);
241 return DBA::isResult($r);
244 private static function dropContact($orig_record)
246 $owner = Model\User::getOwnerDataById(local_user());
247 if (!DBA::isResult($owner)) {
251 Model\Contact::terminateFriendship($owner, $orig_record, true);
252 Model\Contact::remove($orig_record['id']);
255 public static function content(array $parameters = [], $update = 0)
258 return Login::form($_SERVER['REQUEST_URI']);
263 $search = Strings::escapeTags(trim($_GET['search'] ?? ''));
264 $nets = Strings::escapeTags(trim($_GET['nets'] ?? ''));
265 $rel = Strings::escapeTags(trim($_GET['rel'] ?? ''));
266 $group = Strings::escapeTags(trim($_GET['group'] ?? ''));
268 $accounttype = $_GET['accounttype'] ?? '';
269 $accounttypeid = User::getAccountTypeByString($accounttype);
273 $page->registerFooterScript(Theme::getPathForFile('asset/typeahead.js/dist/typeahead.bundle.js'));
274 $page->registerFooterScript(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.js'));
275 $page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.css'));
276 $page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput-typeahead.css'));
279 // @TODO: Replace with parameter from router
280 if ($a->argc == 2 && intval($a->argv[1])
281 || $a->argc == 3 && intval($a->argv[1]) && in_array($a->argv[2], ['posts', 'conversations'])
283 $contact_id = intval($a->argv[1]);
285 // Ensure to use the user contact when the public contact was provided
286 $data = Model\Contact::getPublicAndUserContactID($contact_id, local_user());
287 if (!empty($data['user']) && ($contact_id == $data['public'])) {
288 $contact_id = $data['user'];
291 $contact = DBA::selectFirst('contact', [], ['id' => $contact_id, 'uid' => [0, local_user()], 'deleted' => false]);
293 // Don't display contacts that are about to be deleted
294 if ($contact['network'] == Protocol::PHANTOM) {
299 if (DBA::isResult($contact)) {
300 if ($contact['self']) {
301 // @TODO: Replace with parameter from router
302 if (($a->argc == 3) && intval($a->argv[1]) && in_array($a->argv[2], ['posts', 'conversations'])) {
303 DI::baseUrl()->redirect('profile/' . $contact['nick']);
305 DI::baseUrl()->redirect('profile/' . $contact['nick'] . '/profile');
309 $a->data['contact'] = $contact;
311 if (($contact['network'] != '') && ($contact['network'] != Protocol::DFRN)) {
312 $network_link = Strings::formatNetworkName($contact['network'], $contact['url']);
319 if (in_array($contact['network'], Protocol::NATIVE_SUPPORT)) {
320 if ($contact['uid'] && in_array($contact['rel'], [Model\Contact::SHARING, Model\Contact::FRIEND])) {
321 $unfollow_link = 'unfollow?url=' . urlencode($contact['url']) . '&auto=1';
322 } elseif(!$contact['pending']) {
323 $follow_link = 'follow?url=' . urlencode($contact['url']) . '&auto=1';
327 $wallmessage_link = '';
328 if ($contact['uid'] && Model\Contact::canReceivePrivateMessages($contact)) {
329 $wallmessage_link = 'message/new/' . $contact['id'];
332 $vcard_widget = Renderer::replaceMacros(Renderer::getMarkupTemplate('widget/vcard.tpl'), [
333 '$name' => $contact['name'],
334 '$photo' => Model\Contact::getPhoto($contact),
335 '$url' => Model\Contact::magicLinkByContact($contact, $contact['url']),
336 '$addr' => $contact['addr'] ?? '',
337 '$network_link' => $network_link,
338 '$network' => DI::l10n()->t('Network:'),
339 '$account_type' => Model\Contact::getAccountType($contact),
340 '$follow' => DI::l10n()->t('Follow'),
341 '$follow_link' => $follow_link,
342 '$unfollow' => DI::l10n()->t('Unfollow'),
343 '$unfollow_link' => $unfollow_link,
344 '$wallmessage' => DI::l10n()->t('Message'),
345 '$wallmessage_link' => $wallmessage_link,
348 $findpeople_widget = '';
350 $account_widget = '';
351 $networks_widget = '';
354 if ($contact['uid'] != 0) {
355 $groups_widget = Model\Group::sidebarWidget('contact', 'group', 'full', 'everyone', $contact_id);
361 $findpeople_widget = Widget::findPeople();
362 if (isset($_GET['add'])) {
363 $follow_widget = Widget::follow($_GET['add']);
365 $follow_widget = Widget::follow();
368 $account_widget = Widget::accounttypes($_SERVER['REQUEST_URI'], $accounttype);
369 $networks_widget = Widget::networks($_SERVER['REQUEST_URI'], $nets);
370 $rel_widget = Widget::contactRels($_SERVER['REQUEST_URI'], $rel);
371 $groups_widget = Widget::groups($_SERVER['REQUEST_URI'], $group);
374 DI::page()['aside'] .= $vcard_widget . $findpeople_widget . $follow_widget . $account_widget . $groups_widget . $networks_widget . $rel_widget;
376 $tpl = Renderer::getMarkupTemplate('contacts-head.tpl');
377 DI::page()['htmlhead'] .= Renderer::replaceMacros($tpl, [
378 '$baseurl' => DI::baseUrl()->get(true),
382 Nav::setSelected('contact');
385 notice(DI::l10n()->t('Permission denied.'));
386 return Login::form();
390 $contact_id = intval($a->argv[1]);
392 throw new BadRequestException();
395 // @TODO: Replace with parameter from router
398 $orig_record = DBA::selectFirst('contact', [], ['id' => $contact_id, 'uid' => [0, local_user()], 'self' => false, 'deleted' => false]);
399 if (!DBA::isResult($orig_record)) {
400 throw new NotFoundException(DI::l10n()->t('Contact not found'));
403 if ($cmd === 'update' && ($orig_record['uid'] != 0)) {
404 self::updateContactFromPoll($contact_id);
405 DI::baseUrl()->redirect('contact/' . $contact_id);
409 if ($cmd === 'updateprofile') {
410 self::updateContactFromProbe($contact_id);
411 DI::baseUrl()->redirect('contact/' . $contact_id);
415 if ($cmd === 'block') {
416 if (public_contact() === $contact_id) {
417 throw new BadRequestException(DI::l10n()->t('You can\'t block yourself'));
420 self::blockContact($contact_id);
422 $blocked = Model\Contact\User::isBlocked($contact_id, local_user());
423 info(($blocked ? DI::l10n()->t('Contact has been blocked') : DI::l10n()->t('Contact has been unblocked')));
425 DI::baseUrl()->redirect('contact/' . $contact_id);
429 if ($cmd === 'ignore') {
430 if (public_contact() === $contact_id) {
431 throw new BadRequestException(DI::l10n()->t('You can\'t ignore yourself'));
434 self::ignoreContact($contact_id);
436 $ignored = Model\Contact\User::isIgnored($contact_id, local_user());
437 info(($ignored ? DI::l10n()->t('Contact has been ignored') : DI::l10n()->t('Contact has been unignored')));
439 DI::baseUrl()->redirect('contact/' . $contact_id);
443 if ($cmd === 'archive' && ($orig_record['uid'] != 0)) {
444 $r = self::archiveContact($contact_id, $orig_record);
446 $archived = (($orig_record['archive']) ? 0 : 1);
447 info((($archived) ? DI::l10n()->t('Contact has been archived') : DI::l10n()->t('Contact has been unarchived')));
450 DI::baseUrl()->redirect('contact/' . $contact_id);
454 if ($cmd === 'drop' && ($orig_record['uid'] != 0)) {
455 // Check if we should do HTML-based delete confirmation
456 if (!empty($_REQUEST['confirm'])) {
457 DI::page()['aside'] = '';
459 return Renderer::replaceMacros(Renderer::getMarkupTemplate('contact_drop_confirm.tpl'), [
460 '$header' => DI::l10n()->t('Drop contact'),
461 '$contact' => self::getContactTemplateVars($orig_record),
463 '$message' => DI::l10n()->t('Do you really want to delete this contact?'),
464 '$confirm' => DI::l10n()->t('Yes'),
465 '$confirm_url' => DI::args()->getCommand(),
466 '$confirm_name' => 'confirmed',
467 '$cancel' => DI::l10n()->t('Cancel'),
470 // Now check how the user responded to the confirmation query
471 if (!empty($_REQUEST['canceled'])) {
472 DI::baseUrl()->redirect('contact');
475 self::dropContact($orig_record);
476 info(DI::l10n()->t('Contact has been removed.'));
478 DI::baseUrl()->redirect('contact');
481 if ($cmd === 'posts') {
482 return self::getPostsHTML($a, $contact_id);
484 if ($cmd === 'conversations') {
485 return self::getConversationsHMTL($a, $contact_id, $update);
489 $_SESSION['return_path'] = DI::args()->getQueryString();
491 if (!empty($a->data['contact']) && is_array($a->data['contact'])) {
492 $contact = $a->data['contact'];
494 DI::page()['htmlhead'] .= Renderer::replaceMacros(Renderer::getMarkupTemplate('contact_head.tpl'), [
495 '$baseurl' => DI::baseUrl()->get(true),
498 $contact['blocked'] = Model\Contact\User::isBlocked($contact['id'], local_user());
499 $contact['readonly'] = Model\Contact\User::isIgnored($contact['id'], local_user());
502 switch ($contact['rel']) {
503 case Model\Contact::FRIEND:
504 $relation_text = DI::l10n()->t('You are mutual friends with %s');
507 case Model\Contact::FOLLOWER;
508 $relation_text = DI::l10n()->t('You are sharing with %s');
511 case Model\Contact::SHARING;
512 $relation_text = DI::l10n()->t('%s is sharing with you');
519 if ($contact['uid'] == 0) {
523 if (!in_array($contact['network'], array_merge(Protocol::FEDERATED, [Protocol::TWITTER]))) {
527 $relation_text = sprintf($relation_text, $contact['name']);
529 $url = Model\Contact::magicLinkByContact($contact);
530 if (strpos($url, 'redir/') === 0) {
531 $sparkle = ' class="sparkle" ';
536 $insecure = DI::l10n()->t('Private communications are not available for this contact.');
538 $last_update = (($contact['last-update'] <= DBA::NULL_DATETIME) ? DI::l10n()->t('Never') : DateTimeFormat::local($contact['last-update'], 'D, j M Y, g:i A'));
540 if ($contact['last-update'] > DBA::NULL_DATETIME) {
541 $last_update .= ' ' . ($contact['failed'] ? DI::l10n()->t('(Update was not successful)') : DI::l10n()->t('(Update was successful)'));
543 $lblsuggest = (($contact['network'] === Protocol::DFRN) ? DI::l10n()->t('Suggest friends') : '');
545 $poll_enabled = in_array($contact['network'], [Protocol::DFRN, Protocol::OSTATUS, Protocol::FEED, Protocol::MAIL]);
547 $nettype = DI::l10n()->t('Network type: %s', ContactSelector::networkToName($contact['network'], $contact['url'], $contact['protocol'], $contact['gsid']));
550 $tab_str = self::getTabsHTML($contact, self::TAB_PROFILE);
552 $lost_contact = (($contact['archive'] && $contact['term-date'] > DBA::NULL_DATETIME && $contact['term-date'] < DateTimeFormat::utcNow()) ? DI::l10n()->t('Communications lost with this contact!') : '');
554 $fetch_further_information = null;
555 if ($contact['network'] == Protocol::FEED) {
556 $fetch_further_information = [
557 'fetch_further_information',
558 DI::l10n()->t('Fetch further information for feeds'),
559 $contact['fetch_further_information'],
560 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.'),
562 '0' => DI::l10n()->t('Disabled'),
563 '1' => DI::l10n()->t('Fetch information'),
564 '3' => DI::l10n()->t('Fetch keywords'),
565 '2' => DI::l10n()->t('Fetch information and keywords')
570 // Disable remote self for everything except feeds.
571 // There is an issue when you repeat an item from maybe twitter and you got comments from friendica and twitter
572 // Problem is, you couldn't reply to both networks.
573 $allow_remote_self = in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::FEED, Protocol::DFRN, Protocol::DIASPORA, Protocol::TWITTER])
574 && DI::config()->get('system', 'allow_users_remote_self');
576 if ($contact['network'] == Protocol::FEED) {
577 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
578 Model\Contact::MIRROR_FORWARDED => DI::l10n()->t('Mirror as forwarded posting'),
579 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting')];
580 } elseif (in_array($contact['network'], [Protocol::ACTIVITYPUB])) {
581 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
582 Model\Contact::MIRROR_NATIVE_RESHARE => DI::l10n()->t('Native reshare')];
583 } elseif (in_array($contact['network'], [Protocol::DFRN])) {
584 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
585 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting'),
586 Model\Contact::MIRROR_NATIVE_RESHARE => DI::l10n()->t('Native reshare')];
588 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
589 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting')];
592 $poll_interval = null;
593 if ((($contact['network'] == Protocol::FEED) && !DI::config()->get('system', 'adjust_poll_frequency')) || ($contact['network']== Protocol::MAIL)) {
594 $poll_interval = ContactSelector::pollInterval($contact['priority'], !$poll_enabled);
597 // Load contactact related actions like hide, suggest, delete and others
598 $contact_actions = self::getContactActions($contact);
600 if ($contact['uid'] != 0) {
601 $lbl_info1 = DI::l10n()->t('Contact Information / Notes');
602 $contact_settings_label = DI::l10n()->t('Contact Settings');
605 $contact_settings_label = null;
608 $tpl = Renderer::getMarkupTemplate('contact_edit.tpl');
609 $o .= Renderer::replaceMacros($tpl, [
610 '$header' => DI::l10n()->t('Contact'),
611 '$tab_str' => $tab_str,
612 '$submit' => DI::l10n()->t('Submit'),
613 '$lbl_info1' => $lbl_info1,
614 '$lbl_info2' => DI::l10n()->t('Their personal note'),
615 '$reason' => trim(Strings::escapeTags($contact['reason'])),
616 '$infedit' => DI::l10n()->t('Edit contact notes'),
617 '$common_link' => 'contact/' . $contact['id'] . '/contacts/common',
618 '$relation_text' => $relation_text,
619 '$visit' => DI::l10n()->t('Visit %s\'s profile [%s]', $contact['name'], $contact['url']),
620 '$blockunblock' => DI::l10n()->t('Block/Unblock contact'),
621 '$ignorecont' => DI::l10n()->t('Ignore contact'),
622 '$lblrecent' => DI::l10n()->t('View conversations'),
623 '$lblsuggest' => $lblsuggest,
624 '$nettype' => $nettype,
625 '$poll_interval' => $poll_interval,
626 '$poll_enabled' => $poll_enabled,
627 '$lastupdtext' => DI::l10n()->t('Last update:'),
628 '$lost_contact' => $lost_contact,
629 '$updpub' => DI::l10n()->t('Update public posts'),
630 '$last_update' => $last_update,
631 '$udnow' => DI::l10n()->t('Update now'),
632 '$contact_id' => $contact['id'],
633 '$block_text' => ($contact['blocked'] ? DI::l10n()->t('Unblock') : DI::l10n()->t('Block')),
634 '$ignore_text' => ($contact['readonly'] ? DI::l10n()->t('Unignore') : DI::l10n()->t('Ignore')),
635 '$insecure' => (in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::DFRN, Protocol::MAIL, Protocol::DIASPORA]) ? '' : $insecure),
636 '$info' => $contact['info'],
637 '$cinfo' => ['info', '', $contact['info'], ''],
638 '$blocked' => ($contact['blocked'] ? DI::l10n()->t('Currently blocked') : ''),
639 '$ignored' => ($contact['readonly'] ? DI::l10n()->t('Currently ignored') : ''),
640 '$archived' => ($contact['archive'] ? DI::l10n()->t('Currently archived') : ''),
641 '$pending' => ($contact['pending'] ? DI::l10n()->t('Awaiting connection acknowledge') : ''),
642 '$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')],
643 '$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')],
644 '$fetch_further_information' => $fetch_further_information,
645 '$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')],
646 '$photo' => Model\Contact::getPhoto($contact),
647 '$name' => $contact['name'],
648 '$sparkle' => $sparkle,
650 '$profileurllabel'=> DI::l10n()->t('Profile URL'),
651 '$profileurl' => $contact['url'],
652 '$account_type' => Model\Contact::getAccountType($contact),
653 '$location' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['location']),
654 '$location_label' => DI::l10n()->t('Location:'),
655 '$xmpp' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['xmpp']),
656 '$xmpp_label' => DI::l10n()->t('XMPP:'),
657 '$about' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['about'], BBCode::EXTERNAL),
658 '$about_label' => DI::l10n()->t('About:'),
659 '$keywords' => $contact['keywords'],
660 '$keywords_label' => DI::l10n()->t('Tags:'),
661 '$contact_action_button' => DI::l10n()->t('Actions'),
662 '$contact_actions'=> $contact_actions,
663 '$contact_status' => DI::l10n()->t('Status'),
664 '$contact_settings_label' => $contact_settings_label,
665 '$contact_profile_label' => DI::l10n()->t('Profile'),
666 '$allow_remote_self' => $allow_remote_self,
667 '$remote_self' => ['remote_self',
668 DI::l10n()->t('Mirror postings from this contact'),
669 $contact['remote_self'],
670 DI::l10n()->t('Mark this contact as remote_self, this will cause friendica to repost new entries from this contact.'),
675 $arr = ['contact' => $contact, 'output' => $o];
677 Hook::callAll('contact_edit', $arr);
679 return $arr['output'];
682 $sql_values = [local_user()];
684 // @TODO: Replace with parameter from router
685 $type = $a->argv[1] ?? '';
689 $sql_extra = " AND EXISTS(SELECT `id` from `user-contact` WHERE `contact`.`id` = `user-contact`.`cid` and `user-contact`.`uid` = ? and `user-contact`.`blocked`)";
690 // This makes the query look for contact.uid = 0
691 array_unshift($sql_values, 0);
694 $sql_extra = " AND `hidden` AND NOT `blocked` AND NOT `pending`";
697 $sql_extra = " AND EXISTS(SELECT `id` from `user-contact` WHERE `contact`.`id` = `user-contact`.`cid` and `user-contact`.`uid` = ? and `user-contact`.`ignored`)";
698 // This makes the query look for contact.uid = 0
699 array_unshift($sql_values, 0);
702 $sql_extra = " AND `archive` AND NOT `blocked` AND NOT `pending`";
705 $sql_extra = " AND `pending` AND NOT `archive` AND NOT `failed` AND ((`rel` = ?)
706 OR EXISTS (SELECT `id` FROM `intro` WHERE `contact-id` = `contact`.`id` AND NOT `ignore`))";
707 $sql_values[] = Model\Contact::SHARING;
710 $sql_extra = " AND NOT `archive` AND NOT `blocked` AND NOT `pending`";
714 if (isset($accounttypeid)) {
715 $sql_extra .= " AND `contact-type` = ?";
716 $sql_values[] = $accounttypeid;
723 $search_hdr = $search;
724 $search_txt = preg_quote($search);
725 $sql_extra .= " AND (name REGEXP ? OR url REGEXP ? OR nick REGEXP ?)";
726 $sql_values[] = $search_txt;
727 $sql_values[] = $search_txt;
728 $sql_values[] = $search_txt;
732 $sql_extra .= " AND network = ? ";
733 $sql_values[] = $nets;
738 $sql_extra .= " AND `rel` IN (?, ?)";
739 $sql_values[] = Model\Contact::FOLLOWER;
740 $sql_values[] = Model\Contact::FRIEND;
743 $sql_extra .= " AND `rel` IN (?, ?)";
744 $sql_values[] = Model\Contact::SHARING;
745 $sql_values[] = Model\Contact::FRIEND;
748 $sql_extra .= " AND `rel` = ?";
749 $sql_values[] = Model\Contact::FRIEND;
754 $sql_extra = " AND EXISTS(SELECT `id` FROM `group_member` WHERE `gid` = ? AND `contact`.`id` = `contact-id`)";
755 $sql_values[] = $group;
759 $stmt = DBA::p("SELECT COUNT(*) AS `total`
765 " . Widget::unavailableNetworks(),
768 if (DBA::isResult($stmt)) {
769 $total = DBA::fetch($stmt)['total'];
773 $pager = new Pager(DI::l10n(), DI::args()->getQueryString());
775 $sql_values[] = $pager->getStart();
776 $sql_values[] = $pager->getItemsPerPage();
780 $stmt = DBA::p("SELECT *
790 while ($contact = DBA::fetch($stmt)) {
791 $contact['blocked'] = Model\Contact\User::isBlocked($contact['id'], local_user());
792 $contact['readonly'] = Model\Contact\User::isIgnored($contact['id'], local_user());
793 $contacts[] = self::getContactTemplateVars($contact);
799 'label' => DI::l10n()->t('All Contacts'),
801 'sel' => !$type ? 'active' : '',
802 'title' => DI::l10n()->t('Show all contacts'),
803 'id' => 'showall-tab',
807 'label' => DI::l10n()->t('Pending'),
808 'url' => 'contact/pending',
809 'sel' => $type == 'pending' ? 'active' : '',
810 'title' => DI::l10n()->t('Only show pending contacts'),
811 'id' => 'showpending-tab',
815 'label' => DI::l10n()->t('Blocked'),
816 'url' => 'contact/blocked',
817 'sel' => $type == 'blocked' ? 'active' : '',
818 'title' => DI::l10n()->t('Only show blocked contacts'),
819 'id' => 'showblocked-tab',
823 'label' => DI::l10n()->t('Ignored'),
824 'url' => 'contact/ignored',
825 'sel' => $type == 'ignored' ? 'active' : '',
826 'title' => DI::l10n()->t('Only show ignored contacts'),
827 'id' => 'showignored-tab',
831 'label' => DI::l10n()->t('Archived'),
832 'url' => 'contact/archived',
833 'sel' => $type == 'archived' ? 'active' : '',
834 'title' => DI::l10n()->t('Only show archived contacts'),
835 'id' => 'showarchived-tab',
839 'label' => DI::l10n()->t('Hidden'),
840 'url' => 'contact/hidden',
841 'sel' => $type == 'hidden' ? 'active' : '',
842 'title' => DI::l10n()->t('Only show hidden contacts'),
843 'id' => 'showhidden-tab',
847 'label' => DI::l10n()->t('Groups'),
850 'title' => DI::l10n()->t('Organize your contact groups'),
851 'id' => 'contactgroups-tab',
856 $tabs_tpl = Renderer::getMarkupTemplate('common_tabs.tpl');
857 $tabs_html = Renderer::replaceMacros($tabs_tpl, ['$tabs' => $tabs]);
860 case 'followers': $header = DI::l10n()->t('Followers'); break;
861 case 'following': $header = DI::l10n()->t('Following'); break;
862 case 'mutuals': $header = DI::l10n()->t('Mutual friends'); break;
863 default: $header = DI::l10n()->t('Contacts');
867 case 'pending': $header .= ' - ' . DI::l10n()->t('Pending'); break;
868 case 'blocked': $header .= ' - ' . DI::l10n()->t('Blocked'); break;
869 case 'hidden': $header .= ' - ' . DI::l10n()->t('Hidden'); break;
870 case 'ignored': $header .= ' - ' . DI::l10n()->t('Ignored'); break;
871 case 'archived': $header .= ' - ' . DI::l10n()->t('Archived'); break;
874 $header .= $nets ? ' - ' . ContactSelector::networkToName($nets) : '';
876 $tpl = Renderer::getMarkupTemplate('contacts-template.tpl');
877 $o .= Renderer::replaceMacros($tpl, [
878 '$header' => $header,
879 '$tabs' => $tabs_html,
881 '$search' => $search_hdr,
882 '$desc' => DI::l10n()->t('Search your contacts'),
883 '$finding' => $searching ? DI::l10n()->t('Results for: %s', $search) : '',
884 '$submit' => DI::l10n()->t('Find'),
885 '$cmd' => DI::args()->getCommand(),
886 '$contacts' => $contacts,
887 '$contact_drop_confirm' => DI::l10n()->t('Do you really want to delete this contact?'),
889 '$batch_actions' => [
890 'contacts_batch_update' => DI::l10n()->t('Update'),
891 'contacts_batch_block' => DI::l10n()->t('Block') . '/' . DI::l10n()->t('Unblock'),
892 'contacts_batch_ignore' => DI::l10n()->t('Ignore') . '/' . DI::l10n()->t('Unignore'),
893 'contacts_batch_archive' => DI::l10n()->t('Archive') . '/' . DI::l10n()->t('Unarchive'),
894 'contacts_batch_drop' => DI::l10n()->t('Delete'),
896 '$h_batch_actions' => DI::l10n()->t('Batch Actions'),
897 '$paginate' => $pager->renderFull($total),
904 * List of pages for the Contact TabBar
906 * Available Pages are 'Status', 'Profile', 'Contacts' and 'Common Friends'
908 * @param array $contact The contact array
909 * @param int $active_tab 1 if tab should be marked as active
911 * @return string HTML string of the contact page tabs buttons.
912 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
913 * @throws \ImagickException
915 public static function getTabsHTML(array $contact, int $active_tab)
917 $cid = $pcid = $contact['id'];
918 $data = Model\Contact::getPublicAndUserContactID($contact['id'], local_user());
919 if (!empty($data['user']) && ($contact['id'] == $data['public'])) {
920 $cid = $data['user'];
921 } elseif (!empty($data['public'])) {
922 $pcid = $data['public'];
928 'label' => DI::l10n()->t('Status'),
929 'url' => 'contact/' . $pcid . '/conversations',
930 'sel' => (($active_tab == self::TAB_CONVERSATIONS) ? 'active' : ''),
931 'title' => DI::l10n()->t('Conversations started by this contact'),
932 'id' => 'status-tab',
936 'label' => DI::l10n()->t('Posts and Comments'),
937 'url' => 'contact/' . $pcid . '/posts',
938 'sel' => (($active_tab == self::TAB_POSTS) ? 'active' : ''),
939 'title' => DI::l10n()->t('Status Messages and Posts'),
944 'label' => DI::l10n()->t('Profile'),
945 'url' => 'contact/' . $cid,
946 'sel' => (($active_tab == self::TAB_PROFILE) ? 'active' : ''),
947 'title' => DI::l10n()->t('Profile Details'),
948 'id' => 'profile-tab',
951 ['label' => DI::l10n()->t('Contacts'),
952 'url' => 'contact/' . $pcid . '/contacts',
953 'sel' => (($active_tab == self::TAB_CONTACTS) ? 'active' : ''),
954 'title' => DI::l10n()->t('View all known contacts'),
955 'id' => 'contacts-tab',
960 if (!empty($contact['network']) && in_array($contact['network'], [Protocol::FEED, Protocol::MAIL]) && ($cid != $pcid)) {
961 $tabs[] = ['label' => DI::l10n()->t('Advanced'),
962 'url' => 'contact/' . $cid . '/advanced/',
963 'sel' => (($active_tab == self::TAB_ADVANCED) ? 'active' : ''),
964 'title' => DI::l10n()->t('Advanced Contact Settings'),
965 'id' => 'advanced-tab',
970 $tab_tpl = Renderer::getMarkupTemplate('common_tabs.tpl');
971 $tab_str = Renderer::replaceMacros($tab_tpl, ['$tabs' => $tabs]);
976 public static function getConversationsHMTL($a, $contact_id, $update, $parent = 0)
981 // We need the editor here to be able to reshare an item.
985 'allow_location' => $a->user['allow_location'],
986 'default_location' => $a->user['default-location'],
987 'nickname' => $a->user['nickname'],
988 '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'),
989 'acl' => ACL::getFullSelectorHTML(DI::page(), $a->user, true),
991 'visitor' => 'block',
992 'profile_uid' => local_user(),
994 $o = status_editor($a, $x, 0, true);
998 $contact = DBA::selectFirst('contact', ['uid', 'url', 'id'], ['id' => $contact_id, 'deleted' => false]);
1001 $o .= self::getTabsHTML($contact, self::TAB_CONVERSATIONS);
1004 if (DBA::isResult($contact)) {
1005 DI::page()['aside'] = '';
1008 $profiledata = Model\Contact::getByURLForUser($contact['url'], local_user());
1009 Model\Profile::load($a, '', $profiledata, true);
1012 if ($contact['uid'] == 0) {
1013 $o .= Model\Contact::getPostsFromId($contact['id'], true, $update, $parent);
1015 $o .= Model\Contact::getPostsFromUrl($contact['url'], true, $update, $parent);
1022 private static function getPostsHTML($a, $contact_id)
1024 $contact = DBA::selectFirst('contact', ['uid', 'url', 'id'], ['id' => $contact_id, 'deleted' => false]);
1026 $o = self::getTabsHTML($contact, self::TAB_POSTS);
1028 if (DBA::isResult($contact)) {
1029 DI::page()['aside'] = '';
1031 $profiledata = Model\Contact::getByURLForUser($contact['url'], local_user());
1033 if (local_user() && in_array($profiledata['network'], Protocol::FEDERATED)) {
1034 $profiledata['remoteconnect'] = DI::baseUrl() . '/follow?url=' . urlencode($profiledata['url']);
1037 Model\Profile::load($a, '', $profiledata, true);
1039 if ($contact['uid'] == 0) {
1040 $o .= Model\Contact::getPostsFromId($contact['id']);
1042 $o .= Model\Contact::getPostsFromUrl($contact['url']);
1050 * Return the fields for the contact template
1052 * @param array $contact Contact array
1053 * @return array Template fields
1055 public static function getContactTemplateVars(array $contact)
1059 if (!empty($contact['url']) && isset($contact['uid']) && ($contact['uid'] == 0) && local_user()) {
1060 $personal = Model\Contact::getByURL($contact['url'], false, ['uid', 'rel', 'self'], local_user());
1061 if (!empty($personal)) {
1062 $contact['uid'] = $personal['uid'];
1063 $contact['rel'] = $personal['rel'];
1064 $contact['self'] = $personal['self'];
1068 if (!empty($contact['uid']) && !empty($contact['rel']) && local_user() == $contact['uid']) {
1069 switch ($contact['rel']) {
1070 case Model\Contact::FRIEND:
1071 $alt_text = DI::l10n()->t('Mutual Friendship');
1074 case Model\Contact::FOLLOWER;
1075 $alt_text = DI::l10n()->t('is a fan of yours');
1078 case Model\Contact::SHARING;
1079 $alt_text = DI::l10n()->t('you are a fan of');
1087 $url = Model\Contact::magicLinkByContact($contact);
1089 if (strpos($url, 'redir/') === 0) {
1090 $sparkle = ' class="sparkle" ';
1095 if ($contact['pending']) {
1096 if (in_array($contact['rel'], [Model\Contact::FRIEND, Model\Contact::SHARING])) {
1097 $alt_text = DI::l10n()->t('Pending outgoing contact request');
1099 $alt_text = DI::l10n()->t('Pending incoming contact request');
1103 if ($contact['self']) {
1104 $alt_text = DI::l10n()->t('This is you');
1105 $url = $contact['url'];
1110 'id' => $contact['id'],
1112 'img_hover' => DI::l10n()->t('Visit %s\'s profile [%s]', $contact['name'], $contact['url']),
1113 'photo_menu' => Model\Contact::photoMenu($contact),
1114 'thumb' => Model\Contact::getThumb($contact, true),
1115 'alt_text' => $alt_text,
1116 'name' => $contact['name'],
1117 'nick' => $contact['nick'],
1118 'details' => $contact['location'],
1119 'tags' => $contact['keywords'],
1120 'about' => $contact['about'],
1121 'account_type' => Model\Contact::getAccountType($contact),
1122 'sparkle' => $sparkle,
1123 'itemurl' => ($contact['addr'] ?? '') ?: $contact['url'],
1124 'network' => ContactSelector::networkToName($contact['network'], $contact['url'], $contact['protocol'], $contact['gsid']),
1129 * Gives a array with actions which can performed to a given contact
1131 * This includes actions like e.g. 'block', 'hide', 'archive', 'delete' and others
1133 * @param array $contact Data about the Contact
1134 * @return array with contact related actions
1136 private static function getContactActions($contact)
1138 $poll_enabled = in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::DFRN, Protocol::OSTATUS, Protocol::FEED, Protocol::MAIL]);
1139 $contact_actions = [];
1141 // Provide friend suggestion only for Friendica contacts
1142 if ($contact['network'] === Protocol::DFRN) {
1143 $contact_actions['suggest'] = [
1144 'label' => DI::l10n()->t('Suggest friends'),
1145 'url' => 'fsuggest/' . $contact['id'],
1152 if ($poll_enabled) {
1153 $contact_actions['update'] = [
1154 'label' => DI::l10n()->t('Update now'),
1155 'url' => 'contact/' . $contact['id'] . '/update',
1162 if (in_array($contact['network'], Protocol::NATIVE_SUPPORT)) {
1163 $contact_actions['updateprofile'] = [
1164 'label' => DI::l10n()->t('Refetch contact data'),
1165 'url' => 'contact/' . $contact['id'] . '/updateprofile',
1168 'id' => 'updateprofile',
1172 $contact_actions['block'] = [
1173 'label' => (intval($contact['blocked']) ? DI::l10n()->t('Unblock') : DI::l10n()->t('Block')),
1174 'url' => 'contact/' . $contact['id'] . '/block',
1175 'title' => DI::l10n()->t('Toggle Blocked status'),
1176 'sel' => (intval($contact['blocked']) ? 'active' : ''),
1177 'id' => 'toggle-block',
1180 $contact_actions['ignore'] = [
1181 'label' => (intval($contact['readonly']) ? DI::l10n()->t('Unignore') : DI::l10n()->t('Ignore')),
1182 'url' => 'contact/' . $contact['id'] . '/ignore',
1183 'title' => DI::l10n()->t('Toggle Ignored status'),
1184 'sel' => (intval($contact['readonly']) ? 'active' : ''),
1185 'id' => 'toggle-ignore',
1188 if ($contact['uid'] != 0) {
1189 $contact_actions['archive'] = [
1190 'label' => (intval($contact['archive']) ? DI::l10n()->t('Unarchive') : DI::l10n()->t('Archive')),
1191 'url' => 'contact/' . $contact['id'] . '/archive',
1192 'title' => DI::l10n()->t('Toggle Archive status'),
1193 'sel' => (intval($contact['archive']) ? 'active' : ''),
1194 'id' => 'toggle-archive',
1197 $contact_actions['delete'] = [
1198 'label' => DI::l10n()->t('Delete'),
1199 'url' => 'contact/' . $contact['id'] . '/drop',
1200 'title' => DI::l10n()->t('Delete contact'),
1206 return $contact_actions;