2016-03-28 16:29:05 +02:00
< ? php
/*
* Name : frio
2016-06-13 10:57:42 +02:00
* Description : Bootstrap V3 theme . The theme is currently under construction , so it is far from finished . For further information have a look at the < a href = " https://github.com/friendica/friendica/tree/develop/view/theme/frio/README.md " > ReadMe </ a >.
2017-05-19 13:11:30 +02:00
* Version : V . 0.8
2016-03-28 16:29:05 +02:00
* Author : Rabuzarus < https :// friendica . kommune4 . de / profile / rabuzarus >
2017-01-09 13:06:08 +01:00
*
2016-03-28 16:29:05 +02:00
*/
2017-04-30 06:07:00 +02:00
use Friendica\App ;
2016-03-28 16:29:05 +02:00
$frio = " view/theme/frio " ;
global $frio ;
2017-01-09 13:06:08 +01:00
function frio_init ( App $a ) {
2016-06-20 13:44:11 +02:00
// disable the events module link in the profile tab
$a -> theme_events_in_profile = false ;
2016-03-28 16:29:05 +02:00
set_template_engine ( $a , 'smarty3' );
2016-12-19 14:26:13 +01:00
$baseurl = App :: get_baseurl ();
2016-03-28 16:29:05 +02:00
$style = get_pconfig ( local_user (), 'frio' , 'style' );
$frio = " view/theme/frio " ;
global $frio ;
2016-05-28 13:07:24 +02:00
// if the device is a mobile device set js is_mobile
// variable so the js scripts can use this information
if ( $a -> is_mobile || $a -> is_tablet ) {
$a -> page [ " htmlhead " ] .= <<< EOT
< script >
var is_mobile = 1 ;
</ script >
EOT ;
}
2016-03-28 16:29:05 +02:00
if ( $style == " " )
$style = get_config ( 'frio' , 'style' );
}
function frio_install () {
2016-04-14 14:07:20 +02:00
register_hook ( 'prepare_body_final' , 'view/theme/frio/theme.php' , 'frio_item_photo_links' );
2016-04-17 00:55:43 +02:00
register_hook ( 'item_photo_menu' , 'view/theme/frio/theme.php' , 'frio_item_photo_menu' );
2016-04-20 19:32:19 +02:00
register_hook ( 'contact_photo_menu' , 'view/theme/frio/theme.php' , 'frio_contact_photo_menu' );
2016-04-19 02:06:02 +02:00
register_hook ( 'nav_info' , 'view/theme/frio/theme.php' , 'frio_remote_nav' );
2016-04-28 21:35:33 +02:00
register_hook ( 'acl_lookup_end' , 'view/theme/frio/theme.php' , 'frio_acl_lookup' );
2017-06-07 21:51:37 +02:00
register_hook ( 'construct_acl_data' , 'view/theme/frio/theme.php' , 'frio_construct_acl_data' );
2016-03-28 16:29:05 +02:00
logger ( " installed theme frio " );
}
function frio_uninstall () {
2016-04-14 14:07:20 +02:00
unregister_hook ( 'prepare_body_final' , 'view/theme/frio/theme.php' , 'frio_item_photo_links' );
2016-04-17 00:55:43 +02:00
unregister_hook ( 'item_photo_menu' , 'view/theme/frio/theme.php' , 'frio_item_photo_menu' );
2016-04-28 21:35:33 +02:00
unregister_hook ( 'contact_photo_menu' , 'view/theme/frio/theme.php' , 'frio_contact_photo_menu' );
2016-04-20 19:32:19 +02:00
unregister_hook ( 'nav_info' , 'view/theme/frio/theme.php' , 'frio_remote_nav' );
2016-04-28 21:35:33 +02:00
unregister_hook ( 'acl_lookup_end' , 'view/theme/frio/theme.php' , 'frio_acl_lookup' );
2017-06-07 21:51:37 +02:00
unregister_hook ( 'construct_acl_data' , 'view/theme/frio/theme.php' , 'frio_construct_acl_data' );
2016-03-28 16:29:05 +02:00
logger ( " uninstalled theme frio " );
}
2016-04-14 14:07:20 +02:00
/**
2017-01-09 13:06:08 +01:00
* @ brief Replace friendica photo links hook
*
2016-04-14 14:07:20 +02:00
* This function does replace the links to photos
* of other friendica users . Original the photos are
* linked to the photo page . Now they will linked directly
* to the photo file . This function is nessesary to use colorbox
* in the network stream
2017-01-09 13:06:08 +01:00
*
* @ param App $a Unused but required by hook definition
2016-04-14 14:07:20 +02:00
* @ param array $body_info The item and its html output
*/
2017-01-09 13:06:08 +01:00
function frio_item_photo_links ( App $a , & $body_info ) {
2016-04-14 14:07:20 +02:00
require_once ( 'include/Photo.php' );
$phototypes = Photo :: supportedTypes ();
$occurence = 1 ;
$p = bb_find_open_close ( $body_info [ 'html' ], " <a " , " > " );
2017-03-21 17:02:59 +01:00
while ( $p !== false && ( $occurence ++ < 500 )) {
2016-04-14 14:07:20 +02:00
$link = substr ( $body_info [ 'html' ], $p [ 'start' ], $p [ 'end' ] - $p [ 'start' ]);
$matches = array ();
preg_match ( " / \ /photos \ /[ \ w]+ \ /image \ /([ \ w]+)/ " , $link , $matches );
2017-03-21 17:02:59 +01:00
if ( $matches ) {
2016-04-14 14:07:20 +02:00
// Replace the link for the photo's page with a direct link to the photo itself
$newlink = str_replace ( $matches [ 0 ], " /photo/ { $matches [ 1 ] } " , $link );
// Add a "quiet" parameter to any redir links to prevent the "XX welcomes YY" info boxes
$newlink = preg_replace ( " /href= \" ([^ \" ]+) \ /redir \ /([^ \" ]+)&url=([^ \" ]+) \" / " , 'href="$1/redir/$2&quiet=1&url=$3"' , $newlink );
// Having any arguments to the link for Colorbox causes it to fetch base64 code instead of the image
$newlink = preg_replace ( " / \ /[?&]zrl=([^& \" ]+)/ " , '' , $newlink );
$body_info [ 'html' ] = str_replace ( $link , $newlink , $body_info [ 'html' ]);
}
$p = bb_find_open_close ( $body_info [ 'html' ], " <a " , " > " , $occurence );
}
}
2016-04-17 00:55:43 +02:00
/**
2017-01-09 13:06:08 +01:00
* @ brief Replace links of the item_photo_menu hook
*
2016-04-17 00:55:43 +02:00
* This function replaces the original poke and the message links
* to call the addToModal javascript function so this pages can
* be loaded in a bootstrap modal
2017-01-09 13:06:08 +01:00
*
* @ param App $a Unused but required by the hook definition
2016-04-17 00:55:43 +02:00
* @ param array $arr Contains item data and the original photo_menu
*/
2017-01-09 13:06:08 +01:00
function frio_item_photo_menu ( App $a , & $arr ) {
2016-04-17 00:55:43 +02:00
foreach ( $arr [ " menu " ] as $k => $v ) {
if ( strpos ( $v , 'poke/?f=&c=' ) === 0 || strpos ( $v , 'message/new/' ) === 0 ) {
$v = " javascript:addToModal(' " . $v . " '); return false; " ;
$arr [ " menu " ][ $k ] = $v ;
}
}
$args = array ( 'item' => $item , 'menu' => $menu );
}
2016-04-19 02:06:02 +02:00
2016-04-20 19:32:19 +02:00
/**
* @ brief Replace links of the contact_photo_menu
2017-01-09 13:06:08 +01:00
*
2016-04-20 19:32:19 +02:00
* This function replaces the original poke and the message links
* to call the addToModal javascript function so this pages can
* be loaded in a bootstrap modal
2016-04-20 19:59:44 +02:00
* Additionally the profile , status and photo page links will be changed
* to don ' t open in a new tab if the contact is a friendica contact .
2017-01-09 13:06:08 +01:00
*
2017-04-30 06:17:49 +02:00
* @ param App $a The app data
2016-04-20 19:32:19 +02:00
* @ param array $args Contains contact data and the original photo_menu
*/
2017-01-09 13:06:08 +01:00
function frio_contact_photo_menu ( App $a , & $args ){
2016-04-20 19:32:19 +02:00
$pokelink = " " ;
$pmlink = " " ;
$cid = " " ;
$cid = $args [ " contact " ][ " id " ];
$pokelink = $args [ " menu " ][ " poke " ][ 1 ];
$pmlink = $args [ " menu " ][ " pm " ][ 1 ];
2016-04-20 19:59:44 +02:00
// Set the the indicator for opening the status, profile and photo pages
// in a new tab to false if the contact a dfrn (friendica) contact
// We do this because we can go back on foreign friendica pages throuhg
// friendicas "magic-link" which indicates a friendica user on froreign
// friendica servers as remote user or visitor
//
2017-01-09 13:06:08 +01:00
// The value for opening in a new tab is e.g. when
2016-04-20 19:59:44 +02:00
// $args["menu"]["status"][2] is true. If the value of the [2] key is true
// and if it's a friendica contact we set it to false
foreach ( $args [ " menu " ] as $k => $v ) {
if ( $k === " status " || $k === " profile " || $k === " photos " ) {
$v [ 2 ] = (( $args [ " contact " ][ " network " ] === " dfrn " ) ? false : true );
$args [ " menu " ][ $k ][ 2 ] = $v [ 2 ];
}
}
2016-04-20 19:32:19 +02:00
// Add to pm and poke links a new key with the value 'modal'.
// Later we can make conditions in the corresponing templates (e.g.
// contact_template.tpl)
if ( strpos ( $pokelink , 'poke/?f=&c=' . $cid ) !== false )
$args [ " menu " ][ " poke " ][ 3 ] = " modal " ;
if ( strpos ( $pmlink , 'message/new/' . $cid ) !== false )
$args [ " menu " ][ " pm " ][ 3 ] = " modal " ;
$args = array ( 'contact' => $contact , 'menu' => & $menu );
}
2016-04-19 02:06:02 +02:00
/**
* @ brief Construct remote nav menu
2017-01-09 13:06:08 +01:00
*
2016-04-19 02:06:02 +02:00
* It creates a remote baseurl form $_SESSION for remote users and friendica
2017-01-09 13:06:08 +01:00
* visitors . This url will be added to some of the nav links . With this behaviour
2016-04-19 02:06:02 +02:00
* the user will come back to her / his own pages on his / her friendica server .
* Not all possible links are available ( notifications , administrator , manage ,
* notes aren ' t available because we have no way the check remote permissions ) ..
* Some links will point to the local pages because the user would expect
* local page ( these pages are : search , community , help , apps , directory ) .
2017-01-09 13:06:08 +01:00
*
2017-04-30 06:17:49 +02:00
* @ param App $a The App class
2016-04-19 02:06:02 +02:00
* @ param array $nav The original nav menu
*/
function frio_remote_nav ( $a , & $nav ) {
// get the homelink from $_XSESSION
$homelink = get_my_url ();
if ( ! $homelink )
$homelink = (( x ( $_SESSION , 'visitor_home' )) ? $_SESSION [ 'visitor_home' ] : '' );
// split up the url in it's parts (protocol,domain/directory, /profile/, nickname
// I'm not familiar with regex, so someone might find a better solutionen
2017-01-09 13:06:08 +01:00
//
2016-04-19 02:06:02 +02:00
// E.g $homelink = 'https://friendica.domain.com/profile/mickey' should result in an array
// with 0 => 'https://friendica.domain.com/profile/mickey' 1 => 'https://',
// 2 => 'friendica.domain.com' 3 => '/profile/' 4 => 'mickey'
2017-01-09 13:06:08 +01:00
//
2016-04-19 02:06:02 +02:00
//$server_url = preg_match('/^(https?:\/\/.*?)\/profile\//2', $homelink);
preg_match ( '/^(https?:\/\/)?(.*?)(\/profile\/)(.*)/' , $homelink , $url_parts );
// Construct the server url of the visitor. So we could link back to his/her own menu.
// And construct a webbie (e.g. mickey@friendica.domain.com for the search in gcontact
// We use the webbie for search in gcontact because we don't know if gcontact table stores
// the right value if its http or https protocol
if ( count ( $url_parts )) {
$server_url = $url_parts [ 1 ] . $url_parts [ 2 ];
$webbie = $url_parts [ 4 ] . '@' . $url_parts [ 2 ];
}
// since $userinfo isn't available for the hook we write it to the nav array
// this isn't optimal because the contact query will be done now twice
if ( local_user ()) {
// empty the server url for local user because we won't need it
$server_url = '' ;
// user info
2017-01-09 10:37:37 +01:00
$r = q ( " SELECT `micro` FROM `contact` WHERE `uid` = %d AND `self` " , intval ( $a -> user [ 'uid' ]));
2017-01-09 13:06:08 +01:00
2016-12-13 10:44:13 +01:00
$r [ 0 ][ 'photo' ] = ( dbm :: is_result ( $r ) ? $a -> remove_baseurl ( $r [ 0 ][ 'micro' ]) : " images/person-48.jpg " );
2016-04-19 02:06:02 +02:00
$r [ 0 ][ 'name' ] = $a -> user [ 'username' ];
} elseif ( ! local_user () && remote_user ()) {
$r = q ( " SELECT `name`, `nick`, `micro` AS `photo` FROM `contact` WHERE `id` = %d " , intval ( remote_user ()));
2016-08-09 14:39:18 +02:00
$nav [ 'remote' ] = t ( " Guest " );
2016-04-19 02:06:02 +02:00
} elseif ( get_my_url ()) {
$r = q ( " SELECT `name`, `nick`, `photo` FROM `gcontact`
WHERE `addr` = '%s' AND `network` = 'dfrn' " ,
dbesc ( $webbie ));
2016-04-20 18:03:22 +02:00
$nav [ 'remote' ] = t ( " Visitor " );
2016-04-19 02:06:02 +02:00
}
2016-12-14 09:42:28 +01:00
if ( dbm :: is_result ( $r )){
2016-04-19 02:06:02 +02:00
$nav [ 'userinfo' ] = array (
2016-12-13 10:44:13 +01:00
'icon' => ( dbm :: is_result ( $r ) ? $r [ 0 ][ 'photo' ] : " images/person-48.jpg " ),
2016-04-19 03:17:06 +02:00
'name' => $r [ 0 ][ 'name' ],
2016-04-19 02:06:02 +02:00
);
}
if ( ! local_user () && ! empty ( $server_url )) {
2017-04-08 16:57:24 +02:00
$nav [ 'logout' ] = Array ( $server_url . '/logout' , t ( 'Logout' ), " " , t ( 'End this session' ));
2016-04-19 02:06:02 +02:00
// user menu
$nav [ 'usermenu' ][] = Array ( $server_url . '/profile/' . $a -> user [ 'nickname' ], t ( 'Status' ), " " , t ( 'Your posts and conversations' ));
$nav [ 'usermenu' ][] = Array ( $server_url . '/profile/' . $a -> user [ 'nickname' ] . '?tab=profile' , t ( 'Profile' ), " " , t ( 'Your profile page' ));
$nav [ 'usermenu' ][] = Array ( $server_url . '/photos/' . $a -> user [ 'nickname' ], t ( 'Photos' ), " " , t ( 'Your photos' ));
$nav [ 'usermenu' ][] = Array ( $server_url . '/videos/' . $a -> user [ 'nickname' ], t ( 'Videos' ), " " , t ( 'Your videos' ));
$nav [ 'usermenu' ][] = Array ( $server_url . '/events/' , t ( 'Events' ), " " , t ( 'Your events' ));
// navbar links
$nav [ 'network' ] = array ( $server_url . '/network' , t ( 'Network' ), " " , t ( 'Conversations from your friends' ));
$nav [ 'events' ] = Array ( $server_url . '/events' , t ( 'Events' ), " " , t ( 'Events and Calendar' ));
$nav [ 'messages' ] = array ( $server_url . '/message' , t ( 'Messages' ), " " , t ( 'Private mail' ));
$nav [ 'settings' ] = array ( $server_url . '/settings' , t ( 'Settings' ), " " , t ( 'Account settings' ));
$nav [ 'contacts' ] = array ( $server_url . '/contacts' , t ( 'Contacts' ), " " , t ( 'Manage/edit friends and contacts' ));
2016-04-20 18:03:22 +02:00
$nav [ 'sitename' ] = $a -> config [ 'sitename' ];
2016-04-19 02:06:02 +02:00
}
}
2016-04-28 21:35:33 +02:00
/**
* @ brief : Search for contacts
2017-01-09 13:06:08 +01:00
*
2016-04-28 21:35:33 +02:00
* This function search for a users contacts . The code is copied from contact search
* in / mod / contacts . php . With this function the contacts will permitted to acl_lookup ()
2017-01-09 13:06:08 +01:00
* and can grabbed as json . For this we use the type = " r " . This is usful to to let js
2016-04-28 21:35:33 +02:00
* grab the contact data .
* We use this to give the data to textcomplete and have a filter function at the
* contact page .
2017-01-09 13:06:08 +01:00
*
2016-12-20 21:16:01 +01:00
* @ param App $a The app data @ TODO Unused
2016-04-28 21:35:33 +02:00
* @ param array $results The array with the originals from acl_lookup ()
*/
2017-04-30 06:17:49 +02:00
function frio_acl_lookup ( App $a , & $results ) {
2016-04-28 21:35:33 +02:00
require_once ( " mod/contacts.php " );
$nets = (( x ( $_GET , " nets " )) ? notags ( trim ( $_GET [ " nets " ])) : " " );
// we introduce a new search type, r should do the same query like it's
// done in /mod/contacts for connections
2016-12-20 21:16:01 +01:00
if ( $results [ " type " ] == " r " ) {
2016-04-28 21:35:33 +02:00
$searching = false ;
2016-12-20 21:16:01 +01:00
if ( $search ) {
2016-04-28 21:35:33 +02:00
$search_hdr = $search ;
$search_txt = dbesc ( protect_sprintf ( preg_quote ( $search )));
$searching = true ;
}
$sql_extra .= (( $searching ) ? " AND (`attag` LIKE '%% " . dbesc ( $search_txt ) . " %%' OR `name` LIKE '%% " . dbesc ( $search_txt ) . " %%' OR `nick` LIKE '%% " . dbesc ( $search_txt ) . " %%') " : " " );
2016-12-20 21:16:01 +01:00
if ( $nets ) {
2016-04-28 21:35:33 +02:00
$sql_extra .= sprintf ( " AND network = '%s' " , dbesc ( $nets ));
2016-12-20 21:16:01 +01:00
}
2016-04-28 21:35:33 +02:00
$sql_extra2 = ((( $sort_type > 0 ) && ( $sort_type <= CONTACT_IS_FRIEND )) ? sprintf ( " AND `rel` = %d " , intval ( $sort_type )) : '' );
$r = q ( " SELECT COUNT(*) AS `total` FROM `contact`
2017-01-09 10:37:37 +01:00
WHERE `uid` = % d AND NOT `self` AND NOT `pending` $sql_extra $sql_extra2 " ,
2016-04-28 21:35:33 +02:00
intval ( $_SESSION [ 'uid' ]));
2016-12-14 09:42:28 +01:00
if ( dbm :: is_result ( $r )) {
2016-04-28 21:35:33 +02:00
$total = $r [ 0 ][ " total " ];
}
$sql_extra3 = unavailable_networks ();
2017-01-09 10:37:37 +01:00
$r = q ( " SELECT * FROM `contact` WHERE `uid` = %d AND NOT `self` AND NOT `pending` $sql_extra $sql_extra2 $sql_extra3 ORDER BY `name` ASC LIMIT 100 " ,
2016-04-28 21:35:33 +02:00
intval ( $_SESSION [ 'uid' ])
);
$contacts = array ();
2016-12-14 09:42:28 +01:00
if ( dbm :: is_result ( $r )) {
2016-12-20 21:16:01 +01:00
foreach ( $r as $rr ) {
2016-04-28 21:35:33 +02:00
$contacts [] = _contact_detail_for_template ( $rr );
}
}
$results [ " items " ] = $contacts ;
$results [ " tot " ] = $total ;
}
2016-06-25 12:21:13 +02:00
}
2017-06-07 21:51:37 +02:00
/**
* @ brief : Deactivate the old style acl selector
*
* We don ' t need the old one for Frio . Deactivating makes page loading much faster
*
* @ param App $a The app data @ TODO Unused
* @ param array $arr We use this array to stop processing . See construct_acl_data in include / acl_selectors . php
*/
function frio_construct_acl_data ( App $a , & $arr ) {
$arr [ 'cancel' ] = true ;
}