]> git.mxchange.org Git - friendica.git/blob - src/Module/Settings/Display.php
New user option to hide the page drop checkbox
[friendica.git] / src / Module / Settings / Display.php
1 <?php
2 /**
3  * @copyright Copyright (C) 2010-2023, the Friendica project
4  *
5  * @license GNU AGPL version 3 or any later version
6  *
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.
11  *
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.
16  *
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/>.
19  *
20  */
21
22 namespace Friendica\Module\Settings;
23
24 use Friendica\App;
25 use Friendica\Content\Conversation\Collection\Timelines;
26 use Friendica\Content\Text\BBCode;
27 use Friendica\Content\Conversation\Factory\Channel as ChannelFactory;
28 use Friendica\Content\Conversation\Factory\Community as CommunityFactory;
29 use Friendica\Content\Conversation\Factory\Network as NetworkFactory;
30 use Friendica\Content\Conversation\Factory\Timeline as TimelineFactory;
31 use Friendica\Content\Conversation\Repository;
32 use Friendica\Core\Config\Capability\IManageConfigValues;
33 use Friendica\Core\Hook;
34 use Friendica\Core\L10n;
35 use Friendica\Core\PConfig\Capability\IManagePersonalConfigValues;
36 use Friendica\Core\Renderer;
37 use Friendica\Core\Session\Capability\IHandleUserSessions;
38 use Friendica\Core\Theme;
39 use Friendica\Model\User;
40 use Friendica\Module\BaseSettings;
41 use Friendica\Module\Response;
42 use Friendica\Navigation\SystemMessages;
43 use Friendica\Network\HTTPException;
44 use Friendica\Util\Profiler;
45 use Psr\Log\LoggerInterface;
46
47 /**
48  * Module to update user settings
49  */
50 class Display extends BaseSettings
51 {
52         /** @var IManageConfigValues */
53         private $config;
54         /** @var IManagePersonalConfigValues */
55         private $pConfig;
56         /** @var App */
57         private $app;
58         /** @var SystemMessages */
59         private $systemMessages;
60         /** @var ChannelFactory */
61         protected $channel;
62         /** @var Repository\UserDefinedChannel */
63         protected $userDefinedChannel;
64         /** @var CommunityFactory */
65         protected $community;
66         /** @var NetworkFactory */
67         protected $network;
68         /** @var TimelineFactory */
69         protected $timeline;
70
71         public function __construct(Repository\UserDefinedChannel $userDefinedChannel, NetworkFactory $network, CommunityFactory $community, ChannelFactory $channel, TimelineFactory $timeline, SystemMessages $systemMessages, App $app, IManagePersonalConfigValues $pConfig, IManageConfigValues $config, IHandleUserSessions $session, App\Page $page, L10n $l10n, App\BaseURL $baseUrl, App\Arguments $args, LoggerInterface $logger, Profiler $profiler, Response $response, array $server, array $parameters = [])
72         {
73                 parent::__construct($session, $page, $l10n, $baseUrl, $args, $logger, $profiler, $response, $server, $parameters);
74
75                 $this->config             = $config;
76                 $this->pConfig            = $pConfig;
77                 $this->app                = $app;
78                 $this->systemMessages     = $systemMessages;
79                 $this->timeline           = $timeline;
80                 $this->channel            = $channel;
81                 $this->community          = $community;
82                 $this->network            = $network;
83                 $this->userDefinedChannel = $userDefinedChannel;
84         }
85
86         protected function post(array $request = [])
87         {
88                 $uid = $this->session->getLocalUserId();
89                 if (!$uid) {
90                         throw new HTTPException\ForbiddenException($this->t('Permission denied.'));
91                 }
92
93                 self::checkFormSecurityTokenRedirectOnError('/settings/display', 'settings_display');
94
95                 $user = User::getById($uid);
96
97                 $theme                  = trim($request['theme']);
98                 $mobile_theme           = trim($request['mobile_theme']);
99                 $enable_smile           = (bool)$request['enable_smile'];
100                 $enable                 = (array)$request['enable'];
101                 $bookmark               = (array)$request['bookmark'];
102                 $channel_languages      = (array)$request['channel_languages'];
103                 $first_day_of_week      = (bool)$request['first_day_of_week'];
104                 $calendar_default_view  = trim($request['calendar_default_view']);
105                 $infinite_scroll        = (bool)$request['infinite_scroll'];
106                 $enable_smart_threading = (bool)$request['enable_smart_threading'];
107                 $enable_dislike         = (bool)$request['enable_dislike'];
108                 $display_resharer       = (bool)$request['display_resharer'];
109                 $stay_local             = (bool)$request['stay_local'];
110                 $show_page_drop         = (bool)$request['show_page_drop'];
111                 $preview_mode           = (bool)$request['preview_mode'];
112                 $browser_update         = (bool)$request['browser_update'];
113                 if ($browser_update != -1) {
114                         $browser_update = $browser_update * 1000;
115                         if ($browser_update < 10000) {
116                                 $browser_update = 10000;
117                         }
118                 }
119
120                 $enabled_timelines = [];
121                 foreach ($enable as $code => $enabled) {
122                         if ($enabled) {
123                                 $enabled_timelines[] = $code;
124                         }
125                 }
126
127                 $network_timelines = [];
128                 foreach ($bookmark as $code => $bookmarked) {
129                         if ($bookmarked) {
130                                 $network_timelines[] = $code;
131                         }
132                 }
133
134                 $itemspage_network = !empty($request['itemspage_network']) ?
135                         intval($request['itemspage_network']) :
136                         $this->config->get('system', 'itemspage_network');
137                 if ($itemspage_network > 100) {
138                         $itemspage_network = 100;
139                 }
140                 $itemspage_mobile_network = !empty($request['itemspage_mobile_network']) ?
141                         intval($request['itemspage_mobile_network']) :
142                         $this->config->get('system', 'itemspage_network_mobile');
143                 if ($itemspage_mobile_network > 100) {
144                         $itemspage_mobile_network = 100;
145                 }
146
147                 if ($mobile_theme !== '') {
148                         $this->pConfig->set($uid, 'system', 'mobile_theme', $mobile_theme);
149                 }
150
151                 $this->pConfig->set($uid, 'system', 'itemspage_network'       , $itemspage_network);
152                 $this->pConfig->set($uid, 'system', 'itemspage_mobile_network', $itemspage_mobile_network);
153                 $this->pConfig->set($uid, 'system', 'update_interval'         , $browser_update);
154                 $this->pConfig->set($uid, 'system', 'no_smilies'              , !$enable_smile);
155                 $this->pConfig->set($uid, 'system', 'infinite_scroll'         , $infinite_scroll);
156                 $this->pConfig->set($uid, 'system', 'no_smart_threading'      , !$enable_smart_threading);
157                 $this->pConfig->set($uid, 'system', 'hide_dislike'            , !$enable_dislike);
158                 $this->pConfig->set($uid, 'system', 'display_resharer'        , $display_resharer);
159                 $this->pConfig->set($uid, 'system', 'stay_local'              , $stay_local);
160                 $this->pConfig->set($uid, 'system', 'show_page_drop'          , $show_page_drop);
161                 $this->pConfig->set($uid, 'system', 'preview_mode'            , $preview_mode);
162
163                 $this->pConfig->set($uid, 'system', 'network_timelines'       , $network_timelines);
164                 $this->pConfig->set($uid, 'system', 'enabled_timelines'       , $enabled_timelines);
165                 $this->pConfig->set($uid, 'channel', 'languages'              , $channel_languages);
166
167                 $this->pConfig->set($uid, 'calendar', 'first_day_of_week'     , $first_day_of_week);
168                 $this->pConfig->set($uid, 'calendar', 'default_view'          , $calendar_default_view);
169
170                 if (in_array($theme, Theme::getAllowedList())) {
171                         if ($theme == $user['theme']) {
172                                 // call theme_post only if theme has not been changed
173                                 if ($themeconfigfile = Theme::getConfigFile($theme)) {
174                                         require_once $themeconfigfile;
175                                         theme_post($this->app);
176                                 }
177                         } else {
178                                 User::update(['theme' => $theme], $uid);
179                         }
180                 } else {
181                         $this->systemMessages->addNotice($this->t('The theme you chose isn\'t available.'));
182                 }
183
184                 Hook::callAll('display_settings_post', $request);
185
186                 $this->baseUrl->redirect('settings/display');
187         }
188
189         protected function content(array $request = []): string
190         {
191                 parent::content();
192
193                 $uid = $this->session->getLocalUserId();
194                 if (!$uid) {
195                         throw new HTTPException\ForbiddenException($this->t('Permission denied.'));
196                 }
197
198                 $default_theme = $this->config->get('system', 'theme');
199                 if (!$default_theme) {
200                         $default_theme = 'default';
201                 }
202
203                 $default_mobile_theme = $this->config->get('system', 'mobile-theme');
204                 if (!$default_mobile_theme) {
205                         $default_mobile_theme = 'none';
206                 }
207
208                 $user = User::getById($uid);
209
210                 $allowed_themes = Theme::getAllowedList();
211
212                 $themes = [];
213                 $mobile_themes = ['---' => $this->t('No special theme for mobile devices')];
214                 foreach ($allowed_themes as $theme) {
215                         $is_experimental = file_exists('view/theme/' . $theme . '/experimental');
216                         $is_unsupported  = file_exists('view/theme/' . $theme . '/unsupported');
217                         $is_mobile       = file_exists('view/theme/' . $theme . '/mobile');
218                         if (!$is_experimental || $this->config->get('experimental', 'exp_themes')) {
219                                 $theme_name = ucfirst($theme);
220                                 if ($is_unsupported) {
221                                         $theme_name = $this->t('%s - (Unsupported)', $theme_name);
222                                 } elseif ($is_experimental) {
223                                         $theme_name = $this->t('%s - (Experimental)', $theme_name);
224                                 }
225
226                                 if ($is_mobile) {
227                                         $mobile_themes[$theme] = $theme_name;
228                                 } else {
229                                         $themes[$theme] = $theme_name;
230                                 }
231                         }
232                 }
233
234                 $theme_selected        = $user['theme'] ?: $default_theme;
235                 $mobile_theme_selected = $this->session->get('mobile-theme', $default_mobile_theme);
236
237                 $itemspage_network = intval($this->pConfig->get($uid, 'system', 'itemspage_network'));
238                 $itemspage_network = (($itemspage_network > 0 && $itemspage_network < 101) ? $itemspage_network : $this->config->get('system', 'itemspage_network'));
239                 $itemspage_mobile_network = intval($this->pConfig->get($uid, 'system', 'itemspage_mobile_network'));
240                 $itemspage_mobile_network = (($itemspage_mobile_network > 0 && $itemspage_mobile_network < 101) ? $itemspage_mobile_network : $this->config->get('system', 'itemspage_network_mobile'));
241
242                 $browser_update = intval($this->pConfig->get($uid, 'system', 'update_interval'));
243                 if ($browser_update != -1) {
244                         $browser_update = (($browser_update == 0) ? 40 : $browser_update / 1000); // default if not set: 40 seconds
245                 }
246
247                 $enable_smile           = !$this->pConfig->get($uid, 'system', 'no_smilies', false);
248                 $infinite_scroll        =  $this->pConfig->get($uid, 'system', 'infinite_scroll', false);
249                 $enable_smart_threading = !$this->pConfig->get($uid, 'system', 'no_smart_threading', false);
250                 $enable_dislike         = !$this->pConfig->get($uid, 'system', 'hide_dislike', false);
251                 $display_resharer       =  $this->pConfig->get($uid, 'system', 'display_resharer', false);
252                 $stay_local             =  $this->pConfig->get($uid, 'system', 'stay_local', false);
253                 $show_page_drop         =  $this->pConfig->get($uid, 'system', 'show_page_drop', true);
254
255                 $preview_mode  =  $this->pConfig->get($uid, 'system', 'preview_mode', BBCode::PREVIEW_LARGE);
256                 $preview_modes = [
257                         BBCode::PREVIEW_NONE     => $this->t('No preview'),
258                         BBCode::PREVIEW_NO_IMAGE => $this->t('No image'),
259                         BBCode::PREVIEW_SMALL    => $this->t('Small Image'),
260                         BBCode::PREVIEW_LARGE    => $this->t('Large Image'),
261                 ];
262
263                 $bookmarked_timelines = $this->pConfig->get($uid, 'system', 'network_timelines', $this->getAvailableTimelines($uid, true)->column('code'));
264                 $enabled_timelines    = $this->pConfig->get($uid, 'system', 'enabled_timelines', $this->getAvailableTimelines($uid, false)->column('code'));
265                 $channel_languages = $this->pConfig->get($uid, 'channel', 'languages', [User::getLanguageCode($uid)]);
266                 $languages         = $this->l10n->getLanguageCodes(true);
267
268                 $timelines = [];
269                 foreach ($this->getAvailableTimelines($uid) as $timeline) {
270                         $timelines[] = [
271                                 'label'        => $timeline->label,
272                                 'description'  => $timeline->description,
273                                 'enable'       => ["enable[{$timeline->code}]", '', in_array($timeline->code, $enabled_timelines)],
274                                 'bookmark'     => ["bookmark[{$timeline->code}]", '', in_array($timeline->code, $bookmarked_timelines)],
275                         ];
276                 }
277
278                 $first_day_of_week = $this->pConfig->get($uid, 'calendar', 'first_day_of_week', 0);
279                 $weekdays          = [
280                         0 => $this->t('Sunday'),
281                         1 => $this->t('Monday'),
282                         2 => $this->t('Tuesday'),
283                         3 => $this->t('Wednesday'),
284                         4 => $this->t('Thursday'),
285                         5 => $this->t('Friday'),
286                         6 => $this->t('Saturday')
287                 ];
288
289                 $calendar_default_view = $this->pConfig->get($uid, 'calendar', 'default_view', 'month');
290                 $calendarViews         = [
291                         'month'      => $this->t('month'),
292                         'agendaWeek' => $this->t('week'),
293                         'agendaDay'  => $this->t('day'),
294                         'listMonth'  => $this->t('list')
295                 ];
296
297                 $theme_config = '';
298                 if ($themeconfigfile = Theme::getConfigFile($theme_selected)) {
299                         require_once $themeconfigfile;
300                         $theme_config = theme_content($this->app);
301                 }
302
303                 $tpl = Renderer::getMarkupTemplate('settings/display.tpl');
304                 return Renderer::replaceMacros($tpl, [
305                         '$ptitle'         => $this->t('Display Settings'),
306                         '$submit'         => $this->t('Save Settings'),
307                         '$d_tset'         => $this->t('General Theme Settings'),
308                         '$d_ctset'        => $this->t('Custom Theme Settings'),
309                         '$d_cset'         => $this->t('Content Settings'),
310                         '$stitle'         => $this->t('Theme settings'),
311                         '$timeline_title' => $this->t('Timelines'),
312                         '$channel_title'  => $this->t('Channels'),
313                         '$calendar_title' => $this->t('Calendar'),
314
315                         '$form_security_token' => self::getFormSecurityToken('settings_display'),
316                         '$uid'                 => $uid,
317
318                         '$theme'            => ['theme', $this->t('Display Theme:'), $theme_selected, '', $themes, true],
319                         '$mobile_theme' => ['mobile_theme', $this->t('Mobile Theme:'), $mobile_theme_selected, '', $mobile_themes, false],
320                         '$theme_config' => $theme_config,
321
322                         '$itemspage_network'        => ['itemspage_network'       , $this->t('Number of items to display per page:'), $itemspage_network, $this->t('Maximum of 100 items')],
323                         '$itemspage_mobile_network' => ['itemspage_mobile_network', $this->t('Number of items to display per page when viewed from mobile device:'), $itemspage_mobile_network, $this->t('Maximum of 100 items')],
324                         '$ajaxint'                  => ['browser_update'          , $this->t('Update browser every xx seconds'), $browser_update, $this->t('Minimum of 10 seconds. Enter -1 to disable it.')],
325                         '$enable_smile'             => ['enable_smile'            , $this->t('Display emoticons'), $enable_smile, $this->t('When enabled, emoticons are replaced with matching symbols.')],
326                         '$infinite_scroll'          => ['infinite_scroll'         , $this->t('Infinite scroll'), $infinite_scroll, $this->t('Automatic fetch new items when reaching the page end.')],
327                         '$enable_smart_threading'   => ['enable_smart_threading'  , $this->t('Enable Smart Threading'), $enable_smart_threading, $this->t('Enable the automatic suppression of extraneous thread indentation.')],
328                         '$enable_dislike'           => ['enable_dislike'          , $this->t('Display the Dislike feature'), $enable_dislike, $this->t('Display the Dislike button and dislike reactions on posts and comments.')],
329                         '$display_resharer'         => ['display_resharer'        , $this->t('Display the resharer'), $display_resharer, $this->t('Display the first resharer as icon and text on a reshared item.')],
330                         '$stay_local'               => ['stay_local'              , $this->t('Stay local'), $stay_local, $this->t("Don't go to a remote system when following a contact link.")],
331                         '$show_page_drop'           => ['show_page_drop'          , $this->t('Show the page drop checkbox'), $show_page_drop, $this->t("Display the checkbox for the page drop on the network page.")],
332                         '$preview_mode'             => ['preview_mode'            , $this->t('Link preview mode'), $preview_mode, $this->t('Appearance of the link preview that is added to each post with a link.'), $preview_modes, false],
333
334                         '$timeline_label'       => $this->t('Label'),
335                         '$timeline_descriptiom' => $this->t('Description'),
336                         '$timeline_enable'      => $this->t('Enable'),
337                         '$timeline_bookmark'    => $this->t('Bookmark'),
338                         '$timelines'            => $timelines,
339                         '$timeline_explanation' => $this->t('Enable timelines that you want to see in the channels widget. Bookmark timelines that you want to see in the top menu.'),
340
341                         '$channel_languages' => ['channel_languages[]', $this->t('Channel languages:'), $channel_languages, $this->t('Select all languages that you want to see in your channels.'), $languages, 'multiple'],
342
343                         '$first_day_of_week'     => ['first_day_of_week'    , $this->t('Beginning of week:')    , $first_day_of_week    , '', $weekdays     , false],
344                         '$calendar_default_view' => ['calendar_default_view', $this->t('Default calendar view:'), $calendar_default_view, '', $calendarViews, false],
345                 ]);
346         }
347
348         private function getAvailableTimelines(int $uid, bool $only_network = false): Timelines
349         {
350                 $timelines = [];
351
352                 foreach ($this->network->getTimelines('') as $channel) {
353                         $timelines[] = $channel;
354                 }
355
356                 if ($only_network) {
357                         return new Timelines($timelines);
358                 }
359
360                 foreach ($this->channel->getTimelines($uid) as $channel) {
361                         $timelines[] = $channel;
362                 }
363
364                 foreach ($this->userDefinedChannel->selectByUid($uid) as $channel) {
365                         $timelines[] = $channel;
366                 }
367
368                 foreach ($this->community->getTimelines(true) as $community) {
369                         $timelines[] = $community;
370                 }
371
372                 return new Timelines($timelines);
373         }
374 }