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\Hook;
31 use Friendica\Core\Protocol;
32 use Friendica\Core\Renderer;
33 use Friendica\Core\Theme;
34 use Friendica\Core\Worker;
35 use Friendica\Database\DBA;
38 use Friendica\Model\User;
39 use Friendica\Module\Security\Login;
40 use Friendica\Network\HTTPException\BadRequestException;
41 use Friendica\Network\HTTPException\NotFoundException;
42 use Friendica\Util\DateTimeFormat;
43 use Friendica\Util\Strings;
46 * Manages and show Contacts and their content
48 class Contact extends BaseModule
50 const TAB_CONVERSATIONS = 1;
52 const TAB_PROFILE = 3;
53 const TAB_CONTACTS = 4;
54 const TAB_ADVANCED = 5;
56 private static function batchActions()
58 if (empty($_POST['contact_batch']) || !is_array($_POST['contact_batch'])) {
62 $redirectUrl = $_POST['redirect_url'] ?? 'contact';
64 self::checkFormSecurityTokenRedirectOnError($redirectUrl, 'contact_batch_actions');
66 $orig_records = Model\Contact::selectToArray(['id', 'uid'], ['id' => $_POST['contact_batch'], 'uid' => [0, local_user()], 'self' => false, 'deleted' => false]);
69 foreach ($orig_records as $orig_record) {
70 $cdata = Model\Contact::getPublicAndUserContactID($orig_record['id'], local_user());
75 if (!empty($_POST['contacts_batch_update']) && $cdata['user']) {
76 self::updateContactFromPoll($cdata['user']);
80 if (!empty($_POST['contacts_batch_block'])) {
81 self::toggleBlockContact($cdata['public']);
85 if (!empty($_POST['contacts_batch_ignore'])) {
86 self::toggleIgnoreContact($cdata['public']);
90 if (!empty($_POST['contacts_batch_drop']) && $cdata['user']
91 && self::dropContact($cdata['user'], local_user())
96 if ($count_actions > 0) {
97 info(DI::l10n()->tt('%d contact edited.', '%d contacts edited.', $count_actions));
100 DI::baseUrl()->redirect($redirectUrl);
103 public static function post(array $parameters = [])
109 // @TODO: Replace with parameter from router
110 if (DI::args()->getArgv()[1] === 'batch') {
111 self::batchActions();
115 // @TODO: Replace with parameter from router
116 $contact_id = intval(DI::args()->getArgv()[1]);
121 if (!DBA::exists('contact', ['id' => $contact_id, 'uid' => local_user(), 'deleted' => false])) {
122 notice(DI::l10n()->t('Could not access contact record.'));
123 DI::baseUrl()->redirect('contact');
124 return; // NOTREACHED
127 Hook::callAll('contact_edit_post', $_POST);
129 $hidden = !empty($_POST['hidden']);
131 $notify = !empty($_POST['notify']);
133 $fetch_further_information = intval($_POST['fetch_further_information'] ?? 0);
135 $remote_self = $_POST['remote_self'] ?? false;
137 $ffi_keyword_denylist = Strings::escapeHtml(trim($_POST['ffi_keyword_denylist'] ?? ''));
139 $priority = intval($_POST['poll'] ?? 0);
140 if ($priority > 5 || $priority < 0) {
144 $info = Strings::escapeHtml(trim($_POST['info'] ?? ''));
146 $r = Model\Contact::update([
147 'priority' => $priority,
150 'notify_new_posts' => $notify,
151 'fetch_further_information' => $fetch_further_information,
152 'remote_self' => $remote_self,
153 'ffi_keyword_denylist' => $ffi_keyword_denylist],
154 ['id' => $contact_id, 'uid' => local_user()]
157 if (!DBA::isResult($r)) {
158 notice(DI::l10n()->t('Failed to update contact record.'));
163 /* contact actions */
166 * @param int $contact_id Id of contact with uid != 0
167 * @throws NotFoundException
168 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
169 * @throws \ImagickException
171 private static function updateContactFromPoll(int $contact_id)
173 $contact = DBA::selectFirst('contact', ['uid', 'url', 'network'], ['id' => $contact_id, 'uid' => local_user(), 'deleted' => false]);
174 if (!DBA::isResult($contact)) {
178 if ($contact['network'] == Protocol::OSTATUS) {
179 $result = Model\Contact::createFromProbeForUser($contact['uid'], $contact['url'], $contact['network']);
181 if ($result['success']) {
182 Model\Contact::update(['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);
193 * @param int $contact_id Id of the contact with uid != 0
194 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
195 * @throws \ImagickException
197 private static function updateContactFromProbe(int $contact_id)
199 $contact = DBA::selectFirst('contact', ['url'], ['id' => $contact_id, 'uid' => local_user(), 'deleted' => false]);
200 if (!DBA::isResult($contact)) {
204 // Update the entry in the contact table
205 Model\Contact::updateFromProbe($contact_id);
209 * Toggles the blocked status of a contact identified by id.
211 * @param int $contact_id Id of the contact with uid = 0
214 private static function toggleBlockContact(int $contact_id)
216 $blocked = !Model\Contact\User::isBlocked($contact_id, local_user());
217 Model\Contact\User::setBlocked($contact_id, local_user(), $blocked);
221 * Toggles the ignored status of a contact identified by id.
223 * @param int $contact_id Id of the contact with uid = 0
226 private static function toggleIgnoreContact(int $contact_id)
228 $ignored = !Model\Contact\User::isIgnored($contact_id, local_user());
229 Model\Contact\User::setIgnored($contact_id, local_user(), $ignored);
233 * @param int $contact_id Id for contact with uid != 0
234 * @param int $uid Id for user we want to drop the contact for
236 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
237 * @throws \ImagickException
239 private static function dropContact(int $contact_id, int $uid): bool
241 $contact = Model\Contact::getContactForUser($contact_id, $uid);
242 if (!DBA::isResult($contact)) {
246 $owner = Model\User::getOwnerDataById($uid);
247 if (!DBA::isResult($owner)) {
251 Model\Contact::terminateFriendship($owner, $contact, true);
252 Model\Contact::remove($contact['id']);
257 public static function content(array $parameters = [], $update = 0)
260 return Login::form($_SERVER['REQUEST_URI']);
265 $search = Strings::escapeTags(trim($_GET['search'] ?? ''));
266 $nets = Strings::escapeTags(trim($_GET['nets'] ?? ''));
267 $rel = Strings::escapeTags(trim($_GET['rel'] ?? ''));
268 $group = Strings::escapeTags(trim($_GET['group'] ?? ''));
270 $accounttype = $_GET['accounttype'] ?? '';
271 $accounttypeid = User::getAccountTypeByString($accounttype);
275 $page->registerFooterScript(Theme::getPathForFile('asset/typeahead.js/dist/typeahead.bundle.js'));
276 $page->registerFooterScript(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.js'));
277 $page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput.css'));
278 $page->registerStylesheet(Theme::getPathForFile('js/friendica-tagsinput/friendica-tagsinput-typeahead.css'));
281 // @TODO: Replace with parameter from router
282 if (DI::args()->getArgc() == 2 && intval(DI::args()->getArgv()[1])
283 || DI::args()->getArgc() == 3 && intval(DI::args()->getArgv()[1]) && in_array(DI::args()->getArgv()[2], ['posts', 'conversations'])
285 $contact_id = intval(DI::args()->getArgv()[1]);
287 // Ensure to use the user contact when the public contact was provided
288 $data = Model\Contact::getPublicAndUserContactID($contact_id, local_user());
289 if (!empty($data['user']) && ($contact_id == $data['public'])) {
290 $contact_id = $data['user'];
294 $contact = DBA::selectFirst('contact', [], [
296 'uid' => [0, local_user()],
300 // Don't display contacts that are about to be deleted
301 if (DBA::isResult($contact) && !empty($contact['network']) && $contact['network'] == Protocol::PHANTOM) {
307 if (DBA::isResult($contact)) {
308 if ($contact['self']) {
309 // @TODO: Replace with parameter from router
310 if ((DI::args()->getArgc() == 3) && intval(DI::args()->getArgv()[1]) && in_array(DI::args()->getArgv()[2], ['posts', 'conversations'])) {
311 DI::baseUrl()->redirect('profile/' . $contact['nick']);
313 DI::baseUrl()->redirect('profile/' . $contact['nick'] . '/profile');
317 $vcard_widget = Widget\VCard::getHTML($contact);
319 $findpeople_widget = '';
321 $account_widget = '';
322 $networks_widget = '';
325 if ($contact['uid'] != 0) {
326 $groups_widget = Model\Group::sidebarWidget('contact', 'group', 'full', 'everyone', $contact_id);
332 $findpeople_widget = Widget::findPeople();
333 if (isset($_GET['add'])) {
334 $follow_widget = Widget::follow($_GET['add']);
336 $follow_widget = Widget::follow();
339 $account_widget = Widget::accounttypes($_SERVER['REQUEST_URI'], $accounttype);
340 $networks_widget = Widget::networks($_SERVER['REQUEST_URI'], $nets);
341 $rel_widget = Widget::contactRels($_SERVER['REQUEST_URI'], $rel);
342 $groups_widget = Widget::groups($_SERVER['REQUEST_URI'], $group);
345 DI::page()['aside'] .= $vcard_widget . $findpeople_widget . $follow_widget . $account_widget . $groups_widget . $networks_widget . $rel_widget;
347 $tpl = Renderer::getMarkupTemplate('contacts-head.tpl');
348 DI::page()['htmlhead'] .= Renderer::replaceMacros($tpl, [
349 '$baseurl' => DI::baseUrl()->get(true),
353 Nav::setSelected('contact');
356 notice(DI::l10n()->t('Permission denied.'));
357 return Login::form();
360 if (DI::args()->getArgc() == 3) {
361 $contact_id = intval(DI::args()->getArgv()[1]);
363 throw new BadRequestException();
366 // @TODO: Replace with parameter from router
367 $cmd = DI::args()->getArgv()[2];
369 $orig_record = DBA::selectFirst('contact', [], ['id' => $contact_id, 'uid' => [0, local_user()], 'self' => false, 'deleted' => false]);
370 if (!DBA::isResult($orig_record)) {
371 throw new NotFoundException(DI::l10n()->t('Contact not found'));
374 if ($cmd === 'posts') {
375 return self::getPostsHTML($a, $contact_id);
378 if ($cmd === 'conversations') {
379 return self::getConversationsHMTL($a, $contact_id, $update);
382 self::checkFormSecurityTokenRedirectOnError('contact/' . $contact_id, 'contact_action', 't');
384 $cdata = Model\Contact::getPublicAndUserContactID($orig_record['id'], local_user());
386 throw new NotFoundException(DI::l10n()->t('Contact not found'));
389 if ($cmd === 'update' && $cdata['user']) {
390 self::updateContactFromPoll($cdata['user']);
391 DI::baseUrl()->redirect('contact/' . $cdata['public']);
395 if ($cmd === 'updateprofile' && $cdata['user']) {
396 self::updateContactFromProbe($cdata['user']);
397 DI::baseUrl()->redirect('contact/' . $cdata['public']);
401 if ($cmd === 'block') {
402 if (public_contact() === $cdata['public']) {
403 throw new BadRequestException(DI::l10n()->t('You can\'t block yourself'));
406 self::toggleBlockContact($cdata['public']);
408 $blocked = Model\Contact\User::isBlocked($contact_id, local_user());
409 info(($blocked ? DI::l10n()->t('Contact has been blocked') : DI::l10n()->t('Contact has been unblocked')));
411 DI::baseUrl()->redirect('contact/' . $cdata['public']);
415 if ($cmd === 'ignore') {
416 if (public_contact() === $cdata['public']) {
417 throw new BadRequestException(DI::l10n()->t('You can\'t ignore yourself'));
420 self::toggleIgnoreContact($cdata['public']);
422 $ignored = Model\Contact\User::isIgnored($cdata['public'], local_user());
423 info(($ignored ? DI::l10n()->t('Contact has been ignored') : DI::l10n()->t('Contact has been unignored')));
425 DI::baseUrl()->redirect('contact/' . $cdata['public']);
429 if ($cmd === 'drop' && $cdata['user']) {
430 // Check if we should do HTML-based delete confirmation
431 if (!empty($_REQUEST['confirm'])) {
432 DI::page()['aside'] = '';
434 return Renderer::replaceMacros(Renderer::getMarkupTemplate('contact_drop_confirm.tpl'), [
436 'header' => DI::l10n()->t('Drop contact'),
437 'message' => DI::l10n()->t('Do you really want to delete this contact?'),
438 'confirm' => DI::l10n()->t('Yes'),
439 'cancel' => DI::l10n()->t('Cancel'),
441 '$contact' => self::getContactTemplateVars($orig_record),
443 '$confirm_url' => DI::args()->getCommand(),
444 '$confirm_name' => 't',
445 '$confirm_value' => BaseModule::getFormSecurityToken('contact_action'),
448 // Now check how the user responded to the confirmation query
449 if (!empty($_REQUEST['canceled'])) {
450 DI::baseUrl()->redirect('contact');
453 if (self::dropContact($cdata['user'], local_user())) {
454 info(DI::l10n()->t('Contact has been removed.'));
457 DI::baseUrl()->redirect('contact');
462 $_SESSION['return_path'] = DI::args()->getQueryString();
464 if (!empty($contact)) {
465 DI::page()['htmlhead'] .= Renderer::replaceMacros(Renderer::getMarkupTemplate('contact_head.tpl'), [
466 '$baseurl' => DI::baseUrl()->get(true),
469 $contact['blocked'] = Model\Contact\User::isBlocked($contact['id'], local_user());
470 $contact['readonly'] = Model\Contact\User::isIgnored($contact['id'], local_user());
473 switch ($contact['rel']) {
474 case Model\Contact::FRIEND:
475 $relation_text = DI::l10n()->t('You are mutual friends with %s');
478 case Model\Contact::FOLLOWER;
479 $relation_text = DI::l10n()->t('You are sharing with %s');
482 case Model\Contact::SHARING;
483 $relation_text = DI::l10n()->t('%s is sharing with you');
490 if ($contact['uid'] == 0) {
494 if (!in_array($contact['network'], array_merge(Protocol::FEDERATED, [Protocol::TWITTER]))) {
498 $relation_text = sprintf($relation_text, $contact['name']);
500 $url = Model\Contact::magicLinkByContact($contact);
501 if (strpos($url, 'redir/') === 0) {
502 $sparkle = ' class="sparkle" ';
507 $insecure = DI::l10n()->t('Private communications are not available for this contact.');
509 $last_update = (($contact['last-update'] <= DBA::NULL_DATETIME) ? DI::l10n()->t('Never') : DateTimeFormat::local($contact['last-update'], 'D, j M Y, g:i A'));
511 if ($contact['last-update'] > DBA::NULL_DATETIME) {
512 $last_update .= ' ' . ($contact['failed'] ? DI::l10n()->t('(Update was not successful)') : DI::l10n()->t('(Update was successful)'));
514 $lblsuggest = (($contact['network'] === Protocol::DFRN) ? DI::l10n()->t('Suggest friends') : '');
516 $poll_enabled = in_array($contact['network'], [Protocol::DFRN, Protocol::OSTATUS, Protocol::FEED, Protocol::MAIL]);
518 $nettype = DI::l10n()->t('Network type: %s', ContactSelector::networkToName($contact['network'], $contact['url'], $contact['protocol'], $contact['gsid']));
521 $tab_str = self::getTabsHTML($contact, self::TAB_PROFILE);
523 $lost_contact = (($contact['archive'] && $contact['term-date'] > DBA::NULL_DATETIME && $contact['term-date'] < DateTimeFormat::utcNow()) ? DI::l10n()->t('Communications lost with this contact!') : '');
525 $fetch_further_information = null;
526 if ($contact['network'] == Protocol::FEED) {
527 $fetch_further_information = [
528 'fetch_further_information',
529 DI::l10n()->t('Fetch further information for feeds'),
530 $contact['fetch_further_information'],
531 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.'),
533 '0' => DI::l10n()->t('Disabled'),
534 '1' => DI::l10n()->t('Fetch information'),
535 '3' => DI::l10n()->t('Fetch keywords'),
536 '2' => DI::l10n()->t('Fetch information and keywords')
541 // Disable remote self for everything except feeds.
542 // There is an issue when you repeat an item from maybe twitter and you got comments from friendica and twitter
543 // Problem is, you couldn't reply to both networks.
544 $allow_remote_self = in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::FEED, Protocol::DFRN, Protocol::DIASPORA, Protocol::TWITTER])
545 && DI::config()->get('system', 'allow_users_remote_self');
547 if ($contact['network'] == Protocol::FEED) {
548 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
549 Model\Contact::MIRROR_FORWARDED => DI::l10n()->t('Mirror as forwarded posting'),
550 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting')];
551 } elseif (in_array($contact['network'], [Protocol::ACTIVITYPUB])) {
552 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
553 Model\Contact::MIRROR_NATIVE_RESHARE => DI::l10n()->t('Native reshare')];
554 } elseif (in_array($contact['network'], [Protocol::DFRN])) {
555 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
556 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting'),
557 Model\Contact::MIRROR_NATIVE_RESHARE => DI::l10n()->t('Native reshare')];
559 $remote_self_options = [Model\Contact::MIRROR_DEACTIVATED => DI::l10n()->t('No mirroring'),
560 Model\Contact::MIRROR_OWN_POST => DI::l10n()->t('Mirror as my own posting')];
563 $poll_interval = null;
564 if ((($contact['network'] == Protocol::FEED) && !DI::config()->get('system', 'adjust_poll_frequency')) || ($contact['network'] == Protocol::MAIL)) {
565 $poll_interval = ContactSelector::pollInterval($contact['priority'], !$poll_enabled);
568 // Load contactact related actions like hide, suggest, delete and others
569 $contact_actions = self::getContactActions($contact);
571 if ($contact['uid'] != 0) {
572 $lbl_info1 = DI::l10n()->t('Contact Information / Notes');
573 $contact_settings_label = DI::l10n()->t('Contact Settings');
576 $contact_settings_label = null;
579 $tpl = Renderer::getMarkupTemplate('contact_edit.tpl');
580 $o .= Renderer::replaceMacros($tpl, [
581 '$header' => DI::l10n()->t('Contact'),
582 '$tab_str' => $tab_str,
583 '$submit' => DI::l10n()->t('Submit'),
584 '$lbl_info1' => $lbl_info1,
585 '$lbl_info2' => DI::l10n()->t('Their personal note'),
586 '$reason' => trim(Strings::escapeTags($contact['reason'])),
587 '$infedit' => DI::l10n()->t('Edit contact notes'),
588 '$common_link' => 'contact/' . $contact['id'] . '/contacts/common',
589 '$relation_text' => $relation_text,
590 '$visit' => DI::l10n()->t('Visit %s\'s profile [%s]', $contact['name'], $contact['url']),
591 '$blockunblock' => DI::l10n()->t('Block/Unblock contact'),
592 '$ignorecont' => DI::l10n()->t('Ignore contact'),
593 '$lblrecent' => DI::l10n()->t('View conversations'),
594 '$lblsuggest' => $lblsuggest,
595 '$nettype' => $nettype,
596 '$poll_interval' => $poll_interval,
597 '$poll_enabled' => $poll_enabled,
598 '$lastupdtext' => DI::l10n()->t('Last update:'),
599 '$lost_contact' => $lost_contact,
600 '$updpub' => DI::l10n()->t('Update public posts'),
601 '$last_update' => $last_update,
602 '$udnow' => DI::l10n()->t('Update now'),
603 '$contact_id' => $contact['id'],
604 '$block_text' => ($contact['blocked'] ? DI::l10n()->t('Unblock') : DI::l10n()->t('Block')),
605 '$ignore_text' => ($contact['readonly'] ? DI::l10n()->t('Unignore') : DI::l10n()->t('Ignore')),
606 '$insecure' => (in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::DFRN, Protocol::MAIL, Protocol::DIASPORA]) ? '' : $insecure),
607 '$info' => $contact['info'],
608 '$cinfo' => ['info', '', $contact['info'], ''],
609 '$blocked' => ($contact['blocked'] ? DI::l10n()->t('Currently blocked') : ''),
610 '$ignored' => ($contact['readonly'] ? DI::l10n()->t('Currently ignored') : ''),
611 '$archived' => ($contact['archive'] ? DI::l10n()->t('Currently archived') : ''),
612 '$pending' => ($contact['pending'] ? DI::l10n()->t('Awaiting connection acknowledge') : ''),
613 '$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')],
614 '$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')],
615 '$fetch_further_information' => $fetch_further_information,
616 '$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')],
617 '$photo' => Model\Contact::getPhoto($contact),
618 '$name' => $contact['name'],
619 '$sparkle' => $sparkle,
621 '$profileurllabel'=> DI::l10n()->t('Profile URL'),
622 '$profileurl' => $contact['url'],
623 '$account_type' => Model\Contact::getAccountType($contact),
624 '$location' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['location']),
625 '$location_label' => DI::l10n()->t('Location:'),
626 '$xmpp' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['xmpp']),
627 '$xmpp_label' => DI::l10n()->t('XMPP:'),
628 '$matrix' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['matrix']),
629 '$matrix_label' => DI::l10n()->t('Matrix:'),
630 '$about' => BBCode::convertForUriId($contact['uri-id'] ?? 0, $contact['about'], BBCode::EXTERNAL),
631 '$about_label' => DI::l10n()->t('About:'),
632 '$keywords' => $contact['keywords'],
633 '$keywords_label' => DI::l10n()->t('Tags:'),
634 '$contact_action_button' => DI::l10n()->t('Actions'),
635 '$contact_actions'=> $contact_actions,
636 '$contact_status' => DI::l10n()->t('Status'),
637 '$contact_settings_label' => $contact_settings_label,
638 '$contact_profile_label' => DI::l10n()->t('Profile'),
639 '$allow_remote_self' => $allow_remote_self,
640 '$remote_self' => ['remote_self',
641 DI::l10n()->t('Mirror postings from this contact'),
642 $contact['remote_self'],
643 DI::l10n()->t('Mark this contact as remote_self, this will cause friendica to repost new entries from this contact.'),
648 $arr = ['contact' => $contact, 'output' => $o];
650 Hook::callAll('contact_edit', $arr);
652 return $arr['output'];
655 $sql_values = [local_user()];
657 // @TODO: Replace with parameter from router
658 $type = DI::args()->getArgv()[1] ?? '';
662 $sql_extra = " AND EXISTS(SELECT `id` from `user-contact` WHERE `contact`.`id` = `user-contact`.`cid` and `user-contact`.`uid` = ? and `user-contact`.`blocked`)";
663 // This makes the query look for contact.uid = 0
664 array_unshift($sql_values, 0);
667 $sql_extra = " AND `hidden` AND NOT `blocked` AND NOT `pending`";
670 $sql_extra = " AND EXISTS(SELECT `id` from `user-contact` WHERE `contact`.`id` = `user-contact`.`cid` and `user-contact`.`uid` = ? and `user-contact`.`ignored`)";
671 // This makes the query look for contact.uid = 0
672 array_unshift($sql_values, 0);
675 $sql_extra = " AND `archive` AND NOT `blocked` AND NOT `pending`";
678 $sql_extra = " AND `pending` AND NOT `archive` AND NOT `failed` AND ((`rel` = ?)
679 OR EXISTS (SELECT `id` FROM `intro` WHERE `contact-id` = `contact`.`id` AND NOT `ignore`))";
680 $sql_values[] = Model\Contact::SHARING;
683 $sql_extra = " AND NOT `archive` AND NOT `blocked` AND NOT `pending`";
687 if (isset($accounttypeid)) {
688 $sql_extra .= " AND `contact-type` = ?";
689 $sql_values[] = $accounttypeid;
696 $search_hdr = $search;
697 $search_txt = preg_quote($search);
698 $sql_extra .= " AND (name REGEXP ? OR url REGEXP ? OR nick REGEXP ?)";
699 $sql_values[] = $search_txt;
700 $sql_values[] = $search_txt;
701 $sql_values[] = $search_txt;
705 $sql_extra .= " AND network = ? ";
706 $sql_values[] = $nets;
711 $sql_extra .= " AND `rel` IN (?, ?)";
712 $sql_values[] = Model\Contact::FOLLOWER;
713 $sql_values[] = Model\Contact::FRIEND;
716 $sql_extra .= " AND `rel` IN (?, ?)";
717 $sql_values[] = Model\Contact::SHARING;
718 $sql_values[] = Model\Contact::FRIEND;
721 $sql_extra .= " AND `rel` = ?";
722 $sql_values[] = Model\Contact::FRIEND;
727 $sql_extra = " AND EXISTS(SELECT `id` FROM `group_member` WHERE `gid` = ? AND `contact`.`id` = `contact-id`)";
728 $sql_values[] = $group;
732 $stmt = DBA::p("SELECT COUNT(*) AS `total`
738 " . Widget::unavailableNetworks(),
741 if (DBA::isResult($stmt)) {
742 $total = DBA::fetch($stmt)['total'];
746 $pager = new Pager(DI::l10n(), DI::args()->getQueryString());
748 $sql_values[] = $pager->getStart();
749 $sql_values[] = $pager->getItemsPerPage();
753 $stmt = DBA::p("SELECT *
763 while ($contact = DBA::fetch($stmt)) {
764 $contact['blocked'] = Model\Contact\User::isBlocked($contact['id'], local_user());
765 $contact['readonly'] = Model\Contact\User::isIgnored($contact['id'], local_user());
766 $contacts[] = self::getContactTemplateVars($contact);
772 'label' => DI::l10n()->t('All Contacts'),
774 'sel' => !$type ? 'active' : '',
775 'title' => DI::l10n()->t('Show all contacts'),
776 'id' => 'showall-tab',
780 'label' => DI::l10n()->t('Pending'),
781 'url' => 'contact/pending',
782 'sel' => $type == 'pending' ? 'active' : '',
783 'title' => DI::l10n()->t('Only show pending contacts'),
784 'id' => 'showpending-tab',
788 'label' => DI::l10n()->t('Blocked'),
789 'url' => 'contact/blocked',
790 'sel' => $type == 'blocked' ? 'active' : '',
791 'title' => DI::l10n()->t('Only show blocked contacts'),
792 'id' => 'showblocked-tab',
796 'label' => DI::l10n()->t('Ignored'),
797 'url' => 'contact/ignored',
798 'sel' => $type == 'ignored' ? 'active' : '',
799 'title' => DI::l10n()->t('Only show ignored contacts'),
800 'id' => 'showignored-tab',
804 'label' => DI::l10n()->t('Archived'),
805 'url' => 'contact/archived',
806 'sel' => $type == 'archived' ? 'active' : '',
807 'title' => DI::l10n()->t('Only show archived contacts'),
808 'id' => 'showarchived-tab',
812 'label' => DI::l10n()->t('Hidden'),
813 'url' => 'contact/hidden',
814 'sel' => $type == 'hidden' ? 'active' : '',
815 'title' => DI::l10n()->t('Only show hidden contacts'),
816 'id' => 'showhidden-tab',
820 'label' => DI::l10n()->t('Groups'),
823 'title' => DI::l10n()->t('Organize your contact groups'),
824 'id' => 'contactgroups-tab',
829 $tabs_tpl = Renderer::getMarkupTemplate('common_tabs.tpl');
830 $tabs_html = Renderer::replaceMacros($tabs_tpl, ['$tabs' => $tabs]);
833 case 'followers': $header = DI::l10n()->t('Followers'); break;
834 case 'following': $header = DI::l10n()->t('Following'); break;
835 case 'mutuals': $header = DI::l10n()->t('Mutual friends'); break;
836 default: $header = DI::l10n()->t('Contacts');
840 case 'pending': $header .= ' - ' . DI::l10n()->t('Pending'); break;
841 case 'blocked': $header .= ' - ' . DI::l10n()->t('Blocked'); break;
842 case 'hidden': $header .= ' - ' . DI::l10n()->t('Hidden'); break;
843 case 'ignored': $header .= ' - ' . DI::l10n()->t('Ignored'); break;
844 case 'archived': $header .= ' - ' . DI::l10n()->t('Archived'); break;
847 $header .= $nets ? ' - ' . ContactSelector::networkToName($nets) : '';
849 $tpl = Renderer::getMarkupTemplate('contacts-template.tpl');
850 $o .= Renderer::replaceMacros($tpl, [
851 '$header' => $header,
852 '$tabs' => $tabs_html,
854 '$search' => $search_hdr,
855 '$desc' => DI::l10n()->t('Search your contacts'),
856 '$finding' => $searching ? DI::l10n()->t('Results for: %s', $search) : '',
857 '$submit' => DI::l10n()->t('Find'),
858 '$cmd' => DI::args()->getCommand(),
859 '$contacts' => $contacts,
860 '$form_security_token' => BaseModule::getFormSecurityToken('contact_batch_actions'),
861 '$contact_drop_confirm' => DI::l10n()->t('Do you really want to delete this contact?'),
863 '$batch_actions' => [
864 'contacts_batch_update' => DI::l10n()->t('Update'),
865 'contacts_batch_block' => DI::l10n()->t('Block') . '/' . DI::l10n()->t('Unblock'),
866 'contacts_batch_ignore' => DI::l10n()->t('Ignore') . '/' . DI::l10n()->t('Unignore'),
867 'contacts_batch_drop' => DI::l10n()->t('Delete'),
869 '$h_batch_actions' => DI::l10n()->t('Batch Actions'),
870 '$paginate' => $pager->renderFull($total),
877 * List of pages for the Contact TabBar
879 * Available Pages are 'Status', 'Profile', 'Contacts' and 'Common Friends'
881 * @param array $contact The contact array
882 * @param int $active_tab 1 if tab should be marked as active
884 * @return string HTML string of the contact page tabs buttons.
885 * @throws \Friendica\Network\HTTPException\InternalServerErrorException
886 * @throws \ImagickException
888 public static function getTabsHTML(array $contact, int $active_tab)
890 $cid = $pcid = $contact['id'];
891 $data = Model\Contact::getPublicAndUserContactID($contact['id'], local_user());
892 if (!empty($data['user']) && ($contact['id'] == $data['public'])) {
893 $cid = $data['user'];
894 } elseif (!empty($data['public'])) {
895 $pcid = $data['public'];
901 'label' => DI::l10n()->t('Status'),
902 'url' => 'contact/' . $pcid . '/conversations',
903 'sel' => (($active_tab == self::TAB_CONVERSATIONS) ? 'active' : ''),
904 'title' => DI::l10n()->t('Conversations started by this contact'),
905 'id' => 'status-tab',
909 'label' => DI::l10n()->t('Posts and Comments'),
910 'url' => 'contact/' . $pcid . '/posts',
911 'sel' => (($active_tab == self::TAB_POSTS) ? 'active' : ''),
912 'title' => DI::l10n()->t('Status Messages and Posts'),
917 'label' => DI::l10n()->t('Profile'),
918 'url' => 'contact/' . $cid,
919 'sel' => (($active_tab == self::TAB_PROFILE) ? 'active' : ''),
920 'title' => DI::l10n()->t('Profile Details'),
921 'id' => 'profile-tab',
924 ['label' => DI::l10n()->t('Contacts'),
925 'url' => 'contact/' . $pcid . '/contacts',
926 'sel' => (($active_tab == self::TAB_CONTACTS) ? 'active' : ''),
927 'title' => DI::l10n()->t('View all known contacts'),
928 'id' => 'contacts-tab',
933 if (!empty($contact['network']) && in_array($contact['network'], [Protocol::FEED, Protocol::MAIL]) && ($cid != $pcid)) {
934 $tabs[] = ['label' => DI::l10n()->t('Advanced'),
935 'url' => 'contact/' . $cid . '/advanced/',
936 'sel' => (($active_tab == self::TAB_ADVANCED) ? 'active' : ''),
937 'title' => DI::l10n()->t('Advanced Contact Settings'),
938 'id' => 'advanced-tab',
943 $tab_tpl = Renderer::getMarkupTemplate('common_tabs.tpl');
944 $tab_str = Renderer::replaceMacros($tab_tpl, ['$tabs' => $tabs]);
949 public static function getConversationsHMTL($a, $contact_id, $update, $parent = 0)
954 // We need the editor here to be able to reshare an item.
956 $o = DI::conversation()->statusEditor([], 0, true);
960 $contact = DBA::selectFirst('contact', ['uid', 'url', 'id'], ['id' => $contact_id, 'deleted' => false]);
963 $o .= self::getTabsHTML($contact, self::TAB_CONVERSATIONS);
966 if (DBA::isResult($contact)) {
968 $profiledata = Model\Contact::getByURLForUser($contact['url'], local_user());
969 DI::page()['aside'] = Widget\VCard::getHTML($profiledata);
971 DI::page()['aside'] = '';
974 if ($contact['uid'] == 0) {
975 $o .= Model\Contact::getPostsFromId($contact['id'], true, $update, $parent);
977 $o .= Model\Contact::getPostsFromUrl($contact['url'], true, $update, $parent);
984 private static function getPostsHTML($a, $contact_id)
986 $contact = DBA::selectFirst('contact', ['uid', 'url', 'id'], ['id' => $contact_id, 'deleted' => false]);
988 $o = self::getTabsHTML($contact, self::TAB_POSTS);
990 if (DBA::isResult($contact)) {
991 $profiledata = Model\Contact::getByURLForUser($contact['url'], local_user());
993 if (local_user() && in_array($profiledata['network'], Protocol::FEDERATED)) {
994 $profiledata['remoteconnect'] = DI::baseUrl() . '/follow?url=' . urlencode($profiledata['url']);
997 DI::page()['aside'] = Widget\VCard::getHTML($profiledata);
999 if ($contact['uid'] == 0) {
1000 $o .= Model\Contact::getPostsFromId($contact['id']);
1002 $o .= Model\Contact::getPostsFromUrl($contact['url']);
1010 * Return the fields for the contact template
1012 * @param array $contact Contact array
1013 * @return array Template fields
1015 public static function getContactTemplateVars(array $contact)
1019 if (!empty($contact['url']) && isset($contact['uid']) && ($contact['uid'] == 0) && local_user()) {
1020 $personal = Model\Contact::getByURL($contact['url'], false, ['uid', 'rel', 'self'], local_user());
1021 if (!empty($personal)) {
1022 $contact['uid'] = $personal['uid'];
1023 $contact['rel'] = $personal['rel'];
1024 $contact['self'] = $personal['self'];
1028 if (!empty($contact['uid']) && !empty($contact['rel']) && local_user() == $contact['uid']) {
1029 switch ($contact['rel']) {
1030 case Model\Contact::FRIEND:
1031 $alt_text = DI::l10n()->t('Mutual Friendship');
1034 case Model\Contact::FOLLOWER;
1035 $alt_text = DI::l10n()->t('is a fan of yours');
1038 case Model\Contact::SHARING;
1039 $alt_text = DI::l10n()->t('you are a fan of');
1047 $url = Model\Contact::magicLinkByContact($contact);
1049 if (strpos($url, 'redir/') === 0) {
1050 $sparkle = ' class="sparkle" ';
1055 if ($contact['pending']) {
1056 if (in_array($contact['rel'], [Model\Contact::FRIEND, Model\Contact::SHARING])) {
1057 $alt_text = DI::l10n()->t('Pending outgoing contact request');
1059 $alt_text = DI::l10n()->t('Pending incoming contact request');
1063 if ($contact['self']) {
1064 $alt_text = DI::l10n()->t('This is you');
1065 $url = $contact['url'];
1070 'id' => $contact['id'],
1072 'img_hover' => DI::l10n()->t('Visit %s\'s profile [%s]', $contact['name'], $contact['url']),
1073 'photo_menu' => Model\Contact::photoMenu($contact),
1074 'thumb' => Model\Contact::getThumb($contact, true),
1075 'alt_text' => $alt_text,
1076 'name' => $contact['name'],
1077 'nick' => $contact['nick'],
1078 'details' => $contact['location'],
1079 'tags' => $contact['keywords'],
1080 'about' => $contact['about'],
1081 'account_type' => Model\Contact::getAccountType($contact),
1082 'sparkle' => $sparkle,
1083 'itemurl' => ($contact['addr'] ?? '') ?: $contact['url'],
1084 'network' => ContactSelector::networkToName($contact['network'], $contact['url'], $contact['protocol'], $contact['gsid']),
1089 * Gives a array with actions which can performed to a given contact
1091 * This includes actions like e.g. 'block', 'hide', 'delete' and others
1093 * @param array $contact Data about the Contact
1094 * @return array with contact related actions
1096 private static function getContactActions($contact)
1098 $poll_enabled = in_array($contact['network'], [Protocol::ACTIVITYPUB, Protocol::DFRN, Protocol::OSTATUS, Protocol::FEED, Protocol::MAIL]);
1099 $contact_actions = [];
1101 $formSecurityToken = self::getFormSecurityToken('contact_action');
1103 // Provide friend suggestion only for Friendica contacts
1104 if ($contact['network'] === Protocol::DFRN) {
1105 $contact_actions['suggest'] = [
1106 'label' => DI::l10n()->t('Suggest friends'),
1107 'url' => 'fsuggest/' . $contact['id'],
1114 if ($poll_enabled) {
1115 $contact_actions['update'] = [
1116 'label' => DI::l10n()->t('Update now'),
1117 'url' => 'contact/' . $contact['id'] . '/update?t=' . $formSecurityToken,
1124 if (in_array($contact['network'], Protocol::NATIVE_SUPPORT)) {
1125 $contact_actions['updateprofile'] = [
1126 'label' => DI::l10n()->t('Refetch contact data'),
1127 'url' => 'contact/' . $contact['id'] . '/updateprofile?t=' . $formSecurityToken,
1130 'id' => 'updateprofile',
1134 $contact_actions['block'] = [
1135 'label' => (intval($contact['blocked']) ? DI::l10n()->t('Unblock') : DI::l10n()->t('Block')),
1136 'url' => 'contact/' . $contact['id'] . '/block?t=' . $formSecurityToken,
1137 'title' => DI::l10n()->t('Toggle Blocked status'),
1138 'sel' => (intval($contact['blocked']) ? 'active' : ''),
1139 'id' => 'toggle-block',
1142 $contact_actions['ignore'] = [
1143 'label' => (intval($contact['readonly']) ? DI::l10n()->t('Unignore') : DI::l10n()->t('Ignore')),
1144 'url' => 'contact/' . $contact['id'] . '/ignore?t=' . $formSecurityToken,
1145 'title' => DI::l10n()->t('Toggle Ignored status'),
1146 'sel' => (intval($contact['readonly']) ? 'active' : ''),
1147 'id' => 'toggle-ignore',
1150 if ($contact['uid'] != 0) {
1151 if (Protocol::supportsRevokeFollow($contact['network']) && in_array($contact['rel'], [Model\Contact::FOLLOWER, Model\Contact::FRIEND])) {
1152 $contact_actions['revoke_follow'] = [
1153 'label' => DI::l10n()->t('Revoke Follow'),
1154 'url' => 'contact/' . $contact['id'] . '/revoke',
1155 'title' => DI::l10n()->t('Revoke the follow from this contact'),
1157 'id' => 'revoke_follow',
1161 $contact_actions['delete'] = [
1162 'label' => DI::l10n()->t('Delete'),
1163 'url' => 'contact/' . $contact['id'] . '/drop?t=' . $formSecurityToken,
1164 'title' => DI::l10n()->t('Delete contact'),
1170 return $contact_actions;