2011-04-11 06:21:16 +02:00
< ? php
2017-11-16 19:24:59 +01:00
/**
2020-02-09 16:18:46 +01:00
* @ copyright Copyright ( C ) 2020 , Friendica
*
* @ license GNU AGPL version 3 or any later version
*
* This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation , either version 3 of the
* License , or ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU Affero General Public License for more details .
*
* You should have received a copy of the GNU Affero General Public License
* along with this program . If not , see < https :// www . gnu . org / licenses />.
*
2017-11-16 19:24:59 +01:00
*/
2018-01-25 03:08:45 +01:00
2017-04-30 06:07:00 +02:00
use Friendica\App ;
2018-01-10 04:42:04 +01:00
use Friendica\Content\ContactSelector ;
2017-12-04 15:04:36 +01:00
use Friendica\Content\Feature ;
2018-02-05 01:23:49 +01:00
use Friendica\Content\Text\BBCode ;
2018-12-26 07:06:24 +01:00
use Friendica\Core\Hook ;
2018-10-29 22:20:46 +01:00
use Friendica\Core\Logger ;
2018-08-11 22:40:44 +02:00
use Friendica\Core\Protocol ;
2018-10-31 15:35:50 +01:00
use Friendica\Core\Renderer ;
2019-05-26 22:15:38 +02:00
use Friendica\Core\Session ;
2020-09-03 16:03:36 +02:00
use Friendica\Core\Theme ;
2018-07-20 14:19:26 +02:00
use Friendica\Database\DBA ;
2019-12-15 23:28:01 +01:00
use Friendica\DI ;
2017-12-08 05:33:36 +01:00
use Friendica\Model\Contact ;
2018-06-10 09:26:37 +02:00
use Friendica\Model\Item ;
2018-07-20 04:15:21 +02:00
use Friendica\Model\Profile ;
2020-04-26 17:24:58 +02:00
use Friendica\Model\Tag ;
2020-05-26 07:18:50 +02:00
use Friendica\Model\Verb ;
2017-12-08 05:33:36 +01:00
use Friendica\Object\Post ;
2018-01-25 03:08:45 +01:00
use Friendica\Object\Thread ;
2019-10-23 02:05:11 +02:00
use Friendica\Protocol\Activity ;
2019-10-25 00:10:20 +02:00
use Friendica\Util\Crypto ;
2018-01-27 03:38:34 +01:00
use Friendica\Util\DateTimeFormat ;
2018-11-08 16:14:37 +01:00
use Friendica\Util\Strings ;
2019-10-25 00:10:20 +02:00
use Friendica\Util\Temporal ;
2018-02-03 18:25:58 +01:00
use Friendica\Util\XML ;
2017-04-30 06:07:00 +02:00
2012-07-08 00:20:24 +02:00
function item_extract_images ( $body ) {
2018-01-15 14:05:12 +01:00
$saved_image = [];
2012-09-10 10:36:30 +02:00
$orig_body = $body ;
$new_body = '' ;
2012-07-08 00:20:24 +02:00
2012-09-10 10:36:30 +02:00
$cnt = 0 ;
$img_start = strpos ( $orig_body , '[img' );
$img_st_close = ( $img_start !== false ? strpos ( substr ( $orig_body , $img_start ), ']' ) : false );
$img_end = ( $img_start !== false ? strpos ( substr ( $orig_body , $img_start ), '[/img]' ) : false );
2017-04-04 19:46:56 +02:00
while (( $img_st_close !== false ) && ( $img_end !== false )) {
2012-07-08 00:20:24 +02:00
2012-09-10 10:36:30 +02:00
$img_st_close ++ ; // make it point to AFTER the closing bracket
$img_end += $img_start ;
2012-07-08 00:20:24 +02:00
2018-02-27 10:29:11 +01:00
if ( ! strcmp ( substr ( $orig_body , $img_start + $img_st_close , 5 ), 'data:' )) {
2012-09-10 10:36:30 +02:00
// This is an embedded image
2012-07-08 00:20:24 +02:00
2012-09-10 10:36:30 +02:00
$saved_image [ $cnt ] = substr ( $orig_body , $img_start + $img_st_close , $img_end - ( $img_start + $img_st_close ));
$new_body = $new_body . substr ( $orig_body , 0 , $img_start ) . '[!#saved_image' . $cnt . '#!]' ;
2012-07-08 02:47:13 +02:00
2012-09-10 10:36:30 +02:00
$cnt ++ ;
2017-04-08 19:05:50 +02:00
} else {
2012-09-10 10:36:30 +02:00
$new_body = $new_body . substr ( $orig_body , 0 , $img_end + strlen ( '[/img]' ));
2017-04-08 19:05:50 +02:00
}
2012-07-08 00:20:24 +02:00
2012-09-10 10:36:30 +02:00
$orig_body = substr ( $orig_body , $img_end + strlen ( '[/img]' ));
2012-07-08 00:20:24 +02:00
2017-04-08 19:05:50 +02:00
if ( $orig_body === false ) {
// in case the body ends on a closing image tag
2012-09-10 10:36:30 +02:00
$orig_body = '' ;
2017-04-08 19:05:50 +02:00
}
2012-07-08 00:20:24 +02:00
2012-09-10 10:36:30 +02:00
$img_start = strpos ( $orig_body , '[img' );
$img_st_close = ( $img_start !== false ? strpos ( substr ( $orig_body , $img_start ), ']' ) : false );
$img_end = ( $img_start !== false ? strpos ( substr ( $orig_body , $img_start ), '[/img]' ) : false );
}
2012-07-08 00:20:24 +02:00
2012-09-10 10:36:30 +02:00
$new_body = $new_body . $orig_body ;
2012-07-08 00:20:24 +02:00
2018-01-15 14:05:12 +01:00
return [ 'body' => $new_body , 'images' => $saved_image ];
2017-12-12 05:52:33 +01:00
}
2012-07-08 00:20:24 +02:00
function item_redir_and_replace_images ( $body , $images , $cid ) {
2012-09-10 10:36:30 +02:00
$origbody = $body ;
$newbody = '' ;
$cnt = 1 ;
2018-02-05 01:23:49 +01:00
$pos = BBCode :: getTagPosition ( $origbody , 'url' , 0 );
2017-04-04 19:46:56 +02:00
while ( $pos !== false && $cnt < 1000 ) {
2012-09-10 10:36:30 +02:00
$search = '/\[url\=(.*?)\]\[!#saved_image([0-9]*)#!\]\[\/url\]' . '/is' ;
2019-12-30 23:00:08 +01:00
$replace = '[url=' . DI :: baseUrl () . '/redir/' . $cid
2020-01-13 21:10:13 +01:00
. '?url=' . '$1' . '][!#saved_image' . '$2' . '#!][/url]' ;
2012-09-10 10:36:30 +02:00
$newbody .= substr ( $origbody , 0 , $pos [ 'start' ][ 'open' ]);
$subject = substr ( $origbody , $pos [ 'start' ][ 'open' ], $pos [ 'end' ][ 'close' ] - $pos [ 'start' ][ 'open' ]);
$origbody = substr ( $origbody , $pos [ 'end' ][ 'close' ]);
2017-04-08 20:00:54 +02:00
if ( $origbody === false ) {
2012-09-10 10:36:30 +02:00
$origbody = '' ;
2017-04-08 20:00:54 +02:00
}
2012-09-10 10:36:30 +02:00
$subject = preg_replace ( $search , $replace , $subject );
$newbody .= $subject ;
$cnt ++ ;
2018-02-05 01:23:49 +01:00
// Isn't this supposed to use $cnt value for $occurrences? - @MrPetovan
$pos = BBCode :: getTagPosition ( $origbody , 'url' , 0 );
2012-09-10 10:36:30 +02:00
}
$newbody .= $origbody ;
$cnt = 0 ;
2016-12-20 21:13:50 +01:00
foreach ( $images as $image ) {
2017-04-08 19:05:50 +02:00
/*
* We 're depending on the property of ' foreach ' ( specified on the PHP website ) that
* it loops over the array starting from the first element and going sequentially
* to the last element .
*/
2012-09-10 10:36:30 +02:00
$newbody = str_replace ( '[!#saved_image' . $cnt . '#!]' , '[img]' . $image . '[/img]' , $newbody );
$cnt ++ ;
}
return $newbody ;
2017-12-12 05:52:33 +01:00
}
2012-07-08 00:20:24 +02:00
2011-04-18 17:37:02 +02:00
/**
* Render actions localized
2019-01-07 16:24:06 +01:00
*
* @ param $item
* @ throws ImagickException
* @ throws \Friendica\Network\HTTPException\InternalServerErrorException
2011-04-18 17:37:02 +02:00
*/
2018-06-19 21:06:17 +02:00
function localize_item ( & $item )
{
2012-09-10 10:36:30 +02:00
$extracted = item_extract_images ( $item [ 'body' ]);
2017-04-08 20:00:54 +02:00
if ( $extracted [ 'images' ]) {
2012-09-10 10:36:30 +02:00
$item [ 'body' ] = item_redir_and_replace_images ( $extracted [ 'body' ], $extracted [ 'images' ], $item [ 'contact-id' ]);
2017-04-08 20:00:54 +02:00
}
2012-09-10 10:36:30 +02:00
2020-04-04 22:00:40 +02:00
/// @todo The following functionality needs to be cleaned up.
if ( ! empty ( $item [ 'verb' ])) {
$activity = DI :: activity ();
2020-04-05 19:25:36 +02:00
$xmlhead = " < " . " ?xml version='1.0' encoding='UTF-8' ? " . " > " ;
2020-04-04 22:00:40 +02:00
if ( stristr ( $item [ 'verb' ], Activity :: POKE )) {
2020-04-05 09:08:20 +02:00
$verb = urldecode ( substr ( $item [ 'verb' ], strpos ( $item [ 'verb' ], '#' ) + 1 ));
2020-04-04 22:00:40 +02:00
if ( ! $verb ) {
return ;
}
2020-04-05 09:08:20 +02:00
if ( $item [ 'object-type' ] == " " || $item [ 'object-type' ] !== Activity\ObjectType :: PERSON ) {
2020-04-04 22:00:40 +02:00
return ;
2012-09-10 10:36:30 +02:00
}
2020-04-04 22:00:40 +02:00
$Aname = $item [ 'author-name' ];
$Alink = $item [ 'author-link' ];
2012-09-10 10:36:30 +02:00
2020-04-05 09:08:20 +02:00
$obj = XML :: parseString ( $xmlhead . $item [ 'object' ]);
2012-09-10 10:36:30 +02:00
2020-04-04 22:00:40 +02:00
$Bname = $obj -> title ;
$Blink = $obj -> id ;
$Bphoto = " " ;
2018-08-29 15:00:01 +02:00
2020-04-04 22:00:40 +02:00
foreach ( $obj -> link as $l ) {
$atts = $l -> attributes ();
switch ( $atts [ 'rel' ]) {
case " alternate " : $Blink = $atts [ 'href' ];
case " photo " : $Bphoto = $atts [ 'href' ];
}
2012-09-10 10:36:30 +02:00
}
2020-04-04 22:00:40 +02:00
$A = '[url=' . Contact :: magicLink ( $Alink ) . ']' . $Aname . '[/url]' ;
$B = '[url=' . Contact :: magicLink ( $Blink ) . ']' . $Bname . '[/url]' ;
if ( $Bphoto != " " ) {
$Bphoto = '[url=' . Contact :: magicLink ( $Blink ) . '][img=80x80]' . $Bphoto . '[/img][/url]' ;
}
2015-10-04 21:17:28 +02:00
2020-04-04 22:00:40 +02:00
/*
* we can ' t have a translation string with three positions but no distinguishable text
* So here is the translate string .
*/
$txt = DI :: l10n () -> t ( '%1$s poked %2$s' );
2012-09-10 10:36:30 +02:00
2020-04-04 22:00:40 +02:00
// now translate the verb
2020-04-05 09:11:47 +02:00
$poked_t = trim ( sprintf ( $txt , '' , '' ));
2020-04-04 22:00:40 +02:00
$txt = str_replace ( $poked_t , DI :: l10n () -> t ( $verb ), $txt );
2012-09-10 10:36:30 +02:00
2020-04-04 22:00:40 +02:00
// then do the sprintf on the translation string
2012-09-10 10:36:30 +02:00
2020-04-05 09:11:47 +02:00
$item [ 'body' ] = sprintf ( $txt , $A , $B ) . " \n \n \n " . $Bphoto ;
2012-09-10 10:36:30 +02:00
2017-04-08 20:00:54 +02:00
}
2020-04-05 09:08:20 +02:00
if ( $activity -> match ( $item [ 'verb' ], Activity :: TAG )) {
2020-04-04 22:00:40 +02:00
$fields = [ 'author-id' , 'author-link' , 'author-name' , 'author-network' ,
'verb' , 'object-type' , 'resource-id' , 'body' , 'plink' ];
$obj = Item :: selectFirst ( $fields , [ 'uri' => $item [ 'parent-uri' ]]);
if ( ! DBA :: isResult ( $obj )) {
return ;
}
2012-09-10 10:36:30 +02:00
2020-04-04 22:00:40 +02:00
$author_arr = [ 'uid' => 0 , 'id' => $item [ 'author-id' ],
'network' => $item [ 'author-network' ], 'url' => $item [ 'author-link' ]];
$author = '[url=' . Contact :: magicLinkByContact ( $author_arr ) . ']' . $item [ 'author-name' ] . '[/url]' ;
$author_arr = [ 'uid' => 0 , 'id' => $obj [ 'author-id' ],
'network' => $obj [ 'author-network' ], 'url' => $obj [ 'author-link' ]];
$objauthor = '[url=' . Contact :: magicLinkByContact ( $author_arr ) . ']' . $obj [ 'author-name' ] . '[/url]' ;
switch ( $obj [ 'verb' ]) {
case Activity :: POST :
switch ( $obj [ 'object-type' ]) {
case Activity\ObjectType :: EVENT :
$post_type = DI :: l10n () -> t ( 'event' );
break ;
default :
$post_type = DI :: l10n () -> t ( 'status' );
}
break ;
default :
if ( $obj [ 'resource-id' ]) {
$post_type = DI :: l10n () -> t ( 'photo' );
$m = []; preg_match ( " / \ [url=([^]]*) \ ]/ " , $obj [ 'body' ], $m );
$rr [ 'plink' ] = $m [ 1 ];
} else {
$post_type = DI :: l10n () -> t ( 'status' );
}
// Let's break everthing ... ;-)
break ;
}
$plink = '[url=' . $obj [ 'plink' ] . ']' . $post_type . '[/url]' ;
2012-09-10 10:36:30 +02:00
2020-04-05 19:25:36 +02:00
$parsedobj = XML :: parseString ( $xmlhead . $item [ 'object' ]);
2012-09-10 10:36:30 +02:00
2020-04-04 22:00:40 +02:00
$tag = sprintf ( '#[url=%s]%s[/url]' , $parsedobj -> id , $parsedobj -> content );
$item [ 'body' ] = DI :: l10n () -> t ( '%1$s tagged %2$s\'s %3$s with %4$s' , $author , $objauthor , $plink , $tag );
2012-09-10 10:36:30 +02:00
}
}
2020-04-04 22:00:40 +02:00
2012-09-10 10:36:30 +02:00
$matches = null ;
2017-04-08 20:00:54 +02:00
if ( preg_match_all ( '/@\[url=(.*?)\]/is' , $item [ 'body' ], $matches , PREG_SET_ORDER )) {
2017-04-04 19:46:56 +02:00
foreach ( $matches as $mtch ) {
2018-02-27 10:29:11 +01:00
if ( ! strpos ( $mtch [ 1 ], 'zrl=' )) {
2018-06-02 10:05:06 +02:00
$item [ 'body' ] = str_replace ( $mtch [ 0 ], '@[url=' . Contact :: magicLink ( $mtch [ 1 ]) . ']' , $item [ 'body' ]);
2017-04-08 20:00:54 +02:00
}
2012-09-10 10:36:30 +02:00
}
}
// add zrl's to public images
2016-01-15 23:32:13 +01:00
$photo_pattern = " / \ [url=(.*?) \ /photos \ /(.*?) \ /image \ /(.*?) \ ] \ [img(.*?) \ ]h(.*?) \ [ \ /img \ ] \ [ \ /url \ ]/is " ;
2017-04-08 20:00:54 +02:00
if ( preg_match ( $photo_pattern , $item [ 'body' ])) {
2018-06-02 10:28:01 +02:00
$photo_replace = '[url=' . Profile :: zrl ( '$1' . '/photos/' . '$2' . '/image/' . '$3' , true ) . '][img' . '$4' . ']h' . '$5' . '[/img][/url]' ;
2018-02-05 01:23:49 +01:00
$item [ 'body' ] = BBCode :: pregReplaceInTag ( $photo_pattern , $photo_replace , 'url' , $item [ 'body' ]);
2016-01-15 23:32:13 +01:00
}
2012-09-10 10:36:30 +02:00
// add sparkle links to appropriate permalinks
2018-07-02 07:41:55 +02:00
$author = [ 'uid' => 0 , 'id' => $item [ 'author-id' ],
'network' => $item [ 'author-network' ], 'url' => $item [ 'author-link' ]];
2018-07-10 14:27:56 +02:00
2018-10-21 22:23:08 +02:00
// Only create a redirection to a magic link when logged in
2019-09-28 20:09:11 +02:00
if ( ! empty ( $item [ 'plink' ]) && Session :: isAuthenticated ()) {
2019-02-22 03:29:22 +01:00
$item [ 'plink' ] = Contact :: magicLinkByContact ( $author , $item [ 'plink' ]);
2018-07-10 14:27:56 +02:00
}
2011-04-18 17:37:02 +02:00
}
2012-08-07 09:53:53 +02:00
/**
* Count the total of comments on this item and its desendants
2017-04-08 20:00:54 +02:00
* @ TODO proper type - hint + doc - tag
2019-01-07 16:24:06 +01:00
* @ param $item
* @ return int
2012-08-07 09:53:53 +02:00
*/
function count_descendants ( $item ) {
2012-09-10 10:36:30 +02:00
$total = count ( $item [ 'children' ]);
2017-04-04 19:46:56 +02:00
if ( $total > 0 ) {
foreach ( $item [ 'children' ] as $child ) {
2018-02-27 10:29:11 +01:00
if ( ! visible_activity ( $child )) {
2012-09-10 10:36:30 +02:00
$total -- ;
2017-04-08 20:03:21 +02:00
}
2012-09-10 10:36:30 +02:00
$total += count_descendants ( $child );
}
}
return $total ;
2012-08-07 09:53:53 +02:00
}
2012-09-19 05:12:55 +02:00
function visible_activity ( $item ) {
2019-12-15 23:28:01 +01:00
$activity = DI :: activity ();
2019-10-23 02:05:11 +02:00
2019-11-18 07:39:56 +01:00
if ( empty ( $item [ 'verb' ]) || $activity -> isHidden ( $item [ 'verb' ])) {
2019-10-24 00:25:43 +02:00
return false ;
2015-06-01 01:23:04 +02:00
}
2012-09-19 05:12:55 +02:00
2018-05-14 22:25:39 +02:00
// @TODO below if() block can be rewritten to a single line: $isVisible = allConditionsHere;
2019-10-24 00:25:43 +02:00
if ( $activity -> match ( $item [ 'verb' ], Activity :: FOLLOW ) &&
2019-10-25 00:10:20 +02:00
$item [ 'object-type' ] === Activity\ObjectType :: NOTE &&
2019-10-24 00:25:43 +02:00
empty ( $item [ 'self' ]) &&
$item [ 'uid' ] == local_user ()) {
2018-05-13 16:58:40 +02:00
return false ;
2012-09-28 04:53:55 +02:00
}
2012-09-19 05:12:55 +02:00
return true ;
}
2018-08-10 06:27:52 +02:00
function conv_get_blocklist ()
{
if ( ! local_user ()) {
return [];
}
2020-07-19 03:21:15 +02:00
$str_blocked = str_replace ([ " \n " , " \r " ], " , " , DI :: pConfig () -> get ( local_user (), 'system' , 'blocked' ));
2018-08-10 06:27:52 +02:00
if ( empty ( $str_blocked )) {
return [];
}
$blocklist = [];
foreach ( explode ( ',' , $str_blocked ) as $entry ) {
2018-08-19 12:14:12 +02:00
// The 4th parameter guarantees that there always will be a public contact entry
2020-07-15 23:08:42 +02:00
$cid = Contact :: getIdForURL ( trim ( $entry ), 0 , false , [ 'url' => trim ( $entry )]);
2018-08-10 06:27:52 +02:00
if ( ! empty ( $cid )) {
$blocklist [] = $cid ;
}
}
return $blocklist ;
}
2011-04-13 02:58:16 +02:00
/**
* " Render " a conversation or list of items for HTML display .
* There are two major forms of display :
* - Sequential or unthreaded ( " New Item View " or search results )
* - conversation view
* The $mode parameter decides between the various renderings and also
2012-09-10 09:19:08 +02:00
* figures out how to determine page owner and other contextual items
2011-04-13 02:58:16 +02:00
* that are based on unique features of the calling module .
2019-01-07 16:24:06 +01:00
* @ param App $a
* @ param array $items
* @ param $mode
* @ param $update
* @ param bool $preview
* @ param string $order
* @ param int $uid
* @ return string
* @ throws ImagickException
* @ throws \Friendica\Network\HTTPException\InternalServerErrorException
2011-04-13 02:58:16 +02:00
*/
2020-02-14 05:40:00 +01:00
function conversation ( App $a , array $items , $mode , $update , $preview = false , $order = 'commented' , $uid = 0 )
2018-10-24 08:15:24 +02:00
{
2020-09-03 16:03:36 +02:00
$page = DI :: page ();
$page -> registerFooterScript ( Theme :: getPathForFile ( 'asset/typeahead.js/dist/typeahead.bundle.js' ));
$page -> registerFooterScript ( Theme :: getPathForFile ( 'js/friendica-tagsinput/friendica-tagsinput.js' ));
$page -> registerStylesheet ( Theme :: getPathForFile ( 'js/friendica-tagsinput/friendica-tagsinput.css' ));
$page -> registerStylesheet ( Theme :: getPathForFile ( 'js/friendica-tagsinput/friendica-tagsinput-typeahead.css' ));
2018-08-10 06:27:52 +02:00
$ssl_state = ( local_user () ? true : false );
2012-09-10 09:19:08 +02:00
2012-09-10 10:36:30 +02:00
$profile_owner = 0 ;
2012-10-09 17:41:33 +02:00
$live_update_div = '' ;
2012-09-10 09:19:08 +02:00
2018-08-10 06:27:52 +02:00
$blocklist = conv_get_blocklist ();
2013-04-29 06:02:53 +02:00
2012-09-10 10:36:30 +02:00
$previewing = (( $preview ) ? ' preview ' : '' );
2012-09-10 09:19:08 +02:00
2017-04-04 19:46:56 +02:00
if ( $mode === 'network' ) {
2018-05-26 20:07:27 +02:00
$items = conversation_add_children ( $items , false , $order , $uid );
2012-09-10 10:36:30 +02:00
$profile_owner = local_user ();
2017-04-04 19:46:56 +02:00
if ( ! $update ) {
2017-04-08 19:05:50 +02:00
/*
* The special div is needed for liveUpdate to kick in for this page .
* We only launch liveUpdate if you aren ' t filtering in some incompatible
* way and also you aren ' t writing a comment ( discovered in javascript ) .
*/
2012-10-09 17:41:33 +02:00
$live_update_div = '<div id="live-network"></div>' . " \r \n "
2015-10-08 00:25:55 +02:00
. " <script> var profile_uid = " . $_SESSION [ 'uid' ]
2019-12-16 01:33:13 +01:00
. " ; var netargs = ' " . substr ( DI :: args () -> getCommand (), 8 )
2012-10-09 17:41:33 +02:00
. '?f='
2020-05-17 17:01:27 +02:00
. ( ! empty ( $_GET [ 'contactid' ]) ? '&contactid=' . rawurlencode ( $_GET [ 'contactid' ]) : '' )
. ( ! empty ( $_GET [ 'search' ]) ? '&search=' . rawurlencode ( $_GET [ 'search' ]) : '' )
. ( ! empty ( $_GET [ 'star' ]) ? '&star=' . rawurlencode ( $_GET [ 'star' ]) : '' )
. ( ! empty ( $_GET [ 'order' ]) ? '&order=' . rawurlencode ( $_GET [ 'order' ]) : '' )
. ( ! empty ( $_GET [ 'bmark' ]) ? '&bmark=' . rawurlencode ( $_GET [ 'bmark' ]) : '' )
. ( ! empty ( $_GET [ 'liked' ]) ? '&liked=' . rawurlencode ( $_GET [ 'liked' ]) : '' )
. ( ! empty ( $_GET [ 'conv' ]) ? '&conv=' . rawurlencode ( $_GET [ 'conv' ]) : '' )
. ( ! empty ( $_GET [ 'nets' ]) ? '&nets=' . rawurlencode ( $_GET [ 'nets' ]) : '' )
. ( ! empty ( $_GET [ 'cmin' ]) ? '&cmin=' . rawurlencode ( $_GET [ 'cmin' ]) : '' )
. ( ! empty ( $_GET [ 'cmax' ]) ? '&cmax=' . rawurlencode ( $_GET [ 'cmax' ]) : '' )
. ( ! empty ( $_GET [ 'file' ]) ? '&file=' . rawurlencode ( $_GET [ 'file' ]) : '' )
2012-10-09 17:41:33 +02:00
2020-02-14 05:40:00 +01:00
. " '; </script> \r \n " ;
2012-10-09 17:41:33 +02:00
}
2017-04-08 19:05:50 +02:00
} elseif ( $mode === 'profile' ) {
2018-08-20 06:26:05 +02:00
$items = conversation_add_children ( $items , false , $order , $uid );
2019-11-03 02:19:42 +01:00
$profile_owner = $a -> profile [ 'uid' ];
2012-09-10 09:19:08 +02:00
2017-04-04 19:46:56 +02:00
if ( ! $update ) {
2018-01-01 21:27:33 +01:00
$tab = 'posts' ;
2018-11-30 15:06:22 +01:00
if ( ! empty ( $_GET [ 'tab' ])) {
2018-11-09 19:29:42 +01:00
$tab = Strings :: escapeTags ( trim ( $_GET [ 'tab' ]));
2018-01-01 21:27:33 +01:00
}
2017-04-04 19:46:56 +02:00
if ( $tab === 'posts' ) {
2017-04-08 19:05:50 +02:00
/*
* This is ugly , but we can ' t pass the profile_uid through the session to the ajax updater ,
* because browser prefetching might change it on us . We have to deliver it with the page .
*/
2012-10-09 17:41:33 +02:00
$live_update_div = '<div id="live-profile"></div>' . " \r \n "
2019-11-03 02:19:42 +01:00
. " <script> var profile_uid = " . $a -> profile [ 'uid' ]
2020-02-14 05:40:00 +01:00
. " ; var netargs = '?f='; </script> \r \n " ;
2012-10-09 17:41:33 +02:00
}
}
2017-04-08 19:05:50 +02:00
} elseif ( $mode === 'notes' ) {
2018-12-13 21:09:19 +01:00
$items = conversation_add_children ( $items , false , $order , local_user ());
2012-09-10 10:36:30 +02:00
$profile_owner = local_user ();
2018-07-31 04:06:22 +02:00
2017-04-04 19:46:56 +02:00
if ( ! $update ) {
2012-10-09 17:41:33 +02:00
$live_update_div = '<div id="live-notes"></div>' . " \r \n "
2015-10-08 00:25:55 +02:00
. " <script> var profile_uid = " . local_user ()
2020-02-14 05:40:00 +01:00
. " ; var netargs = '/?f='; </script> \r \n " ;
2012-10-09 17:41:33 +02:00
}
2017-04-08 19:05:50 +02:00
} elseif ( $mode === 'display' ) {
2018-08-20 22:32:55 +02:00
$items = conversation_add_children ( $items , false , $order , $uid );
2012-09-10 10:36:30 +02:00
$profile_owner = $a -> profile [ 'uid' ];
2018-07-31 04:06:22 +02:00
2017-04-04 19:46:56 +02:00
if ( ! $update ) {
2012-11-02 01:31:50 +01:00
$live_update_div = '<div id="live-display"></div>' . " \r \n "
2019-05-26 22:15:38 +02:00
. " <script> var profile_uid = " . Session :: get ( 'uid' , 0 ) . " ; "
2020-02-14 05:40:00 +01:00
. " </script> " ;
2012-11-02 01:31:50 +01:00
}
2017-04-08 19:05:50 +02:00
} elseif ( $mode === 'community' ) {
2018-05-26 22:03:30 +02:00
$items = conversation_add_children ( $items , true , $order , $uid );
2012-09-10 10:36:30 +02:00
$profile_owner = 0 ;
2018-07-31 04:06:22 +02:00
2018-01-04 16:46:56 +01:00
if ( ! $update ) {
$live_update_div = '<div id="live-community"></div>' . " \r \n "
2019-12-16 01:33:13 +01:00
. " <script> var profile_uid = -1; var netargs = ' " . substr ( DI :: args () -> getCommand (), 10 )
2020-09-26 20:01:10 +02:00
. '?f='
. ( ! empty ( $_GET [ 'no_sharer' ]) ? '&no_sharer=' . rawurlencode ( $_GET [ 'no_sharer' ]) : '' )
. " '; </script> \r \n " ;
2018-01-04 16:46:56 +01:00
}
2018-08-25 15:48:00 +02:00
} elseif ( $mode === 'contacts' ) {
2018-11-25 19:50:41 +01:00
$items = conversation_add_children ( $items , false , $order , $uid );
2018-08-25 15:48:00 +02:00
$profile_owner = 0 ;
if ( ! $update ) {
$live_update_div = '<div id="live-contacts"></div>' . " \r \n "
2019-12-16 01:33:13 +01:00
. " <script> var profile_uid = -1; var netargs = ' " . substr ( DI :: args () -> getCommand (), 9 )
2020-02-14 05:40:00 +01:00
. " /?f='; </script> \r \n " ;
2018-08-25 15:48:00 +02:00
}
2017-04-08 19:05:50 +02:00
} elseif ( $mode === 'search' ) {
2012-10-09 17:41:33 +02:00
$live_update_div = '<div id="live-search"></div>' . " \r \n " ;
2012-09-10 10:36:30 +02:00
}
2012-09-10 09:19:08 +02:00
2012-09-10 10:36:30 +02:00
$page_dropping = (( local_user () && local_user () == $profile_owner ) ? true : false );
2012-09-10 09:19:08 +02:00
2018-01-04 03:12:19 +01:00
if ( ! $update ) {
2019-12-16 01:30:34 +01:00
$_SESSION [ 'return_path' ] = DI :: args () -> getQueryString ();
2017-04-08 19:05:50 +02:00
}
2012-03-15 05:20:20 +01:00
2018-01-15 14:05:12 +01:00
$cb = [ 'items' => $items , 'mode' => $mode , 'update' => $update , 'preview' => $preview ];
2018-12-26 07:06:24 +01:00
Hook :: callAll ( 'conversation_start' , $cb );
2012-01-10 05:03:00 +01:00
2012-09-10 10:36:30 +02:00
$items = $cb [ 'items' ];
2011-04-11 08:01:38 +02:00
2018-01-15 14:05:12 +01:00
$conv_responses = [
2020-02-23 02:41:12 +01:00
'like' => [],
'dislike' => [],
'attendyes' => [],
'attendno' => [],
2020-02-26 21:53:46 +01:00
'attendmaybe' => [],
'announce' => [],
2018-01-15 14:05:12 +01:00
];
2011-05-20 10:15:02 +02:00
2020-02-20 04:20:26 +01:00
if ( DI :: pConfig () -> get ( local_user (), 'system' , 'hide_dislike' )) {
unset ( $conv_responses [ 'dislike' ]);
}
2012-09-10 10:36:30 +02:00
// array with html for each thread (parent+comments)
2018-01-15 14:05:12 +01:00
$threads = [];
2012-09-10 10:36:30 +02:00
$threadsid = - 1 ;
2011-07-05 06:52:21 +02:00
2018-10-31 15:44:06 +01:00
$page_template = Renderer :: getMarkupTemplate ( " conversation.tpl " );
2012-08-10 08:01:37 +02:00
2018-07-31 04:06:22 +02:00
if ( ! empty ( $items )) {
2018-08-25 15:48:00 +02:00
if ( in_array ( $mode , [ 'community' , 'contacts' ])) {
2018-02-01 00:22:41 +01:00
$writable = true ;
2017-12-20 11:16:25 +01:00
} else {
2019-07-01 20:00:55 +02:00
$writable = ( $items [ 0 ][ 'uid' ] == 0 ) && in_array ( $items [ 0 ][ 'network' ], Protocol :: FEDERATED );
2017-12-20 11:16:25 +01:00
}
2012-08-10 08:01:37 +02:00
2018-01-04 15:39:47 +01:00
if ( ! local_user ()) {
$writable = false ;
}
2018-02-01 00:22:41 +01:00
if ( in_array ( $mode , [ 'network-new' , 'search' , 'contact-posts' ])) {
2011-04-11 08:01:38 +02:00
2017-04-08 20:00:54 +02:00
/*
* " New Item View " on network page or search page results
* - just loop through the items and format them minimally for display
*/
2011-04-11 06:21:16 +02:00
2012-09-10 10:36:30 +02:00
$tpl = 'search_item.tpl' ;
2017-04-04 19:46:56 +02:00
foreach ( $items as $item ) {
2014-01-05 16:22:42 +01:00
2018-02-27 10:29:11 +01:00
if ( ! visible_activity ( $item )) {
continue ;
}
2018-08-10 06:27:52 +02:00
if ( in_array ( $item [ 'author-id' ], $blocklist )) {
continue ;
2013-04-29 06:02:53 +02:00
}
2013-09-15 10:40:58 +02:00
2012-09-10 10:36:30 +02:00
$threadsid ++ ;
// prevent private email from leaking.
2018-08-11 22:40:44 +02:00
if ( $item [ 'network' ] === Protocol :: MAIL && local_user () != $item [ 'uid' ]) {
2017-04-08 19:05:50 +02:00
continue ;
}
2012-09-10 10:36:30 +02:00
2018-06-24 12:48:29 +02:00
$profile_name = $item [ 'author-name' ];
if ( ! empty ( $item [ 'author-link' ]) && empty ( $item [ 'author-name' ])) {
2012-09-10 10:36:30 +02:00
$profile_name = $item [ 'author-link' ];
2017-04-08 19:05:50 +02:00
}
2012-09-10 10:36:30 +02:00
2020-05-01 08:01:22 +02:00
$tags = Tag :: populateFromItem ( $item );
2013-01-13 17:13:01 +01:00
2018-07-02 07:41:55 +02:00
$author = [ 'uid' => 0 , 'id' => $item [ 'author-id' ],
'network' => $item [ 'author-network' ], 'url' => $item [ 'author-link' ]];
2019-02-22 03:29:22 +01:00
$profile_link = Contact :: magicLinkByContact ( $author );
2017-04-08 19:05:50 +02:00
2020-09-14 01:50:39 +02:00
$sparkle = '' ;
2018-06-01 08:46:34 +02:00
if ( strpos ( $profile_link , 'redir/' ) === 0 ) {
2012-09-10 10:36:30 +02:00
$sparkle = ' sparkle' ;
2017-04-08 19:05:50 +02:00
}
2012-09-10 10:36:30 +02:00
2018-01-15 14:05:12 +01:00
$locate = [ 'location' => $item [ 'location' ], 'coord' => $item [ 'coord' ], 'html' => '' ];
2018-12-26 07:06:24 +01:00
Hook :: callAll ( 'render_location' , $locate );
2020-09-14 01:33:57 +02:00
$location_html = $locate [ 'html' ] ? : Strings :: escapeHtml ( $locate [ 'location' ] ? : $locate [ 'coord' ] ? : '' );
2012-09-10 10:36:30 +02:00
localize_item ( $item );
2017-04-08 19:05:50 +02:00
if ( $mode === 'network-new' ) {
2012-09-10 10:36:30 +02:00
$dropping = true ;
2017-04-08 19:05:50 +02:00
} else {
2012-09-10 10:36:30 +02:00
$dropping = false ;
2017-04-08 19:05:50 +02:00
}
2012-09-10 10:36:30 +02:00
2018-01-15 14:05:12 +01:00
$drop = [
2012-09-10 10:36:30 +02:00
'dropping' => $dropping ,
2012-09-26 04:10:46 +02:00
'pagedrop' => $page_dropping ,
2020-01-18 20:52:34 +01:00
'select' => DI :: l10n () -> t ( 'Select' ),
'delete' => DI :: l10n () -> t ( 'Delete' ),
2018-01-15 14:05:12 +01:00
];
2012-09-10 10:36:30 +02:00
2020-02-14 02:42:15 +01:00
$likebuttons = [
'like' => null ,
'dislike' => null ,
'share' => null ,
];
2012-09-10 10:36:30 +02:00
2020-02-20 04:20:26 +01:00
if ( DI :: pConfig () -> get ( local_user (), 'system' , 'hide_dislike' )) {
unset ( $likebuttons [ 'dislike' ]);
}
2020-09-14 01:37:43 +02:00
$body_html = Item :: prepareBody ( $item , true , $preview );
2012-09-24 04:22:48 +02:00
2019-12-15 23:28:01 +01:00
list ( $categories , $folders ) = DI :: contentItem () -> determineCategoriesTerms ( $item );
2012-09-26 04:10:46 +02:00
2020-01-18 16:50:57 +01:00
if ( ! empty ( $item [ 'content-warning' ]) && DI :: pConfig () -> get ( local_user (), 'system' , 'disable_cw' , false )) {
2019-02-23 05:03:32 +01:00
$title = ucfirst ( $item [ 'content-warning' ]);
2018-04-05 04:50:39 +02:00
} else {
2019-02-23 05:03:32 +01:00
$title = $item [ 'title' ];
2018-04-05 04:50:39 +02:00
}
2018-01-15 14:05:12 +01:00
$tmp_item = [
2012-09-10 10:36:30 +02:00
'template' => $tpl ,
2018-07-10 14:27:56 +02:00
'id' => ( $preview ? 'P0' : $item [ 'id' ]),
'guid' => ( $preview ? 'Q0' : $item [ 'guid' ]),
2018-06-18 22:36:34 +02:00
'network' => $item [ 'network' ],
2019-12-27 18:24:29 +01:00
'network_name' => ContactSelector :: networkToName ( $item [ 'author-network' ], $item [ 'author-link' ], $item [ 'network' ]),
2019-08-28 08:38:35 +02:00
'network_icon' => ContactSelector :: networkToIcon ( $item [ 'network' ], $item [ 'author-link' ]),
2020-01-18 20:52:34 +01:00
'linktitle' => DI :: l10n () -> t ( 'View %s\'s profile @ %s' , $profile_name , $item [ 'author-link' ]),
2012-09-10 10:36:30 +02:00
'profile_url' => $profile_link ,
2020-09-14 01:40:59 +02:00
'item_photo_menu_html' => item_photo_menu ( $item ),
2019-02-23 05:03:32 +01:00
'name' => $profile_name ,
2012-09-10 10:36:30 +02:00
'sparkle' => $sparkle ,
2020-09-14 01:50:39 +02:00
'lock' => false ,
2020-07-28 21:30:55 +02:00
'thumb' => DI :: baseUrl () -> remove ( $item [ 'author-avatar' ]),
2019-02-23 05:03:32 +01:00
'title' => $title ,
2020-09-14 01:37:43 +02:00
'body_html' => $body_html ,
2019-02-23 05:03:32 +01:00
'tags' => $tags [ 'tags' ],
'hashtags' => $tags [ 'hashtags' ],
'mentions' => $tags [ 'mentions' ],
'implicit_mentions' => $tags [ 'implicit_mentions' ],
2020-01-18 20:52:34 +01:00
'txt_cats' => DI :: l10n () -> t ( 'Categories:' ),
'txt_folders' => DI :: l10n () -> t ( 'Filed under:' ),
2012-09-24 02:16:37 +02:00
'has_cats' => (( count ( $categories )) ? 'true' : '' ),
'has_folders' => (( count ( $folders )) ? 'true' : '' ),
2012-09-19 17:38:32 +02:00
'categories' => $categories ,
'folders' => $folders ,
2020-09-14 01:37:43 +02:00
'text' => strip_tags ( $body_html ),
2018-01-27 03:38:34 +01:00
'localtime' => DateTimeFormat :: local ( $item [ 'created' ], 'r' ),
2020-01-18 20:52:34 +01:00
'ago' => (( $item [ 'app' ]) ? DI :: l10n () -> t ( '%s from %s' , Temporal :: getRelativeDate ( $item [ 'created' ]), $item [ 'app' ]) : Temporal :: getRelativeDate ( $item [ 'created' ])),
2020-09-14 01:33:57 +02:00
'location_html' => $location_html ,
2012-09-10 10:36:30 +02:00
'indent' => '' ,
2020-09-14 01:50:39 +02:00
'owner_name' => '' ,
'owner_url' => '' ,
2020-07-28 21:30:55 +02:00
'owner_photo' => DI :: baseUrl () -> remove ( $item [ 'owner-avatar' ]),
2018-11-07 13:19:39 +01:00
'plink' => Item :: getPlink ( $item ),
2012-09-10 10:36:30 +02:00
'edpost' => false ,
2020-09-14 01:50:39 +02:00
'isstarred' => 'unstarred' ,
'star' => false ,
2012-09-10 10:36:30 +02:00
'drop' => $drop ,
'vote' => $likebuttons ,
2020-09-14 01:44:11 +02:00
'like_html' => '' ,
'dislike_html' => '' ,
2020-09-14 01:43:42 +02:00
'comment_html' => '' ,
2020-01-18 20:52:34 +01:00
'conv' => (( $preview ) ? '' : [ 'href' => 'display/' . $item [ 'guid' ], 'title' => DI :: l10n () -> t ( 'View in context' )]),
2012-09-10 10:36:30 +02:00
'previewing' => $previewing ,
2020-01-18 20:52:34 +01:00
'wait' => DI :: l10n () -> t ( 'Please wait' ),
2012-09-10 10:36:30 +02:00
'thread_level' => 1 ,
2018-01-15 14:05:12 +01:00
];
2012-09-10 10:36:30 +02:00
2018-01-15 14:05:12 +01:00
$arr = [ 'item' => $item , 'output' => $tmp_item ];
2018-12-26 07:06:24 +01:00
Hook :: callAll ( 'display_item' , $arr );
2012-09-10 10:36:30 +02:00
2018-07-10 14:27:56 +02:00
$threads [ $threadsid ][ 'id' ] = $item [ 'id' ];
2018-06-18 22:36:34 +02:00
$threads [ $threadsid ][ 'network' ] = $item [ 'network' ];
2018-01-15 14:05:12 +01:00
$threads [ $threadsid ][ 'items' ] = [ $arr [ 'output' ]];
2012-09-10 10:36:30 +02:00
}
2017-04-08 19:05:50 +02:00
} else {
2012-09-10 10:36:30 +02:00
// Normal View
2018-10-31 15:44:06 +01:00
$page_template = Renderer :: getMarkupTemplate ( " threaded_conversation.tpl " );
2012-09-10 10:36:30 +02:00
2017-12-19 18:15:56 +01:00
$conv = new Thread ( $mode , $preview , $writable );
2012-09-10 10:36:30 +02:00
2017-04-08 19:05:50 +02:00
/*
* get all the topmost parents
* this shouldn ' t be needed , as we should have only them in our array
* But for now , this array respects the old style , just in case
*/
2017-04-04 19:46:56 +02:00
foreach ( $items as $item ) {
2018-08-10 06:27:52 +02:00
if ( in_array ( $item [ 'author-id' ], $blocklist )) {
continue ;
2013-04-29 06:02:53 +02:00
}
2013-09-15 10:40:58 +02:00
2012-09-10 10:36:30 +02:00
// Can we put this after the visibility check?
2015-06-03 20:57:30 +02:00
builtin_activity_puller ( $item , $conv_responses );
2012-09-10 10:36:30 +02:00
// Only add what is visible
2018-08-11 22:40:44 +02:00
if ( $item [ 'network' ] === Protocol :: MAIL && local_user () != $item [ 'uid' ]) {
2012-09-10 10:36:30 +02:00
continue ;
}
2017-04-08 19:05:50 +02:00
2018-02-27 10:29:11 +01:00
if ( ! visible_activity ( $item )) {
2012-09-10 10:36:30 +02:00
continue ;
}
2018-07-08 15:52:11 +02:00
/// @todo Check if this call is needed or not
2018-07-08 14:01:36 +02:00
$arr = [ 'item' => $item ];
2018-12-26 07:06:24 +01:00
Hook :: callAll ( 'display_item' , $arr );
2013-01-22 08:11:13 +01:00
2012-09-26 04:10:46 +02:00
$item [ 'pagedrop' ] = $page_dropping ;
2020-05-27 14:19:06 +02:00
if ( $item [ 'gravity' ] == GRAVITY_PARENT ) {
2017-12-08 05:33:36 +01:00
$item_object = new Post ( $item );
$conv -> addParent ( $item_object );
2012-09-10 10:36:30 +02:00
}
}
2017-11-19 20:15:25 +01:00
$threads = $conv -> getTemplateData ( $conv_responses );
2017-04-04 19:46:56 +02:00
if ( ! $threads ) {
2018-10-30 14:58:45 +01:00
Logger :: log ( '[ERROR] conversation : Failed to get template data.' , Logger :: DEBUG );
2018-01-15 14:05:12 +01:00
$threads = [];
2012-09-10 10:36:30 +02:00
}
}
}
2018-10-31 15:35:50 +01:00
$o = Renderer :: replaceMacros ( $page_template , [
2019-12-30 23:00:08 +01:00
'$baseurl' => DI :: baseUrl () -> get ( $ssl_state ),
2019-12-16 01:30:34 +01:00
'$return_path' => DI :: args () -> getQueryString (),
2012-10-09 17:41:33 +02:00
'$live_update' => $live_update_div ,
2020-01-18 20:52:34 +01:00
'$remove' => DI :: l10n () -> t ( 'remove' ),
2012-09-10 10:36:30 +02:00
'$mode' => $mode ,
'$user' => $a -> user ,
'$threads' => $threads ,
2020-01-18 20:52:34 +01:00
'$dropping' => ( $page_dropping ? DI :: l10n () -> t ( 'Delete Selected Items' ) : False ),
2018-01-15 14:05:12 +01:00
]);
2012-09-10 10:36:30 +02:00
return $o ;
2017-12-12 05:52:33 +01:00
}
2011-04-13 02:58:16 +02:00
2019-04-03 20:51:50 +02:00
/**
* Fetch all comments from a query . Additionally set the newest resharer as thread owner .
*
2020-01-26 20:14:36 +01:00
* @ param mixed $thread_items Database statement with thread posts
2019-11-08 07:52:44 +01:00
* @ param boolean $pinned Is the item pinned ?
*
2019-04-03 20:51:50 +02:00
* @ return array items with parents and comments
*/
2019-11-08 07:52:44 +01:00
function conversation_fetch_comments ( $thread_items , $pinned ) {
2019-04-03 20:51:50 +02:00
$comments = [];
while ( $row = Item :: fetch ( $thread_items )) {
2020-09-22 07:36:01 +02:00
$name = $row [ 'causer-contact-type' ] == Contact :: TYPE_RELAY ? $row [ 'causer-link' ] : $row [ 'causer-name' ];
2020-08-23 15:26:23 +02:00
2020-09-13 16:15:28 +02:00
switch ( $row [ 'post-type' ]) {
case Item :: PT_TO :
$row [ 'direction' ] = [ 'direction' => 7 , 'title' => DI :: l10n () -> t ( 'You had been addressed (%s).' , 'to' )];
break ;
case Item :: PT_CC :
$row [ 'direction' ] = [ 'direction' => 7 , 'title' => DI :: l10n () -> t ( 'You had been addressed (%s).' , 'cc' )];
break ;
case Item :: PT_BTO :
$row [ 'direction' ] = [ 'direction' => 7 , 'title' => DI :: l10n () -> t ( 'You had been addressed (%s).' , 'bto' )];
break ;
case Item :: PT_BCC :
$row [ 'direction' ] = [ 'direction' => 7 , 'title' => DI :: l10n () -> t ( 'You had been addressed (%s).' , 'bcc' )];
break ;
case Item :: PT_FOLLOWER :
2020-09-09 20:03:14 +02:00
$row [ 'direction' ] = [ 'direction' => 6 , 'title' => DI :: l10n () -> t ( 'You are following %s.' , $row [ 'author-name' ])];
2020-09-13 16:15:28 +02:00
break ;
case Item :: PT_TAG :
$row [ 'direction' ] = [ 'direction' => 4 , 'title' => DI :: l10n () -> t ( 'Tagged' )];
break ;
case Item :: PT_ANNOUNCEMENT :
2020-09-22 07:36:01 +02:00
if ( ! empty ( $row [ 'causer-id' ]) && DI :: pConfig () -> get ( local_user (), 'system' , 'display_resharer' ) ) {
$row [ 'owner-link' ] = $row [ 'causer-link' ];
$row [ 'owner-avatar' ] = $row [ 'causer-avatar' ];
$row [ 'owner-name' ] = $row [ 'causer-name' ];
}
if (( $row [ 'gravity' ] == GRAVITY_PARENT ) && ! empty ( $row [ 'causer-id' ])) {
$row [ 'reshared' ] = DI :: l10n () -> t ( '%s reshared this.' , '<a href="' . htmlentities ( Contact :: magicLinkbyId ( $row [ 'causer-id' ])) . '">' . htmlentities ( $name ) . '</a>' );
}
$row [ 'direction' ] = [ 'direction' => 3 , 'title' => ( empty ( $row [ 'causer-id' ]) ? DI :: l10n () -> t ( 'Reshared' ) : DI :: l10n () -> t ( 'Reshared by %s' , $name ))];
2020-09-13 16:15:28 +02:00
break ;
case Item :: PT_COMMENT :
$row [ 'direction' ] = [ 'direction' => 5 , 'title' => DI :: l10n () -> t ( '%s is participating in this thread.' , $row [ 'author-name' ])];
break ;
case Item :: PT_STORED :
$row [ 'direction' ] = [ 'direction' => 8 , 'title' => DI :: l10n () -> t ( 'Stored' )];
break ;
2020-09-14 19:48:57 +02:00
case Item :: PT_GLOBAL :
$row [ 'direction' ] = [ 'direction' => 9 , 'title' => DI :: l10n () -> t ( 'Global' )];
break ;
2020-09-21 14:31:20 +02:00
case Item :: PT_RELAY :
2020-09-22 07:36:01 +02:00
$row [ 'direction' ] = [ 'direction' => 10 , 'title' => ( empty ( $row [ 'causer-id' ]) ? DI :: l10n () -> t ( 'Relayed' ) : DI :: l10n () -> t ( 'Relayed by %s.' , $name ))];
2020-09-21 14:31:20 +02:00
break ;
2020-09-21 17:17:33 +02:00
case Item :: PT_FETCHED :
2020-09-22 07:36:01 +02:00
$row [ 'direction' ] = [ 'direction' => 2 , 'title' => ( empty ( $row [ 'causer-id' ]) ? DI :: l10n () -> t ( 'Fetched' ) : DI :: l10n () -> t ( 'Fetched because of %s' , $name ))];
2020-09-21 17:17:33 +02:00
break ;
}
2020-09-12 19:45:04 +02:00
2019-11-08 07:52:44 +01:00
if ( $row [ 'gravity' ] == GRAVITY_PARENT ) {
$row [ 'pinned' ] = $pinned ;
}
2019-04-03 20:51:50 +02:00
$comments [] = $row ;
}
DBA :: close ( $thread_items );
return $comments ;
}
2018-01-04 11:02:56 +01:00
/**
2020-01-19 07:05:23 +01:00
* Add comments to top level entries that had been fetched before
2018-01-04 11:02:56 +01:00
*
* The system will fetch the comments for the local user whenever possible .
* This behaviour is currently needed to allow commenting on Friendica posts .
*
* @ param array $parents Parent items
*
2019-01-07 16:24:06 +01:00
* @ param $block_authors
* @ param $order
* @ param $uid
2018-01-04 11:02:56 +01:00
* @ return array items with parents and comments
2019-01-07 16:24:06 +01:00
* @ throws \Friendica\Network\HTTPException\InternalServerErrorException
2018-01-04 11:02:56 +01:00
*/
2018-07-13 21:47:14 +02:00
function conversation_add_children ( array $parents , $block_authors , $order , $uid ) {
2020-06-16 08:49:53 +02:00
if ( count ( $parents ) > 1 ) {
$max_comments = DI :: config () -> get ( 'system' , 'max_comments' , 100 );
} else {
$max_comments = DI :: config () -> get ( 'system' , 'max_display_comments' , 1000 );
}
2018-02-26 23:15:57 +01:00
2020-08-10 10:26:09 +02:00
$params = [ 'order' => [ 'gravity' , 'uid' , 'commented' => true ]];
2018-06-10 09:26:37 +02:00
2018-02-26 23:15:57 +01:00
if ( $max_comments > 0 ) {
2018-06-10 09:26:37 +02:00
$params [ 'limit' ] = $max_comments ;
2018-02-26 23:15:57 +01:00
}
2018-01-04 11:02:56 +01:00
2018-01-15 14:05:12 +01:00
$items = [];
2018-01-04 11:02:56 +01:00
2020-05-07 20:39:39 +02:00
foreach ( $parents AS $parent ) {
2020-05-31 16:38:00 +02:00
$condition = [ " `item`.`parent-uri` = ? AND `item`.`uid` IN (0, ?) AND (`vid` != ? OR `vid` IS NULL) " ,
2020-05-26 07:18:50 +02:00
$parent [ 'uri' ], $uid , Verb :: getID ( Activity :: FOLLOW )];
2020-05-07 20:39:39 +02:00
$items = conversation_fetch_items ( $parent , $items , $condition , $block_authors , $params );
2018-01-04 11:02:56 +01:00
}
foreach ( $items as $index => $item ) {
if ( $item [ 'uid' ] == 0 ) {
2019-07-01 20:00:55 +02:00
$items [ $index ][ 'writable' ] = in_array ( $item [ 'network' ], Protocol :: FEDERATED );
2018-01-04 11:02:56 +01:00
}
}
2018-02-26 07:38:27 +01:00
$items = conv_sort ( $items , $order );
2018-01-04 11:02:56 +01:00
return $items ;
}
2020-05-07 20:39:39 +02:00
/**
* Fetch conversation items
*
* @ param array $parent
* @ param array $items
* @ param array $condition
* @ param boolean $block_authors
* @ param array $params
* @ return array
*/
function conversation_fetch_items ( array $parent , array $items , array $condition , bool $block_authors , array $params ) {
if ( $block_authors ) {
2020-05-28 08:08:03 +02:00
$condition [ 0 ] .= " AND NOT `author`.`hidden` " ;
2020-05-07 20:39:39 +02:00
}
2020-08-23 19:48:44 +02:00
$thread_items = Item :: selectForUser ( local_user (), array_merge ( Item :: DISPLAY_FIELDLIST , [ 'contact-uid' , 'gravity' , 'post-type' ]), $condition , $params );
2020-05-07 20:39:39 +02:00
$comments = conversation_fetch_comments ( $thread_items , $parent [ 'pinned' ] ? ? false );
if ( count ( $comments ) != 0 ) {
$items = array_merge ( $items , $comments );
}
return $items ;
}
2017-04-08 19:05:50 +02:00
function item_photo_menu ( $item ) {
2016-10-23 04:49:12 +02:00
$sub_link = '' ;
$poke_link = '' ;
$contact_url = '' ;
$pm_url = '' ;
$status_link = '' ;
$photos_link = '' ;
$posts_link = '' ;
2019-03-11 21:42:32 +01:00
$block_link = '' ;
$ignore_link = '' ;
2012-09-10 10:36:30 +02:00
2020-05-28 18:02:36 +02:00
if ( local_user () && local_user () == $item [ 'uid' ] && $item [ 'gravity' ] == GRAVITY_PARENT && ! $item [ 'self' ]) {
2012-09-28 04:53:55 +02:00
$sub_link = 'javascript:dosubthread(' . $item [ 'id' ] . '); return false;' ;
}
2018-07-02 07:41:55 +02:00
$author = [ 'uid' => 0 , 'id' => $item [ 'author-id' ],
'network' => $item [ 'author-network' ], 'url' => $item [ 'author-link' ]];
2019-02-26 10:19:08 +01:00
$profile_link = Contact :: magicLinkByContact ( $author , $item [ 'author-link' ]);
2018-06-01 08:46:34 +02:00
$sparkle = ( strpos ( $profile_link , 'redir/' ) === 0 );
2012-09-10 10:36:30 +02:00
2016-06-05 20:01:38 +02:00
$cid = 0 ;
2020-07-15 23:08:42 +02:00
$pcid = Contact :: getIdForURL ( $item [ 'author-link' ], 0 , false );
2016-10-23 04:49:12 +02:00
$network = '' ;
2016-06-05 20:01:38 +02:00
$rel = 0 ;
2018-11-08 17:28:29 +01:00
$condition = [ 'uid' => local_user (), 'nurl' => Strings :: normaliseLink ( $item [ 'author-link' ])];
2018-07-20 14:19:26 +02:00
$contact = DBA :: selectFirst ( 'contact' , [ 'id' , 'network' , 'rel' ], $condition );
2018-07-21 14:46:04 +02:00
if ( DBA :: isResult ( $contact )) {
2018-01-11 09:26:30 +01:00
$cid = $contact [ 'id' ];
$network = $contact [ 'network' ];
$rel = $contact [ 'rel' ];
2016-06-05 20:01:38 +02:00
}
2017-04-04 19:46:56 +02:00
if ( $sparkle ) {
2020-01-28 01:21:18 +01:00
$status_link = $profile_link . '/status' ;
2019-02-26 10:19:08 +01:00
$photos_link = str_replace ( '/profile/' , '/photos/' , $profile_link );
2020-01-28 01:21:18 +01:00
$profile_link = $profile_link . '/profile' ;
2016-10-23 04:49:12 +02:00
}
2012-09-10 10:36:30 +02:00
2019-03-11 21:42:32 +01:00
if ( ! empty ( $pcid )) {
$contact_url = 'contact/' . $pcid ;
2020-04-20 17:42:27 +02:00
$posts_link = $contact_url . '/posts' ;
$block_link = $contact_url . '/block' ;
$ignore_link = $contact_url . '/ignore' ;
2019-03-11 21:42:32 +01:00
}
2016-10-23 04:49:12 +02:00
if ( $cid && ! $item [ 'self' ]) {
2018-10-17 13:49:27 +02:00
$contact_url = 'contact/' . $cid ;
2020-04-20 17:42:27 +02:00
$poke_link = $contact_url . '/poke' ;
$posts_link = $contact_url . '/posts' ;
2012-09-10 10:36:30 +02:00
2019-07-01 20:00:55 +02:00
if ( in_array ( $network , [ Protocol :: ACTIVITYPUB , Protocol :: DFRN , Protocol :: DIASPORA ])) {
2016-10-23 04:49:12 +02:00
$pm_url = 'message/new/' . $cid ;
}
2012-09-10 10:36:30 +02:00
}
2015-10-10 16:23:20 +02:00
if ( local_user ()) {
2018-01-15 14:05:12 +01:00
$menu = [
2020-01-18 20:52:34 +01:00
DI :: l10n () -> t ( 'Follow Thread' ) => $sub_link ,
DI :: l10n () -> t ( 'View Status' ) => $status_link ,
DI :: l10n () -> t ( 'View Profile' ) => $profile_link ,
DI :: l10n () -> t ( 'View Photos' ) => $photos_link ,
DI :: l10n () -> t ( 'Network Posts' ) => $posts_link ,
DI :: l10n () -> t ( 'View Contact' ) => $contact_url ,
DI :: l10n () -> t ( 'Send PM' ) => $pm_url ,
DI :: l10n () -> t ( 'Block' ) => $block_link ,
DI :: l10n () -> t ( 'Ignore' ) => $ignore_link
2018-01-15 14:05:12 +01:00
];
2015-10-10 16:23:20 +02:00
2018-08-11 22:40:44 +02:00
if ( $network == Protocol :: DFRN ) {
2020-01-18 20:52:34 +01:00
$menu [ DI :: l10n () -> t ( " Poke " )] = $poke_link ;
2016-10-23 04:49:12 +02:00
}
2015-10-10 16:23:20 +02:00
2018-07-25 04:53:46 +02:00
if ((( $cid == 0 ) || ( $rel == Contact :: FOLLOWER )) &&
2019-07-01 20:00:55 +02:00
in_array ( $item [ 'network' ], Protocol :: FEDERATED )) {
2020-01-18 20:52:34 +01:00
$menu [ DI :: l10n () -> t ( 'Connect/Follow' )] = 'follow?url=' . urlencode ( $item [ 'author-link' ]);
2016-10-23 04:49:12 +02:00
}
} else {
2020-01-18 20:52:34 +01:00
$menu = [ DI :: l10n () -> t ( 'View Profile' ) => $item [ 'author-link' ]];
2016-10-23 04:49:12 +02:00
}
2012-09-10 10:36:30 +02:00
2018-01-15 14:05:12 +01:00
$args = [ 'item' => $item , 'menu' => $menu ];
2012-09-10 10:36:30 +02:00
2018-12-26 07:06:24 +01:00
Hook :: callAll ( 'item_photo_menu' , $args );
2012-09-10 10:36:30 +02:00
$menu = $args [ 'menu' ];
2016-10-23 04:49:12 +02:00
$o = '' ;
foreach ( $menu as $k => $v ) {
if ( strpos ( $v , 'javascript:' ) === 0 ) {
$v = substr ( $v , 11 );
$o .= '<li role="menuitem"><a onclick="' . $v . '">' . $k . '</a></li>' . PHP_EOL ;
} elseif ( $v != '' ) {
$o .= '<li role="menuitem"><a href="' . $v . '">' . $k . '</a></li>' . PHP_EOL ;
2012-09-28 04:53:55 +02:00
}
2012-09-10 10:36:30 +02:00
}
return $o ;
2017-08-09 23:12:41 +02:00
}
2011-04-13 02:58:16 +02:00
2015-06-03 20:57:30 +02:00
/**
2020-01-19 07:05:23 +01:00
* Checks item to see if it is one of the builtin activities ( like / dislike , event attendance , consensus items , etc . )
*
2015-06-03 20:57:30 +02:00
* Increments the count of each matching activity and adds a link to the author as needed .
*
2019-01-07 16:24:06 +01:00
* @ param array $item
2015-06-03 20:57:30 +02:00
* @ param array & $conv_responses ( already created with builtin activity structure )
* @ return void
2019-01-07 16:24:06 +01:00
* @ throws ImagickException
* @ throws \Friendica\Network\HTTPException\InternalServerErrorException
2015-06-03 20:57:30 +02:00
*/
function builtin_activity_puller ( $item , & $conv_responses ) {
2017-04-04 19:46:56 +02:00
foreach ( $conv_responses as $mode => $v ) {
2015-06-03 20:57:30 +02:00
$sparkle = '' ;
2011-04-13 02:58:16 +02:00
2017-04-08 20:00:54 +02:00
switch ( $mode ) {
2015-06-03 20:57:30 +02:00
case 'like' :
2019-10-24 00:25:43 +02:00
$verb = Activity :: LIKE ;
2015-06-03 20:57:30 +02:00
break ;
case 'dislike' :
2019-10-24 00:25:43 +02:00
$verb = Activity :: DISLIKE ;
2015-06-03 20:57:30 +02:00
break ;
case 'attendyes' :
2019-10-24 00:25:43 +02:00
$verb = Activity :: ATTEND ;
2015-06-03 20:57:30 +02:00
break ;
case 'attendno' :
2019-10-24 00:25:43 +02:00
$verb = Activity :: ATTENDNO ;
2015-06-03 20:57:30 +02:00
break ;
case 'attendmaybe' :
2019-10-24 00:25:43 +02:00
$verb = Activity :: ATTENDMAYBE ;
2015-06-03 20:57:30 +02:00
break ;
2019-04-02 07:38:42 +02:00
case 'announce' :
2019-10-24 00:25:43 +02:00
$verb = Activity :: ANNOUNCE ;
2019-04-02 07:38:42 +02:00
break ;
2015-06-03 20:57:30 +02:00
default :
return ;
}
2012-09-10 10:36:30 +02:00
2020-05-27 14:19:06 +02:00
if ( ! empty ( $item [ 'verb' ]) && DI :: activity () -> match ( $item [ 'verb' ], $verb ) && ( $item [ 'gravity' ] != GRAVITY_PARENT )) {
2018-07-02 07:41:55 +02:00
$author = [ 'uid' => 0 , 'id' => $item [ 'author-id' ],
'network' => $item [ 'author-network' ], 'url' => $item [ 'author-link' ]];
2019-02-22 03:29:22 +01:00
$url = Contact :: magicLinkByContact ( $author );
2018-06-01 08:46:34 +02:00
if ( strpos ( $url , 'redir/' ) === 0 ) {
2015-06-03 20:57:30 +02:00
$sparkle = ' class="sparkle" ' ;
2017-04-08 19:05:50 +02:00
}
2015-11-18 23:57:53 +01:00
2015-10-11 20:00:54 +02:00
$url = '<a href="' . $url . '"' . $sparkle . '>' . htmlentities ( $item [ 'author-name' ]) . '</a>' ;
2015-06-01 13:57:45 +02:00
2018-11-30 15:06:22 +01:00
if ( empty ( $item [ 'thr-parent' ])) {
2015-06-03 20:57:30 +02:00
$item [ 'thr-parent' ] = $item [ 'parent-uri' ];
2017-04-08 19:05:50 +02:00
}
2012-09-10 10:36:30 +02:00
2018-02-27 10:29:11 +01:00
if ( ! ( isset ( $conv_responses [ $mode ][ $item [ 'thr-parent' ] . '-l' ])
&& is_array ( $conv_responses [ $mode ][ $item [ 'thr-parent' ] . '-l' ]))) {
2018-01-15 14:05:12 +01:00
$conv_responses [ $mode ][ $item [ 'thr-parent' ] . '-l' ] = [];
2017-04-08 19:05:50 +02:00
}
2012-09-10 10:36:30 +02:00
2015-06-03 20:57:30 +02:00
// only list each unique author once
2017-04-08 19:05:50 +02:00
if ( in_array ( $url , $conv_responses [ $mode ][ $item [ 'thr-parent' ] . '-l' ])) {
2015-06-03 20:57:30 +02:00
continue ;
2017-04-08 19:05:50 +02:00
}
2015-06-03 20:57:30 +02:00
2020-09-28 13:36:47 +02:00
// Skip when the causer of the parent is the same than the author of the announce
2020-09-28 16:13:14 +02:00
if (( $verb == Activity :: ANNOUNCE ) && Item :: exists ([ 'uri' => $item [ 'thr-parent' ],
'uid' => $item [ 'uid' ], 'causer-id' => $item [ 'author-id' ], 'gravity' => GRAVITY_PARENT ])) {
continue ;
2020-09-28 13:36:47 +02:00
}
2018-02-27 10:29:11 +01:00
if ( ! isset ( $conv_responses [ $mode ][ $item [ 'thr-parent' ]])) {
2015-06-03 20:57:30 +02:00
$conv_responses [ $mode ][ $item [ 'thr-parent' ]] = 1 ;
2017-04-08 19:05:50 +02:00
} else {
2015-06-03 20:57:30 +02:00
$conv_responses [ $mode ][ $item [ 'thr-parent' ]] ++ ;
2017-04-08 19:05:50 +02:00
}
2015-06-01 01:23:04 +02:00
2017-03-06 11:07:17 +01:00
if ( public_contact () == $item [ 'author-id' ]) {
2016-09-24 17:50:23 +02:00
$conv_responses [ $mode ][ $item [ 'thr-parent' ] . '-self' ] = 1 ;
2017-03-06 11:07:17 +01:00
}
2016-09-24 17:50:23 +02:00
2015-06-03 20:57:30 +02:00
$conv_responses [ $mode ][ $item [ 'thr-parent' ] . '-l' ][] = $url ;
// there can only be one activity verb per item so if we found anything, we can stop looking
return ;
}
2012-09-10 10:36:30 +02:00
}
2017-12-12 05:52:33 +01:00
}
2011-04-13 02:58:16 +02:00
2017-04-08 19:05:50 +02:00
/**
* Format the vote text for a profile item
2019-01-07 16:24:06 +01:00
*
* @ param int $cnt = number of people who vote the item
* @ param array $arr = array of pre - linked names of likers / dislikers
2017-04-08 19:05:50 +02:00
* @ param string $type = one of 'like, ' dislike ', ' attendyes ', ' attendno ', ' attendmaybe '
2019-01-07 16:24:06 +01:00
* @ param int $id = item id
2018-02-03 18:25:58 +01:00
* @ return string formatted text
2019-01-07 16:24:06 +01:00
* @ throws \Friendica\Network\HTTPException\InternalServerErrorException
2017-04-08 19:05:50 +02:00
*/
function format_like ( $cnt , array $arr , $type , $id ) {
2012-09-10 10:36:30 +02:00
$o = '' ;
2015-06-03 20:57:30 +02:00
$expanded = '' ;
2018-12-14 04:30:43 +01:00
$phrase = '' ;
2015-06-03 20:57:30 +02:00
2017-04-04 19:46:56 +02:00
if ( $cnt == 1 ) {
2015-06-03 20:57:30 +02:00
$likers = $arr [ 0 ];
2012-12-22 20:57:29 +01:00
2015-10-14 19:48:57 +02:00
// Phrase if there is only one liker. In other cases it will be uses for the expanded
// list which show all likers
2017-04-08 20:00:54 +02:00
switch ( $type ) {
2015-10-14 19:48:57 +02:00
case 'like' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '%s likes this.' , $likers );
2015-10-14 19:48:57 +02:00
break ;
case 'dislike' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '%s doesn\'t like this.' , $likers );
2015-10-14 19:48:57 +02:00
break ;
case 'attendyes' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '%s attends.' , $likers );
2015-10-14 19:48:57 +02:00
break ;
case 'attendno' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '%s doesn\'t attend.' , $likers );
2015-10-14 19:48:57 +02:00
break ;
case 'attendmaybe' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '%s attends maybe.' , $likers );
2015-10-14 19:48:57 +02:00
break ;
2019-04-02 07:38:42 +02:00
case 'announce' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '%s reshared this.' , $likers );
2019-04-02 07:38:42 +02:00
break ;
2015-10-14 19:48:57 +02:00
}
}
2017-04-04 19:46:56 +02:00
if ( $cnt > 1 ) {
2012-09-10 10:36:30 +02:00
$total = count ( $arr );
2017-04-04 19:46:56 +02:00
if ( $total < MAX_LIKERS ) {
2020-01-18 20:52:34 +01:00
$last = DI :: l10n () -> t ( 'and' ) . ' ' . $arr [ count ( $arr ) - 1 ];
2012-11-06 16:43:19 +01:00
$arr2 = array_slice ( $arr , 0 , - 1 );
2019-01-07 18:51:48 +01:00
$likers = implode ( ', ' , $arr2 ) . ' ' . $last ;
} else {
$arr = array_slice ( $arr , 0 , MAX_LIKERS - 1 );
$likers = implode ( ', ' , $arr );
2020-01-18 20:52:34 +01:00
$likers .= DI :: l10n () -> t ( 'and %d other people' , $total - MAX_LIKERS );
2012-11-06 16:43:19 +01:00
}
2015-06-03 20:57:30 +02:00
$spanatts = " class= \" fakelink \" onclick= \" openClose(' { $type } list- $id '); \" " ;
2015-11-18 23:57:53 +01:00
2019-01-07 18:51:48 +01:00
$explikers = '' ;
2017-04-08 19:05:50 +02:00
switch ( $type ) {
2015-06-03 20:57:30 +02:00
case 'like' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '<span %1$s>%2$d people</span> like this' , $spanatts , $cnt );
$explikers = DI :: l10n () -> t ( '%s like this.' , $likers );
2015-06-03 20:57:30 +02:00
break ;
case 'dislike' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '<span %1$s>%2$d people</span> don\'t like this' , $spanatts , $cnt );
$explikers = DI :: l10n () -> t ( '%s don\'t like this.' , $likers );
2015-06-03 20:57:30 +02:00
break ;
case 'attendyes' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '<span %1$s>%2$d people</span> attend' , $spanatts , $cnt );
$explikers = DI :: l10n () -> t ( '%s attend.' , $likers );
2015-06-03 20:57:30 +02:00
break ;
case 'attendno' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '<span %1$s>%2$d people</span> don\'t attend' , $spanatts , $cnt );
$explikers = DI :: l10n () -> t ( '%s don\'t attend.' , $likers );
2015-06-03 20:57:30 +02:00
break ;
case 'attendmaybe' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '<span %1$s>%2$d people</span> attend maybe' , $spanatts , $cnt );
$explikers = DI :: l10n () -> t ( '%s attend maybe.' , $likers );
2015-06-03 20:57:30 +02:00
break ;
2019-04-02 07:38:42 +02:00
case 'announce' :
2020-01-18 20:52:34 +01:00
$phrase = DI :: l10n () -> t ( '<span %1$s>%2$d people</span> reshared this' , $spanatts , $cnt );
$explikers = DI :: l10n () -> t ( '%s reshared this.' , $likers );
2019-04-02 07:38:42 +02:00
break ;
2015-06-03 20:57:30 +02:00
}
2015-10-14 19:48:57 +02:00
2019-02-15 04:17:08 +01:00
$expanded .= " \t " . '<p class="wall-item-' . $type . '-expanded" id="' . $type . 'list-' . $id . '" style="display: none;" >' . $explikers . EOL . '</p>' ;
2012-09-10 10:36:30 +02:00
}
2015-06-03 20:57:30 +02:00
2018-10-31 15:44:06 +01:00
$o .= Renderer :: replaceMacros ( Renderer :: getMarkupTemplate ( 'voting_fakelink.tpl' ), [
2015-06-03 20:57:30 +02:00
'$phrase' => $phrase ,
'$type' => $type ,
'$id' => $id
2018-01-15 14:05:12 +01:00
]);
2015-06-03 20:57:30 +02:00
$o .= $expanded ;
2012-09-10 10:36:30 +02:00
return $o ;
2017-12-12 05:52:33 +01:00
}
2011-04-20 14:48:12 +02:00
2018-01-01 21:27:33 +01:00
function status_editor ( App $a , $x , $notes_cid = 0 , $popup = false )
{
2012-09-10 10:36:30 +02:00
$o = '' ;
2018-11-30 15:06:22 +01:00
$geotag = ! empty ( $x [ 'allow_location' ]) ? Renderer :: replaceMacros ( Renderer :: getMarkupTemplate ( 'jot_geotag.tpl' ), []) : '' ;
2012-09-10 10:36:30 +02:00
2018-10-31 15:44:06 +01:00
$tpl = Renderer :: getMarkupTemplate ( 'jot-header.tpl' );
2019-12-30 20:02:09 +01:00
DI :: page ()[ 'htmlhead' ] .= Renderer :: replaceMacros ( $tpl , [
2018-01-01 21:27:33 +01:00
'$newpost' => 'true' ,
2019-12-30 23:00:08 +01:00
'$baseurl' => DI :: baseUrl () -> get ( true ),
2018-01-01 21:27:33 +01:00
'$geotag' => $geotag ,
'$nickname' => $x [ 'nickname' ],
2020-01-18 20:52:34 +01:00
'$ispublic' => DI :: l10n () -> t ( 'Visible to <strong>everybody</strong>' ),
'$linkurl' => DI :: l10n () -> t ( 'Please enter a image/video/audio/webpage URL:' ),
'$term' => DI :: l10n () -> t ( 'Tag term:' ),
'$fileas' => DI :: l10n () -> t ( 'Save to Folder:' ),
'$whereareu' => DI :: l10n () -> t ( 'Where are you right now?' ),
'$delitems' => DI :: l10n () -> t ( " Delete item \x28 s \x29 ? " )
2018-01-15 14:05:12 +01:00
]);
2012-09-10 10:36:30 +02:00
2015-06-27 15:11:18 +02:00
$jotplugins = '' ;
2018-12-26 07:06:24 +01:00
Hook :: callAll ( 'jot_tool' , $jotplugins );
2012-09-10 10:36:30 +02:00
2018-10-31 15:35:50 +01:00
// $tpl = Renderer::replaceMacros($tpl,array('$jotplugins' => $jotplugins));
2018-10-31 15:44:06 +01:00
$tpl = Renderer :: getMarkupTemplate ( " jot.tpl " );
2012-09-10 10:36:30 +02:00
2018-10-31 15:35:50 +01:00
$o .= Renderer :: replaceMacros ( $tpl ,[
2020-01-18 20:52:34 +01:00
'$new_post' => DI :: l10n () -> t ( 'New Post' ),
2020-09-09 06:15:25 +02:00
'$return_path' => DI :: args () -> getQueryString (),
2018-01-01 21:27:33 +01:00
'$action' => 'item' ,
2020-01-18 20:52:34 +01:00
'$share' => ( $x [ 'button' ] ? ? '' ) ? : DI :: l10n () -> t ( 'Share' ),
2020-01-30 04:50:10 +01:00
'$loading' => DI :: l10n () -> t ( 'Loading...' ),
2020-01-18 20:52:34 +01:00
'$upload' => DI :: l10n () -> t ( 'Upload photo' ),
'$shortupload' => DI :: l10n () -> t ( 'upload photo' ),
'$attach' => DI :: l10n () -> t ( 'Attach file' ),
'$shortattach' => DI :: l10n () -> t ( 'attach file' ),
'$edbold' => DI :: l10n () -> t ( 'Bold' ),
'$editalic' => DI :: l10n () -> t ( 'Italic' ),
'$eduline' => DI :: l10n () -> t ( 'Underline' ),
'$edquote' => DI :: l10n () -> t ( 'Quote' ),
'$edcode' => DI :: l10n () -> t ( 'Code' ),
'$edimg' => DI :: l10n () -> t ( 'Image' ),
'$edurl' => DI :: l10n () -> t ( 'Link' ),
'$edattach' => DI :: l10n () -> t ( 'Link or Media' ),
'$setloc' => DI :: l10n () -> t ( 'Set your location' ),
'$shortsetloc' => DI :: l10n () -> t ( 'set location' ),
'$noloc' => DI :: l10n () -> t ( 'Clear browser location' ),
'$shortnoloc' => DI :: l10n () -> t ( 'clear location' ),
2019-10-16 14:43:59 +02:00
'$title' => $x [ 'title' ] ? ? '' ,
2020-01-18 20:52:34 +01:00
'$placeholdertitle' => DI :: l10n () -> t ( 'Set title' ),
2019-10-16 14:43:59 +02:00
'$category' => $x [ 'category' ] ? ? '' ,
2020-01-18 20:52:34 +01:00
'$placeholdercategory' => Feature :: isEnabled ( local_user (), 'categories' ) ? DI :: l10n () -> t ( " Categories \x28 comma-separated list \x29 " ) : '' ,
'$wait' => DI :: l10n () -> t ( 'Please wait' ),
'$permset' => DI :: l10n () -> t ( 'Permission settings' ),
2020-09-08 03:25:04 +02:00
'$shortpermset' => DI :: l10n () -> t ( 'Permissions' ),
2018-07-19 15:52:05 +02:00
'$wall' => $notes_cid ? 0 : 1 ,
'$posttype' => $notes_cid ? Item :: PT_PERSONAL_NOTE : Item :: PT_ARTICLE ,
2019-10-16 14:43:59 +02:00
'$content' => $x [ 'content' ] ? ? '' ,
'$post_id' => $x [ 'post_id' ] ? ? '' ,
2019-12-30 23:00:08 +01:00
'$baseurl' => DI :: baseUrl () -> get ( true ),
2018-01-01 21:27:33 +01:00
'$defloc' => $x [ 'default_location' ],
'$visitor' => $x [ 'visitor' ],
'$pvisit' => $notes_cid ? 'none' : $x [ 'visitor' ],
2020-01-18 20:52:34 +01:00
'$public' => DI :: l10n () -> t ( 'Public post' ),
2018-01-01 21:27:33 +01:00
'$lockstate' => $x [ 'lockstate' ],
'$bang' => $x [ 'bang' ],
'$profile_uid' => $x [ 'profile_uid' ],
2020-01-18 20:52:34 +01:00
'$preview' => DI :: l10n () -> t ( 'Preview' ),
2018-01-01 21:27:33 +01:00
'$jotplugins' => $jotplugins ,
'$notes_cid' => $notes_cid ,
2020-01-18 20:52:34 +01:00
'$sourceapp' => DI :: l10n () -> t ( $a -> sourcename ),
'$cancel' => DI :: l10n () -> t ( 'Cancel' ),
2018-11-05 09:37:03 +01:00
'$rand_num' => Crypto :: randomDigits ( 12 ),
2013-01-26 20:52:21 +01:00
// ACL permissions box
2018-01-01 21:27:33 +01:00
'$acl' => $x [ 'acl' ],
2016-06-25 12:21:13 +02:00
//jot nav tab (used in some themes)
2020-01-18 20:52:34 +01:00
'$message' => DI :: l10n () -> t ( 'Message' ),
'$browser' => DI :: l10n () -> t ( 'Browser' ),
2020-04-01 16:10:57 +02:00
'$compose_link_title' => DI :: l10n () -> t ( 'Open Compose page' ),
2018-01-15 14:05:12 +01:00
]);
2012-09-10 10:36:30 +02:00
2017-04-08 20:00:54 +02:00
if ( $popup == true ) {
2018-01-01 21:27:33 +01:00
$o = '<div id="jot-popup" style="display: none;">' . $o . '</div>' ;
2012-09-10 10:36:30 +02:00
}
return $o ;
2011-07-07 14:02:58 +02:00
}
2012-01-03 01:54:37 +01:00
2017-11-19 06:57:06 +01:00
/**
2017-11-19 16:42:00 +01:00
* Plucks the children of the given parent from a given item list .
2017-11-19 14:41:16 +01:00
*
2017-11-19 06:57:06 +01:00
* @ param array $item_list
* @ param array $parent
2019-01-07 16:24:06 +01:00
* @ param bool $recursive
* @ return array
2017-11-19 06:57:06 +01:00
*/
2017-11-19 16:42:00 +01:00
function get_item_children ( array & $item_list , array $parent , $recursive = true )
2017-11-19 06:57:06 +01:00
{
$children = [];
2017-11-19 16:42:00 +01:00
foreach ( $item_list as $i => $item ) {
2020-05-27 14:19:06 +02:00
if ( $item [ 'gravity' ] != GRAVITY_PARENT ) {
2017-11-19 06:57:06 +01:00
if ( $recursive ) {
2012-09-10 10:36:30 +02:00
// Fallback to parent-uri if thr-parent is not set
$thr_parent = $item [ 'thr-parent' ];
2017-04-08 19:05:50 +02:00
if ( $thr_parent == '' ) {
2012-09-10 10:36:30 +02:00
$thr_parent = $item [ 'parent-uri' ];
2017-04-08 19:05:50 +02:00
}
2012-09-10 10:36:30 +02:00
2017-04-04 19:46:56 +02:00
if ( $thr_parent == $parent [ 'uri' ]) {
2017-11-19 06:57:06 +01:00
$item [ 'children' ] = get_item_children ( $item_list , $item );
2012-09-10 10:36:30 +02:00
$children [] = $item ;
2017-11-19 16:42:00 +01:00
unset ( $item_list [ $i ]);
2012-09-10 10:36:30 +02:00
}
2017-04-08 19:05:50 +02:00
} elseif ( $item [ 'parent' ] == $parent [ 'id' ]) {
2012-09-10 10:36:30 +02:00
$children [] = $item ;
2017-11-19 16:42:00 +01:00
unset ( $item_list [ $i ]);
2012-09-10 10:36:30 +02:00
}
}
}
return $children ;
2012-07-27 22:08:57 +02:00
}
2017-11-19 06:57:06 +01:00
/**
2020-01-19 07:05:23 +01:00
* Recursively sorts a tree - like item array
2017-11-19 06:57:06 +01:00
*
* @ param array $items
* @ return array
*/
function sort_item_children ( array $items )
{
2012-09-10 10:36:30 +02:00
$result = $items ;
2019-07-07 23:30:33 +02:00
usort ( $result , 'sort_thr_received_rev' );
2017-04-04 19:46:56 +02:00
foreach ( $result as $k => $i ) {
2017-11-19 06:57:06 +01:00
if ( isset ( $result [ $k ][ 'children' ])) {
2012-09-10 10:36:30 +02:00
$result [ $k ][ 'children' ] = sort_item_children ( $result [ $k ][ 'children' ]);
}
}
return $result ;
2012-07-27 22:08:57 +02:00
}
2017-11-19 06:57:06 +01:00
/**
2020-01-19 07:05:23 +01:00
* Recursively add all children items at the top level of a list
2017-11-19 06:57:06 +01:00
*
* @ param array $children List of items to append
* @ param array $item_list
*/
function add_children_to_list ( array $children , array & $item_list )
{
foreach ( $children as $child ) {
$item_list [] = $child ;
if ( isset ( $child [ 'children' ])) {
add_children_to_list ( $child [ 'children' ], $item_list );
2017-04-04 19:46:56 +02:00
}
2012-09-10 10:36:30 +02:00
}
2012-07-27 22:08:57 +02:00
}
2017-11-19 06:57:06 +01:00
/**
2020-01-19 07:05:23 +01:00
* Selectively flattens a tree - like item structure to prevent threading stairs
*
2017-11-19 06:57:06 +01:00
* This recursive function takes the item tree structure created by conv_sort () and
* flatten the extraneous depth levels when people reply sequentially , removing the
* stairs effect in threaded conversations limiting the available content width .
*
* The basic principle is the following : if a post item has only one reply and is
* the last reply of its parent , then the reply is moved to the parent .
*
* This process is rendered somewhat more complicated because items can be either
* replies or likes , and these don ' t factor at all in the reply count / last reply .
*
* @ param array $parent A tree - like array of items
* @ return array
*/
function smart_flatten_conversation ( array $parent )
{
2018-02-27 10:29:11 +01:00
if ( ! isset ( $parent [ 'children' ]) || count ( $parent [ 'children' ]) == 0 ) {
2017-11-19 06:57:06 +01:00
return $parent ;
2017-04-04 19:46:56 +02:00
}
2012-09-10 10:36:30 +02:00
2017-11-19 06:57:06 +01:00
// We use a for loop to ensure we process the newly-moved items
for ( $i = 0 ; $i < count ( $parent [ 'children' ]); $i ++ ) {
$child = $parent [ 'children' ][ $i ];
2015-11-18 23:57:53 +01:00
2017-11-19 06:57:06 +01:00
if ( isset ( $child [ 'children' ]) && count ( $child [ 'children' ])) {
// This helps counting only the regular posts
$count_post_closure = function ( $var ) {
2019-10-24 00:25:43 +02:00
return $var [ 'verb' ] === Activity :: POST ;
2017-11-19 06:57:06 +01:00
};
$child_post_count = count ( array_filter ( $child [ 'children' ], $count_post_closure ));
$remaining_post_count = count ( array_filter ( array_slice ( $parent [ 'children' ], $i ), $count_post_closure ));
// If there's only one child's children post and this is the last child post
if ( $child_post_count == 1 && $remaining_post_count == 1 ) {
// Searches the post item in the children
$j = 0 ;
2019-10-24 00:25:43 +02:00
while ( $child [ 'children' ][ $j ][ 'verb' ] !== Activity :: POST && $j < count ( $child [ 'children' ])) {
2017-11-19 06:57:06 +01:00
$j ++ ;
}
$moved_item = $child [ 'children' ][ $j ];
unset ( $parent [ 'children' ][ $i ][ 'children' ][ $j ]);
$parent [ 'children' ][] = $moved_item ;
} else {
$parent [ 'children' ][ $i ] = smart_flatten_conversation ( $child );
}
2017-04-04 19:46:56 +02:00
}
}
2015-11-18 23:57:53 +01:00
2017-11-19 06:57:06 +01:00
return $parent ;
}
2012-09-10 10:36:30 +02:00
2017-11-19 06:57:06 +01:00
/**
2020-01-19 07:05:23 +01:00
* Expands a flat list of items into corresponding tree - like conversation structures .
*
2019-07-07 23:30:33 +02:00
* sort the top - level posts either on " received " or " commented " , and finally
2017-11-19 06:57:06 +01:00
* append all the items at the top level ( ? ? ? )
*
* @ param array $item_list A list of items belonging to one or more conversations
2019-07-07 23:30:33 +02:00
* @ param string $order Either on " received " or " commented "
2017-11-19 06:57:06 +01:00
* @ return array
2019-01-07 16:24:06 +01:00
* @ throws \Friendica\Network\HTTPException\InternalServerErrorException
2017-11-19 06:57:06 +01:00
*/
function conv_sort ( array $item_list , $order )
{
$parents = [];
if ( ! ( is_array ( $item_list ) && count ( $item_list ))) {
return $parents ;
}
2018-08-10 06:27:52 +02:00
$blocklist = conv_get_blocklist ();
2017-11-19 06:57:06 +01:00
$item_array = [];
// Dedupes the item list on the uri to prevent infinite loops
foreach ( $item_list as $item ) {
2018-08-10 06:27:52 +02:00
if ( in_array ( $item [ 'author-id' ], $blocklist )) {
continue ;
}
2017-11-19 06:57:06 +01:00
$item_array [ $item [ 'uri' ]] = $item ;
}
// Extract the top level items
foreach ( $item_array as $item ) {
2020-05-27 14:19:06 +02:00
if ( $item [ 'gravity' ] == GRAVITY_PARENT ) {
2017-11-19 06:57:06 +01:00
$parents [] = $item ;
2017-04-04 19:46:56 +02:00
}
}
2012-09-10 10:36:30 +02:00
2019-11-07 08:39:50 +01:00
if ( stristr ( $order , 'pinned_received' )) {
usort ( $parents , 'sort_thr_pinned_received' );
} elseif ( stristr ( $order , 'received' )) {
2019-07-07 23:30:33 +02:00
usort ( $parents , 'sort_thr_received' );
2017-04-08 20:07:28 +02:00
} elseif ( stristr ( $order , 'commented' )) {
2017-04-14 15:30:50 +02:00
usort ( $parents , 'sort_thr_commented' );
2017-04-04 19:46:56 +02:00
}
2012-09-10 10:36:30 +02:00
2017-11-19 16:42:00 +01:00
/*
* Plucks children from the item_array , second pass collects eventual orphan
* items and add them as children of their top - level post .
*/
2017-11-19 06:57:06 +01:00
foreach ( $parents as $i => $parent ) {
2017-11-19 16:42:00 +01:00
$parents [ $i ][ 'children' ] =
2017-11-28 19:54:39 +01:00
array_merge ( get_item_children ( $item_array , $parent , true ),
2017-11-24 22:39:12 +01:00
get_item_children ( $item_array , $parent , false ));
2017-04-04 19:46:56 +02:00
}
2012-09-10 10:36:30 +02:00
2017-11-19 06:57:06 +01:00
foreach ( $parents as $i => $parent ) {
$parents [ $i ][ 'children' ] = sort_item_children ( $parents [ $i ][ 'children' ]);
}
2020-01-18 16:50:57 +01:00
if ( ! DI :: pConfig () -> get ( local_user (), 'system' , 'no_smart_threading' , 0 )) {
2017-11-19 06:57:06 +01:00
foreach ( $parents as $i => $parent ) {
$parents [ $i ] = smart_flatten_conversation ( $parent );
2012-09-10 10:36:30 +02:00
}
}
2017-11-19 06:57:06 +01:00
/// @TODO: Stop recusrsively adding all children back to the top level (!!!)
/// However, this apparently ensures responses (likes, attendance) display (?!)
foreach ( $parents as $parent ) {
if ( count ( $parent [ 'children' ])) {
add_children_to_list ( $parent [ 'children' ], $parents );
2012-09-10 10:36:30 +02:00
}
}
2017-11-19 06:57:06 +01:00
return $parents ;
2012-01-03 01:54:37 +01:00
}
2017-11-19 06:57:06 +01:00
/**
2020-01-19 07:05:23 +01:00
* usort () callback to sort item arrays by pinned and the received key
2017-11-19 06:57:06 +01:00
*
* @ param array $a
* @ param array $b
* @ return int
*/
2019-11-07 08:39:50 +01:00
function sort_thr_pinned_received ( array $a , array $b )
2017-11-19 06:57:06 +01:00
{
2019-11-07 08:09:46 +01:00
if ( $b [ 'pinned' ] && ! $a [ 'pinned' ]) {
return 1 ;
} elseif ( ! $b [ 'pinned' ] && $a [ 'pinned' ]) {
return - 1 ;
}
2019-07-07 23:30:33 +02:00
return strcmp ( $b [ 'received' ], $a [ 'received' ]);
2012-01-03 01:54:37 +01:00
}
2019-11-07 08:39:50 +01:00
/**
2020-01-19 07:05:23 +01:00
* usort () callback to sort item arrays by the received key
2019-11-07 08:39:50 +01:00
*
* @ param array $a
* @ param array $b
* @ return int
*/
function sort_thr_received ( array $a , array $b )
{
return strcmp ( $b [ 'received' ], $a [ 'received' ]);
}
2017-11-19 06:57:06 +01:00
/**
2020-01-19 07:05:23 +01:00
* usort () callback to reverse sort item arrays by the received key
2017-11-19 06:57:06 +01:00
*
* @ param array $a
* @ param array $b
* @ return int
*/
2019-07-07 23:30:33 +02:00
function sort_thr_received_rev ( array $a , array $b )
2017-11-19 06:57:06 +01:00
{
2019-07-07 23:30:33 +02:00
return strcmp ( $a [ 'received' ], $b [ 'received' ]);
2012-01-03 01:54:37 +01:00
}
2017-11-19 06:57:06 +01:00
/**
2020-01-19 07:05:23 +01:00
* usort () callback to sort item arrays by the commented key
2017-11-19 06:57:06 +01:00
*
* @ param array $a
* @ param array $b
2019-01-21 17:36:01 +01:00
* @ return int
2017-11-19 06:57:06 +01:00
*/
function sort_thr_commented ( array $a , array $b )
{
2017-04-08 19:05:50 +02:00
return strcmp ( $b [ 'commented' ], $a [ 'commented' ]);
2012-01-03 01:54:37 +01:00
}