* * @author Arthur Schiwon * @author Joas Schilling * @author Lukas Reschke * @author Marius Blüm * @author Morris Jobke * @author Robin Appelman * @author Roeland Jago Douma * * @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 . * */ namespace OC\Settings; use OCP\AppFramework\QueryException; use OCP\IL10N; use OCP\ILogger; use OCP\IServerContainer; use OCP\IURLGenerator; use OCP\Settings\ISettings; use OCP\Settings\IManager; use OCP\Settings\ISection; class Manager implements IManager { /** @var ILogger */ private $log; /** @var IL10N */ private $l; /** @var IURLGenerator */ private $url; /** @var IServerContainer */ private $container; public function __construct( ILogger $log, IL10N $l10n, IURLGenerator $url, IServerContainer $container ) { $this->log = $log; $this->l = $l10n; $this->url = $url; $this->container = $container; } /** @var array */ protected $sectionClasses = []; /** @var array */ protected $sections = []; /** * @param string $type 'admin' or 'personal' * @param string $section Class must implement OCP\Settings\ISection * * @return void */ public function registerSection(string $type, string $section) { if (!isset($this->sectionClasses[$type])) { $this->sectionClasses[$type] = []; } $this->sectionClasses[$type][] = $section; } /** * @param string $type 'admin' or 'personal' * * @return ISection[] */ protected function getSections(string $type): array { if (!isset($this->sections[$type])) { $this->sections[$type] = []; } if (!isset($this->sectionClasses[$type])) { return $this->sections[$type]; } foreach ($this->sectionClasses[$type] as $index => $class) { try { /** @var ISection $section */ $section = \OC::$server->query($class); } catch (QueryException $e) { $this->log->logException($e, ['level' => ILogger::INFO]); continue; } if (!$section instanceof ISection) { $this->log->logException(new \InvalidArgumentException('Invalid settings section registered'), ['level' => ILogger::INFO]); continue; } $sectionID = $section->getID(); if (isset($this->sections[$type][$sectionID])) { $this->log->logException(new \InvalidArgumentException('Section with the same ID already registered'), ['level' => ILogger::INFO]); continue; } $this->sections[$type][$sectionID] = $section; unset($this->sectionClasses[$type][$index]); } return $this->sections[$type]; } /** @var array */ protected $settingClasses = []; /** @var array */ protected $settings = []; /** * @param string $type 'admin' or 'personal' * @param string $setting Class must implement OCP\Settings\ISetting * * @return void */ public function registerSetting(string $type, string $setting) { $this->settingClasses[$setting] = $type; } /** * @param string $type 'admin' or 'personal' * @param string $section * * @return ISettings[] */ protected function getSettings(string $type, string $section): array { if (!isset($this->settings[$type])) { $this->settings[$type] = []; } if (!isset($this->settings[$type][$section])) { $this->settings[$type][$section] = []; } foreach ($this->settingClasses as $class => $settingsType) { try { /** @var ISettings $setting */ $setting = \OC::$server->query($class); } catch (QueryException $e) { $this->log->logException($e, ['level' => ILogger::INFO]); continue; } if (!$setting instanceof ISettings) { $this->log->logException(new \InvalidArgumentException('Invalid settings setting registered (' . $class . ')'), ['level' => ILogger::INFO]); continue; } if (!isset($this->settings[$settingsType][$setting->getSection()])) { $this->settings[$settingsType][$setting->getSection()] = []; } $this->settings[$settingsType][$setting->getSection()][] = $setting; unset($this->settingClasses[$class]); } return $this->settings[$type][$section]; } /** * @inheritdoc */ public function getAdminSections(): array { // built-in sections $sections = [ 0 => [new Section('overview', $this->l->t('Overview'), 0, $this->url->imagePath('settings', 'admin.svg'))], 1 => [new Section('server', $this->l->t('Basic settings'), 0, $this->url->imagePath('core', 'actions/settings-dark.svg'))], 5 => [new Section('sharing', $this->l->t('Sharing'), 0, $this->url->imagePath('core', 'actions/share.svg'))], 10 => [new Section('security', $this->l->t('Security'), 0, $this->url->imagePath('core', 'actions/password.svg'))], 50 => [new Section('groupware', $this->l->t('Groupware'), 0, $this->url->imagePath('core', 'places/contacts.svg'))], 98 => [new Section('additional', $this->l->t('Additional settings'), 0, $this->url->imagePath('core', 'actions/settings-dark.svg'))], ]; $appSections = $this->getSections('admin'); foreach ($appSections as $section) { /** @var ISection $section */ if (!isset($sections[$section->getPriority()])) { $sections[$section->getPriority()] = []; } $sections[$section->getPriority()][] = $section; } ksort($sections); return $sections; } /** * @param string $section * * @return ISection[] */ private function getBuiltInAdminSettings($section): array { $forms = []; if ($section === 'overview') { /** @var ISettings $form */ $form = $this->container->query(Admin\Overview::class); $forms[$form->getPriority()] = [$form]; } if ($section === 'server') { /** @var ISettings $form */ $form = $this->container->query(Admin\Server::class); $forms[$form->getPriority()] = [$form]; $form = $this->container->query(Admin\Mail::class); $forms[$form->getPriority()] = [$form]; $form = $this->container->query(Admin\ServerInfo::class); $forms[$form->getPriority()] = [$form]; } if ($section === 'security') { /** @var ISettings $form */ $form = $this->container->query(Admin\Security::class); $forms[$form->getPriority()] = [$form]; } if ($section === 'sharing') { /** @var ISettings $form */ $form = $this->container->query(Admin\Sharing::class); $forms[$form->getPriority()] = [$form]; } return $forms; } /** * @param string $section * * @return ISection[] */ private function getBuiltInPersonalSettings($section): array { $forms = []; if ($section === 'personal-info') { /** @var ISettings $form */ $form = $this->container->query(Personal\PersonalInfo::class); $forms[$form->getPriority()] = [$form]; $form = new Personal\ServerDevNotice(); $forms[$form->getPriority()] = [$form]; } if ($section === 'security') { /** @var ISettings $form */ $form = $this->container->query(Personal\Security::class); $forms[$form->getPriority()] = [$form]; } if ($section === 'additional') { /** @var ISettings $form */ $form = $this->container->query(Personal\Additional::class); $forms[$form->getPriority()] = [$form]; } return $forms; } /** * @inheritdoc */ public function getAdminSettings($section): array { $settings = $this->getBuiltInAdminSettings($section); $appSettings = $this->getSettings('admin', $section); foreach ($appSettings as $setting) { if (!isset($settings[$setting->getPriority()])) { $settings[$setting->getPriority()] = []; } $settings[$setting->getPriority()][] = $setting; } ksort($settings); return $settings; } /** * @inheritdoc */ public function getPersonalSections(): array { $sections = [ 0 => [new Section('personal-info', $this->l->t('Personal info'), 0, $this->url->imagePath('core', 'actions/info.svg'))], 5 => [new Section('security', $this->l->t('Security'), 0, $this->url->imagePath('settings', 'password.svg'))], 15 => [new Section('sync-clients', $this->l->t('Mobile & desktop'), 0, $this->url->imagePath('core', 'clients/phone.svg'))], ]; $legacyForms = \OC_App::getForms('personal'); if (!empty($legacyForms) && $this->hasLegacyPersonalSettingsToRender($legacyForms)) { $sections[98] = [new Section('additional', $this->l->t('Additional settings'), 0, $this->url->imagePath('core', 'actions/settings-dark.svg'))]; } $appSections = $this->getSections('personal'); foreach ($appSections as $section) { /** @var ISection $section */ if (!isset($sections[$section->getPriority()])) { $sections[$section->getPriority()] = []; } $sections[$section->getPriority()][] = $section; } ksort($sections); return $sections; } /** * @param string[] $forms * * @return bool */ private function hasLegacyPersonalSettingsToRender(array $forms): bool { foreach ($forms as $form) { if (trim($form) !== '') { return true; } } return false; } /** * @inheritdoc */ public function getPersonalSettings($section): array { $settings = $this->getBuiltInPersonalSettings($section); $appSettings = $this->getSettings('personal', $section); foreach ($appSettings as $setting) { if (!isset($settings[$setting->getPriority()])) { $settings[$setting->getPriority()] = []; } $settings[$setting->getPriority()][] = $setting; } ksort($settings); return $settings; } }