Compare commits
68 commits
0cb14258ad
...
4d453676dc
| Author | SHA1 | Date | |
|---|---|---|---|
|
4d453676dc |
|||
| 5cd106640c | |||
|
d057cd7665 |
|||
|
520225603b |
|||
|
c2ecc5bf98 |
|||
|
8cca320a43 |
|||
| 19aaf0759b | |||
|
47a83f10cc |
|||
| e1a191121d | |||
|
54f4dce09b |
|||
| c0aaa206ed | |||
|
0124336558 |
|||
|
5ccb56309a |
|||
|
be68f2ef55 |
|||
| 1bf9cee8d6 | |||
|
403a7a72fb |
|||
|
81ae76cdb7 |
|||
|
32e5ecda8f |
|||
|
de98967da5 |
|||
|
f39f3b3aef |
|||
|
ce39df6c2d |
|||
|
bc94b61c12 |
|||
|
c54c969824 |
|||
| e828a914cf | |||
| 9ad73c7bf7 | |||
| acde24ea71 | |||
| 8bd89b4565 | |||
| 58cfbc24af | |||
|
d9051304c8 |
|||
|
|
95d7c1f0c7 |
||
|
|
c6fe0db0b6 |
||
|
|
3fbfd36cba |
||
|
70a3fefb3d |
|||
|
3f16a674e6 |
|||
|
782faf6add |
|||
| 92d15d161b | |||
|
cd4b3b1054 |
|||
|
f58dedf553 |
|||
|
8c6f0ad1da |
|||
| 7485a5b349 | |||
|
b0e01c2eec |
|||
|
d2730afe9f |
|||
|
33ab7dff97 |
|||
| 0da550e3eb | |||
|
4bea4afb07 |
|||
| d485b728e7 | |||
|
8772504b76 |
|||
| 75bea4be38 | |||
| 8787918547 | |||
|
1fcbd89d19 |
|||
|
f177340b13 |
|||
| aaa7afac51 | |||
|
fd4628d209 |
|||
|
f9aa59ca04 |
|||
| 95e4ef136e | |||
|
c0eb65547d |
|||
|
29d061c379 |
|||
| 8e22d9ea62 | |||
|
8957f1ae78 |
|||
|
7b4b447754 |
|||
|
e297a27f83 |
|||
|
3f6ce2df34 |
|||
| 15cc6a129b | |||
|
174c80e654 |
|||
|
17f1d91851 |
|||
|
cdad60ca7a |
|||
|
94efc26cd5 |
|||
|
a47a8d6c63 |
21 changed files with 8338 additions and 85 deletions
4
.gitignore
vendored
4
.gitignore
vendored
|
|
@ -2,8 +2,4 @@
|
||||||
/node_modules
|
/node_modules
|
||||||
/l10n/*
|
/l10n/*
|
||||||
/releases
|
/releases
|
||||||
/package-lock.json
|
|
||||||
!/l10n/.gitkeep
|
!/l10n/.gitkeep
|
||||||
/yarn*.log
|
|
||||||
/src/admin.js.bk
|
|
||||||
/templates/settings/admin-form.php.bk
|
|
||||||
|
|
|
||||||
|
|
@ -32,7 +32,7 @@ steps:
|
||||||
- echo "$APP_CERTIFICATE" > "/tmp/side_menu.key"
|
- echo "$APP_CERTIFICATE" > "/tmp/side_menu.key"
|
||||||
- echo "$APP_PUBLIC_CERTIFICATE" > "/tmp/side_menu.crt"
|
- echo "$APP_PUBLIC_CERTIFICATE" > "/tmp/side_menu.crt"
|
||||||
- mkdir /tmp/app
|
- mkdir /tmp/app
|
||||||
- cp -r README.md CHANGELOG.md appinfo css lib img l10n js src templates screenshots vendor /tmp/app
|
- cp -r README.md CHANGELOG.md appinfo lib img l10n js src templates screenshots vendor /tmp/app
|
||||||
- /usr/src/nextcloud/occ integrity:sign-app
|
- /usr/src/nextcloud/occ integrity:sign-app
|
||||||
--privateKey=/tmp/side_menu.key
|
--privateKey=/tmp/side_menu.key
|
||||||
--certificate=/tmp/side_menu.crt
|
--certificate=/tmp/side_menu.crt
|
||||||
|
|
|
||||||
67
CHANGELOG.md
67
CHANGELOG.md
|
|
@ -1,5 +1,70 @@
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
## 5.2.1
|
||||||
|
### Added
|
||||||
|
- chore: set side_menu as package name
|
||||||
|
### Fixed
|
||||||
|
- fix(LangRepository): check orm capabilities to query entities
|
||||||
|
- fix(admin/\*SaveButton): cast settings to string
|
||||||
|
|
||||||
|
## 5.2.0
|
||||||
|
### Added
|
||||||
|
* add compatibility with NC33
|
||||||
|
### Fixed
|
||||||
|
* fix #468: force nextcloud logo display css rule (opener-only)
|
||||||
|
|
||||||
|
## 5.1.3
|
||||||
|
### Fixed
|
||||||
|
- fix #445: fix build by adding package-lock.json
|
||||||
|
|
||||||
|
## 5.1.2
|
||||||
|
### Added
|
||||||
|
* add new translations
|
||||||
|
### Fixed
|
||||||
|
* fix #441: Side bar not working with Nextcloud 32 (thanks to AndyXheli)
|
||||||
|
|
||||||
|
## 5.1.1
|
||||||
|
### Fixed
|
||||||
|
* fix(build): define appName to fix this error: "The `@nextcloud/vue` library was used without setting / replacing the `appName`"
|
||||||
|
* fix #349: add custom controller to retrieve core apps
|
||||||
|
|
||||||
|
## 5.1.0
|
||||||
|
### Added
|
||||||
|
* fix #425: allow to set a color using hex code
|
||||||
|
### Fixed
|
||||||
|
* #422: usage of `OC\AppFramework\Http\Request` instead of `$_SERVER`
|
||||||
|
|
||||||
|
## 5.0.3
|
||||||
|
### Fixed
|
||||||
|
* fix #422: undefined array key "HTTP_USER_AGENT"
|
||||||
|
|
||||||
|
## 5.0.2
|
||||||
|
### Fixed
|
||||||
|
* fix #413: add user-agent check for memories mobile app
|
||||||
|
* fix #418: allow non admin user to access their settings
|
||||||
|
|
||||||
|
## 5.0.1
|
||||||
|
### Fixed
|
||||||
|
* fix(StandardMenu): appLimit must return a value > 0
|
||||||
|
|
||||||
|
## 5.0.0
|
||||||
|
### Fixed
|
||||||
|
* fix apps's order in the standard menu
|
||||||
|
### Added
|
||||||
|
* add new translations
|
||||||
|
* add route `/apps/side_menu/user/config`
|
||||||
|
* add new UI for admin and personals settings
|
||||||
|
### Changed
|
||||||
|
* migrate to Vue 3 and so add/update or remove dependencies
|
||||||
|
* replace CSS with SCSS
|
||||||
|
* remove route `/apps/side_menu/js/script`
|
||||||
|
* remove generated Javascript using PHP
|
||||||
|
* rewrite the standard menu of Nextcloud
|
||||||
|
### Security
|
||||||
|
* fix CVE-2023-44270
|
||||||
|
* fix CVE-2024-9506
|
||||||
|
* fix CVE-2024-6783
|
||||||
|
|
||||||
## 4.1.1
|
## 4.1.1
|
||||||
### Fixed
|
### Fixed
|
||||||
* fix(CssController): add missing NoCSRFRequired import (#397)
|
* fix(CssController): add missing NoCSRFRequired import (#397)
|
||||||
|
|
@ -155,7 +220,7 @@
|
||||||
|
|
||||||
## 3.5.1
|
## 3.5.1
|
||||||
### Added
|
### Added
|
||||||
* add translations (thanks to p-bo adn gallegonovato)
|
* add translations (thanks to p-bo and gallegonovato)
|
||||||
### Fixed
|
### Fixed
|
||||||
* fix #189: sorting not applied on mobile
|
* fix #189: sorting not applied on mobile
|
||||||
|
|
||||||
|
|
|
||||||
2
Makefile
2
Makefile
|
|
@ -20,7 +20,7 @@ release:
|
||||||
|
|
||||||
test -d $$RELEASE_DIRECTORY/$$VERSION && rm -fr $$RELEASE_DIRECTORY/$$VERSION
|
test -d $$RELEASE_DIRECTORY/$$VERSION && rm -fr $$RELEASE_DIRECTORY/$$VERSION
|
||||||
mkdir -p $$RELEASE_DIRECTORY/$$VERSION/side_menu
|
mkdir -p $$RELEASE_DIRECTORY/$$VERSION/side_menu
|
||||||
cp -r README.md CHANGELOG.md appinfo css lib img l10n js src templates screenshots vendor $$RELEASE_DIRECTORY/$$VERSION/side_menu
|
cp -r README.md CHANGELOG.md appinfo lib img l10n js src templates screenshots vendor $$RELEASE_DIRECTORY/$$VERSION/side_menu
|
||||||
cd $$RELEASE_DIRECTORY/$$VERSION
|
cd $$RELEASE_DIRECTORY/$$VERSION
|
||||||
zip -r side_menu_v$$VERSION.zip side_menu
|
zip -r side_menu_v$$VERSION.zip side_menu
|
||||||
tar cvzf side_menu_v$$VERSION.tar.gz side_menu
|
tar cvzf side_menu_v$$VERSION.tar.gz side_menu
|
||||||
|
|
|
||||||
|
|
@ -24,8 +24,7 @@ You like this app and you want to support me? ☕ [Buy me a coffee](https://www.
|
||||||
Requirements
|
Requirements
|
||||||
------------
|
------------
|
||||||
|
|
||||||
* PHP >= 8.0
|
* PHP >= 8.1
|
||||||
* App `theming` enabled
|
|
||||||
|
|
||||||
Installation and upgrade
|
Installation and upgrade
|
||||||
------------------------
|
------------------------
|
||||||
|
|
@ -41,7 +40,7 @@ If you want to install it from source, go to https://gitnet.fr/deblan/side_menu/
|
||||||
|
|
||||||
```
|
```
|
||||||
$ cd /path/to/nextcloud/apps
|
$ cd /path/to/nextcloud/apps
|
||||||
$ curl -sS https://gitnet.fr/attachments/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx | tar xvfz -
|
$ VERSION=x.y.z; curl -sS "https://gitnet.fr/deblan/side_menu/releases/download/${VERSION}/side_menu_v${VERSION}.tar.gz" | tar xvfz -
|
||||||
```
|
```
|
||||||
|
|
||||||
Administrators can edit many settings using the administration page.
|
Administrators can edit many settings using the administration page.
|
||||||
|
|
|
||||||
|
|
@ -10,14 +10,13 @@ This application is rather suitable for instances that activate a lot of applica
|
||||||
|
|
||||||
Use the shortcut `Ctrl`+`o` to open and to hide the side menu. Use `tab` to navigate.
|
Use the shortcut `Ctrl`+`o` to open and to hide the side menu. Use `tab` to navigate.
|
||||||
|
|
||||||
You can customize colors depending of the theme (Dark theme and Breeze Dark).
|
You can customize colors depending of the theme.
|
||||||
|
|
||||||
You can report a bug or request a feature by opening an issue.
|
To report a bug or request a feature, please open an issue.
|
||||||
|
|
||||||
Requirements:
|
Requirements:
|
||||||
|
|
||||||
* PHP >= 8.1
|
* PHP >= 8.1
|
||||||
* App `theming` enabled
|
|
||||||
|
|
||||||
If you like this application and if you want to support the development:
|
If you like this application and if you want to support the development:
|
||||||
|
|
||||||
|
|
@ -31,7 +30,7 @@ Notice
|
||||||
Because I believe in a free and decentralized Internet, [Gitnet](https://gitnet.fr) is **self-hosted at home**.
|
Because I believe in a free and decentralized Internet, [Gitnet](https://gitnet.fr) is **self-hosted at home**.
|
||||||
In case of downtime, you can download **Custom Menu** from [here](https://kim.deblan.fr/~side_menu/).
|
In case of downtime, you can download **Custom Menu** from [here](https://kim.deblan.fr/~side_menu/).
|
||||||
]]></description>
|
]]></description>
|
||||||
<version>4.1.1</version>
|
<version>5.2.1</version>
|
||||||
<licence>agpl</licence>
|
<licence>agpl</licence>
|
||||||
<author mail="contact@deblan.fr" homepage="https://www.deblan.fr/">Simon Vieille</author>
|
<author mail="contact@deblan.fr" homepage="https://www.deblan.fr/">Simon Vieille</author>
|
||||||
<namespace>SideMenu</namespace>
|
<namespace>SideMenu</namespace>
|
||||||
|
|
@ -54,7 +53,7 @@ In case of downtime, you can download **Custom Menu** from [here](https://kim.de
|
||||||
<screenshot><![CDATA[https://gitnet.fr/deblan/side_menu/raw/branch/master/screenshots/nc25_default_menu.png]]></screenshot>
|
<screenshot><![CDATA[https://gitnet.fr/deblan/side_menu/raw/branch/master/screenshots/nc25_default_menu.png]]></screenshot>
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<php min-version="8.1" max-version="8.4" />
|
<php min-version="8.1" max-version="8.4" />
|
||||||
<nextcloud min-version="30" max-version="32"/>
|
<nextcloud min-version="31" max-version="33"/>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
<settings>
|
<settings>
|
||||||
<admin>OCA\SideMenu\Settings\Admin</admin>
|
<admin>OCA\SideMenu\Settings\Admin</admin>
|
||||||
|
|
|
||||||
|
|
@ -2,8 +2,9 @@
|
||||||
|
|
||||||
namespace OCA\SideMenu\AppInfo;
|
namespace OCA\SideMenu\AppInfo;
|
||||||
|
|
||||||
use OC;
|
use OC\AllConfig;
|
||||||
use OC\App\AppStore\Fetcher\CategoryFetcher;
|
use OC\App\AppStore\Fetcher\CategoryFetcher;
|
||||||
|
use OC\AppFramework\Http\Request;
|
||||||
use OC\Security\CSP\ContentSecurityPolicyNonceManager;
|
use OC\Security\CSP\ContentSecurityPolicyNonceManager;
|
||||||
use OC\User\User;
|
use OC\User\User;
|
||||||
use OCA\SideMenu\Service\AppRepository;
|
use OCA\SideMenu\Service\AppRepository;
|
||||||
|
|
@ -31,23 +32,12 @@ use Psr\Container\ContainerInterface;
|
||||||
class Application extends App implements IBootstrap
|
class Application extends App implements IBootstrap
|
||||||
{
|
{
|
||||||
public const APP_ID = 'side_menu';
|
public const APP_ID = 'side_menu';
|
||||||
|
|
||||||
public const APP_NAME = 'Custom menu';
|
public const APP_NAME = 'Custom menu';
|
||||||
|
|
||||||
/**
|
protected AllConfig $config;
|
||||||
* @var OC\AllConfig
|
protected ContentSecurityPolicyNonceManager $cspnm;
|
||||||
*/
|
protected Request $request;
|
||||||
protected $config;
|
protected ?User $user = null;
|
||||||
|
|
||||||
/**
|
|
||||||
* @var ContentSecurityPolicyNonceManager
|
|
||||||
*/
|
|
||||||
protected $cspnm;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @var User
|
|
||||||
*/
|
|
||||||
protected $user;
|
|
||||||
|
|
||||||
public function __construct(array $urlParams = [])
|
public function __construct(array $urlParams = [])
|
||||||
{
|
{
|
||||||
|
|
@ -96,6 +86,7 @@ class Application extends App implements IBootstrap
|
||||||
$this->config = \OC::$server->getConfig();
|
$this->config = \OC::$server->getConfig();
|
||||||
$this->cspnm = \OC::$server->getContentSecurityPolicyNonceManager();
|
$this->cspnm = \OC::$server->getContentSecurityPolicyNonceManager();
|
||||||
$this->user = \OC::$server[IUserSession::class]->getUser();
|
$this->user = \OC::$server[IUserSession::class]->getUser();
|
||||||
|
$this->request = \OC::$server->getRequest();
|
||||||
|
|
||||||
if (!$this->isEnabled()) {
|
if (!$this->isEnabled()) {
|
||||||
return;
|
return;
|
||||||
|
|
@ -106,6 +97,10 @@ class Application extends App implements IBootstrap
|
||||||
|
|
||||||
protected function isEnabled(): bool
|
protected function isEnabled(): bool
|
||||||
{
|
{
|
||||||
|
if (isset($this->request->server['HTTP_USER_AGENT']) && preg_match('/MemoriesNative/', $this->request->server['HTTP_USER_AGENT'])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
$enabled = true;
|
$enabled = true;
|
||||||
$isForced = (bool) $this->config->getAppValue(self::APP_ID, 'force', '0');
|
$isForced = (bool) $this->config->getAppValue(self::APP_ID, 'force', '0');
|
||||||
|
|
||||||
|
|
@ -128,7 +123,6 @@ class Application extends App implements IBootstrap
|
||||||
protected function addAssets()
|
protected function addAssets()
|
||||||
{
|
{
|
||||||
Util::addScript(self::APP_ID, 'side_menu-menu');
|
Util::addScript(self::APP_ID, 'side_menu-menu');
|
||||||
// Util::addStyle(self::APP_ID, 'sideMenu');
|
|
||||||
|
|
||||||
$assets = [
|
$assets = [
|
||||||
'stylesheet' => [
|
'stylesheet' => [
|
||||||
|
|
@ -139,14 +133,6 @@ class Application extends App implements IBootstrap
|
||||||
'rel' => 'stylesheet',
|
'rel' => 'stylesheet',
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
// 'script' => [
|
|
||||||
// 'route' => 'side_menu.Js.script',
|
|
||||||
// 'type' => 'script',
|
|
||||||
// 'route_attr' => 'src',
|
|
||||||
// 'attr' => [
|
|
||||||
// 'nonce' => $this->cspnm->getNonce(),
|
|
||||||
// ],
|
|
||||||
// ],
|
|
||||||
];
|
];
|
||||||
|
|
||||||
$cache = $this->config->getAppValue(self::APP_ID, 'cache', '0');
|
$cache = $this->config->getAppValue(self::APP_ID, 'cache', '0');
|
||||||
|
|
|
||||||
74
lib/Controller/CoreController.php
Normal file
74
lib/Controller/CoreController.php
Normal file
|
|
@ -0,0 +1,74 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @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 <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\SideMenu\Controller;
|
||||||
|
|
||||||
|
use OCA\SideMenu\Service\AppRepository;
|
||||||
|
use OCA\SideMenu\Service\ConfigProxy;
|
||||||
|
use OCP\AppFramework\Controller;
|
||||||
|
use OCP\AppFramework\Http\Attribute\FrontpageRoute;
|
||||||
|
use OCP\AppFramework\Http\Attribute\NoAdminRequired;
|
||||||
|
use OCP\AppFramework\Http\Attribute\NoCSRFRequired;
|
||||||
|
use OCP\AppFramework\Http\Attribute\PublicPage;
|
||||||
|
use OCP\AppFramework\Http\JSONResponse;
|
||||||
|
use OCP\IRequest;
|
||||||
|
use OCP\IUserSession;
|
||||||
|
|
||||||
|
class CoreController extends Controller
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
string $appName,
|
||||||
|
IRequest $request,
|
||||||
|
protected ConfigProxy $config,
|
||||||
|
protected AppRepository $appRepository,
|
||||||
|
) {
|
||||||
|
parent::__construct($appName, $request);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[NoCSRFRequired]
|
||||||
|
#[NoAdminRequired]
|
||||||
|
#[PublicPage]
|
||||||
|
#[FrontpageRoute(verb: 'GET', url: '/core/apps')]
|
||||||
|
public function items(): JSONResponse
|
||||||
|
{
|
||||||
|
$user = \OC::$server[IUserSession::class]->getUser();
|
||||||
|
$items = [];
|
||||||
|
|
||||||
|
if (!$user) {
|
||||||
|
return new JSONResponse([
|
||||||
|
'items' => $items,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$apps = $this->appRepository->getOrderedApps($user);
|
||||||
|
$keys = ['id', 'name', 'category', 'href', 'icon'];
|
||||||
|
|
||||||
|
foreach ($apps as &$app) {
|
||||||
|
foreach ($app as $key => $value) {
|
||||||
|
if (!in_array($key, $keys)) {
|
||||||
|
unset($app[$key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return new JSONResponse([
|
||||||
|
'items' => $apps,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -20,7 +20,6 @@
|
||||||
namespace OCA\SideMenu\Controller;
|
namespace OCA\SideMenu\Controller;
|
||||||
|
|
||||||
use OC\User\User;
|
use OC\User\User;
|
||||||
use OCA\SideMenu\AppInfo\Application;
|
|
||||||
use OCA\SideMenu\Service\ConfigProxy;
|
use OCA\SideMenu\Service\ConfigProxy;
|
||||||
use OCA\Theming\ThemingDefaults;
|
use OCA\Theming\ThemingDefaults;
|
||||||
use OCP\AppFramework\Controller;
|
use OCP\AppFramework\Controller;
|
||||||
|
|
@ -29,10 +28,12 @@ use OCP\AppFramework\Http\Attribute\NoAdminRequired;
|
||||||
use OCP\AppFramework\Http\Attribute\NoCSRFRequired;
|
use OCP\AppFramework\Http\Attribute\NoCSRFRequired;
|
||||||
use OCP\AppFramework\Http\Attribute\PublicPage;
|
use OCP\AppFramework\Http\Attribute\PublicPage;
|
||||||
use OCP\AppFramework\Http\JSONResponse;
|
use OCP\AppFramework\Http\JSONResponse;
|
||||||
use OCP\AppFramework\Http\TemplateResponse;
|
|
||||||
use OCP\IRequest;
|
use OCP\IRequest;
|
||||||
use OCP\IUserSession;
|
use OCP\IUserSession;
|
||||||
use OCP\L10N\IFactory;
|
use OCP\L10N\IFactory;
|
||||||
|
use OCP\IAvatarManager;
|
||||||
|
use OCP\INavigationManager;
|
||||||
|
use OCP\IURLGenerator;
|
||||||
|
|
||||||
class JsController extends Controller
|
class JsController extends Controller
|
||||||
{
|
{
|
||||||
|
|
@ -44,26 +45,13 @@ class JsController extends Controller
|
||||||
protected ConfigProxy $config,
|
protected ConfigProxy $config,
|
||||||
protected ThemingDefaults $themingDefaults,
|
protected ThemingDefaults $themingDefaults,
|
||||||
protected IFactory $l10nFactory,
|
protected IFactory $l10nFactory,
|
||||||
|
protected IAvatarManager $avatarManager,
|
||||||
|
protected IUserSession $userSession,
|
||||||
|
protected INavigationManager $navigationManager,
|
||||||
|
protected IURLGenerator $urlGenerator,
|
||||||
) {
|
) {
|
||||||
parent::__construct($appName, $request);
|
parent::__construct($appName, $request);
|
||||||
|
$this->user = $this->userSession->getUser();
|
||||||
$this->themingDefaults = $themingDefaults;
|
|
||||||
|
|
||||||
$this->user = \OC::$server[IUserSession::class]->getUser();
|
|
||||||
$this->config = $config;
|
|
||||||
$this->l10nFactory = $l10nFactory;
|
|
||||||
}
|
|
||||||
|
|
||||||
#[NoCSRFRequired]
|
|
||||||
#[NoAdminRequired]
|
|
||||||
#[PublicPage]
|
|
||||||
#[FrontpageRoute(verb: 'GET', url: '/js/script')]
|
|
||||||
public function script(): TemplateResponse
|
|
||||||
{
|
|
||||||
$response = new TemplateResponse(Application::APP_ID, 'js/script', $this->getConfig(), 'blank');
|
|
||||||
$response->addHeader('Content-Type', 'text/javascript');
|
|
||||||
|
|
||||||
return $response;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[NoCSRFRequired]
|
#[NoCSRFRequired]
|
||||||
|
|
@ -111,25 +99,25 @@ class JsController extends Controller
|
||||||
$targetBlankApps = $userTargetBlankApps;
|
$targetBlankApps = $userTargetBlankApps;
|
||||||
}
|
}
|
||||||
|
|
||||||
$isAvatarSet = \OC::$server->getAvatarManager()->getAvatar($this->user->getUid())->exists();
|
$isAvatarSet = $this->avatarManager->getAvatar($this->user->getUID())->exists();
|
||||||
|
|
||||||
if ($useAvatar && $isAvatarSet) {
|
if ($useAvatar && $isAvatarSet) {
|
||||||
$avatar = \OC::$server->getURLGenerator()->linkToRoute('core.avatar.getAvatar', [
|
$avatar = $this->urlGenerator->linkToRoute('core.avatar.getAvatar', [
|
||||||
'userId' => $this->user->getUid(),
|
'userId' => $this->user->getUID(),
|
||||||
'size' => 128,
|
'size' => 128,
|
||||||
'v' => $this->config->getUserValueInt($this->user, 'avatar', 'version', 0),
|
'v' => $this->config->getUserValueInt($this->user, 'avatar', 'version', 0),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->config->getAppValueBool('show-settings', '0')) {
|
if ($this->config->getAppValueBool('show-settings', '0')) {
|
||||||
$settingsNav = \OC::$server->getNavigationManager()->getAll('settings');
|
$settingsNav = $this->navigationManager->getAll('settings');
|
||||||
|
|
||||||
if (isset($settingsNav['settings'])) {
|
if (isset($settingsNav['settings'])) {
|
||||||
$settings = [
|
$settings = [
|
||||||
'href' => $settingsNav['settings']['href'],
|
'href' => $settingsNav['settings']['href'],
|
||||||
'name' => $settingsNav['settings']['name'],
|
'name' => $settingsNav['settings']['name'],
|
||||||
'avatar' => \OC::$server->getURLGenerator()->linkToRoute('core.avatar.getAvatar', [
|
'avatar' => $this->urlGenerator->linkToRoute('core.avatar.getAvatar', [
|
||||||
'userId' => $this->user->getUid(),
|
'userId' => $this->user->getUID(),
|
||||||
'size' => 32,
|
'size' => 32,
|
||||||
'v' => $this->config->getUserValueInt($this->user, 'avatar', 'version', 0),
|
'v' => $this->config->getUserValueInt($this->user, 'avatar', 'version', 0),
|
||||||
]),
|
]),
|
||||||
|
|
@ -138,7 +126,7 @@ class JsController extends Controller
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$indexUrl = \OC::$server->getURLGenerator()->linkTo('', 'index.php');
|
$indexUrl = $this->urlGenerator->linkTo('', 'index.php');
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'opener-position' => $this->config->getAppValue('opener-position', 'before'),
|
'opener-position' => $this->config->getAppValue('opener-position', 'before'),
|
||||||
|
|
|
||||||
|
|
@ -98,6 +98,7 @@ class PersonalSettingController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
#[NoCSRFRequired]
|
#[NoCSRFRequired]
|
||||||
|
#[NoAdminRequired]
|
||||||
#[FrontpageRoute(verb: 'GET', url: '/user/config')]
|
#[FrontpageRoute(verb: 'GET', url: '/user/config')]
|
||||||
public function configuration(): JSONResponse
|
public function configuration(): JSONResponse
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -30,7 +30,12 @@ class LangRepository
|
||||||
->from('preferences')
|
->from('preferences')
|
||||||
;
|
;
|
||||||
|
|
||||||
$stmt = $qb->execute();
|
// Nextcloud >=33+
|
||||||
|
if (method_exists($qb, 'executeQuery')) {
|
||||||
|
$stmt = $qb->executeQuery();
|
||||||
|
} else {
|
||||||
|
$stmt = $qb->execute();
|
||||||
|
}
|
||||||
|
|
||||||
$langs = ['en'];
|
$langs = ['en'];
|
||||||
|
|
||||||
|
|
|
||||||
8133
package-lock.json
generated
Normal file
8133
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
Binary file not shown.
|
Before Width: | Height: | Size: 380 KiB After Width: | Height: | Size: 223 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 79 KiB After Width: | Height: | Size: 246 KiB |
|
|
@ -73,10 +73,10 @@ const save = async () => {
|
||||||
if (Array.isArray(value) || typeof value === 'object') {
|
if (Array.isArray(value) || typeof value === 'object') {
|
||||||
value = JSON.stringify(value)
|
value = JSON.stringify(value)
|
||||||
} else if (typeof value === 'boolean') {
|
} else if (typeof value === 'boolean') {
|
||||||
value = value ? 1 : 0
|
value = value ? '1' : '0'
|
||||||
}
|
}
|
||||||
|
|
||||||
OCP.AppConfig.setValue('side_menu', key, value, {
|
OCP.AppConfig.setValue('side_menu', key, value.toString(), {
|
||||||
success() {
|
success() {
|
||||||
update()
|
update()
|
||||||
},
|
},
|
||||||
|
|
|
||||||
|
|
@ -69,11 +69,11 @@ const save = async () => {
|
||||||
if (Array.isArray(value) || typeof value === 'object') {
|
if (Array.isArray(value) || typeof value === 'object') {
|
||||||
value = JSON.stringify(value)
|
value = JSON.stringify(value)
|
||||||
} else if (typeof value === 'boolean') {
|
} else if (typeof value === 'boolean') {
|
||||||
value = value ? 1 : 0
|
value = value ? '1' : '0'
|
||||||
}
|
}
|
||||||
|
|
||||||
formData.push('name=' + encodeURIComponent(key))
|
formData.push('name=' + encodeURIComponent(key))
|
||||||
formData.push('value=' + encodeURIComponent(value))
|
formData.push('value=' + encodeURIComponent(value.toString()))
|
||||||
|
|
||||||
fetch(url, {
|
fetch(url, {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
|
|
|
||||||
|
|
@ -17,6 +17,7 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
<template>
|
<template>
|
||||||
<NcColorPicker
|
<NcColorPicker
|
||||||
v-model="model"
|
v-model="model"
|
||||||
|
:advancedFields="true"
|
||||||
class="cm-settings-form-colorpicker"
|
class="cm-settings-form-colorpicker"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
|
|
|
||||||
|
|
@ -33,7 +33,7 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
:data-app-id="app.id"
|
:data-app-id="app.id"
|
||||||
class="app-menu-entry"
|
class="app-menu-entry"
|
||||||
:class="{
|
:class="{
|
||||||
'app-menu-entry__active': app.active,
|
'app-menu-entry__active': app.id === activeApp,
|
||||||
'app-menu-entry__hidden-label': hiddenLabels === 1,
|
'app-menu-entry__hidden-label': hiddenLabels === 1,
|
||||||
'app-menu-main__show-hovered': hiddenLabels === 2,
|
'app-menu-main__show-hovered': hiddenLabels === 2,
|
||||||
}"
|
}"
|
||||||
|
|
@ -44,7 +44,7 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
:class="{ 'has-unread': app.unread > 0 }"
|
:class="{ 'has-unread': app.unread > 0 }"
|
||||||
:aria-label="app.name"
|
:aria-label="app.name"
|
||||||
:target="targetBlankApps.indexOf(app.id) !== -1 ? '_blank' : undefined"
|
:target="targetBlankApps.indexOf(app.id) !== -1 ? '_blank' : undefined"
|
||||||
:aria-current="app.active ? 'page' : false"
|
:aria-current="app.id === activeApp ? 'page' : false"
|
||||||
>
|
>
|
||||||
<img
|
<img
|
||||||
:src="app.icon"
|
:src="app.icon"
|
||||||
|
|
@ -69,7 +69,7 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
v-for="app in popoverAppList"
|
v-for="app in popoverAppList"
|
||||||
:key="app.id"
|
:key="app.id"
|
||||||
:aria-label="app.name"
|
:aria-label="app.name"
|
||||||
:aria-current="app.active ? 'page' : false"
|
:aria-current="app.id === activeApp ? 'page' : false"
|
||||||
:href="app.href"
|
:href="app.href"
|
||||||
:style="makeStyle(app)"
|
:style="makeStyle(app)"
|
||||||
class="cm-standardmenu-app-menu-popover-entry app-menu-popover-entry"
|
class="cm-standardmenu-app-menu-popover-entry app-menu-popover-entry"
|
||||||
|
|
@ -101,6 +101,7 @@ import { ref, onMounted } from 'vue'
|
||||||
import { useConfigStore } from '../store/config.js'
|
import { useConfigStore } from '../store/config.js'
|
||||||
import { useNavStore } from '../store/nav.js'
|
import { useNavStore } from '../store/nav.js'
|
||||||
import { NcActions, NcActionLink } from '@nextcloud/vue'
|
import { NcActions, NcActionLink } from '@nextcloud/vue'
|
||||||
|
import { getActiveAppId } from '../lib/app.js'
|
||||||
|
|
||||||
const navStore = useNavStore()
|
const navStore = useNavStore()
|
||||||
const configStore = useConfigStore()
|
const configStore = useConfigStore()
|
||||||
|
|
@ -112,6 +113,7 @@ const topMenuApps = ref([])
|
||||||
const appsOrder = ref([])
|
const appsOrder = ref([])
|
||||||
const mainAppList = ref([])
|
const mainAppList = ref([])
|
||||||
const popoverAppList = ref([])
|
const popoverAppList = ref([])
|
||||||
|
const activeApp = ref(null)
|
||||||
let resizeTimeout = null
|
let resizeTimeout = null
|
||||||
|
|
||||||
const setApps = (value) => {
|
const setApps = (value) => {
|
||||||
|
|
@ -142,7 +144,7 @@ const appLimit = () => {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
return Math.floor((body.offsetWidth - size) / 70)
|
return Math.max(0, Math.floor((body.offsetWidth - size) / 70))
|
||||||
}
|
}
|
||||||
|
|
||||||
const makeStyle = (app) => {
|
const makeStyle = (app) => {
|
||||||
|
|
@ -158,6 +160,11 @@ const computeLists = () => {
|
||||||
popoverAppList.value = appList.value.slice(appLimit()).sort((a, b) => a.order - b.order)
|
popoverAppList.value = appList.value.slice(appLimit()).sort((a, b) => a.order - b.order)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const reComputeLists = (delay) => {
|
||||||
|
window.clearTimeout(resizeTimeout)
|
||||||
|
resizeTimeout = window.setTimeout(computeLists, delay || 100)
|
||||||
|
}
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
const config = await configStore.getConfig()
|
const config = await configStore.getConfig()
|
||||||
|
|
||||||
|
|
@ -165,14 +172,12 @@ onMounted(async () => {
|
||||||
hiddenLabels.value = config['top-menu-mouse-over-hidden-label']
|
hiddenLabels.value = config['top-menu-mouse-over-hidden-label']
|
||||||
topMenuApps.value = config['top-menu-apps']
|
topMenuApps.value = config['top-menu-apps']
|
||||||
appsOrder.value = config['apps-order']
|
appsOrder.value = config['apps-order']
|
||||||
|
activeApp.value = getActiveAppId()
|
||||||
ready.value = true
|
ready.value = true
|
||||||
|
|
||||||
setApps(await navStore.getCoreApps())
|
setApps(await navStore.getCoreApps())
|
||||||
|
|
||||||
window.addEventListener('resize', () => {
|
window.addEventListener('resize', reComputeLists)
|
||||||
window.clearTimeout(resizeTimeout)
|
|
||||||
resizeTimeout = window.setTimeout(computeLists, 100)
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -41,10 +41,7 @@ export const useNavStore = defineStore('nav', () => {
|
||||||
|
|
||||||
async function getCoreApps() {
|
async function getCoreApps() {
|
||||||
if (coreApps == null) {
|
if (coreApps == null) {
|
||||||
coreApps = await axios
|
coreApps = await await axios.get(generateUrl('/apps/side_menu/core/apps')).then((response) => response.data.items)
|
||||||
.get(generateOcsUrl('core/navigation', 2) + '/apps?format=json')
|
|
||||||
.then((response) => response.data)
|
|
||||||
.then((value) => value.ocs.data)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return coreApps
|
return coreApps
|
||||||
|
|
|
||||||
|
|
@ -38,7 +38,7 @@ body[data-theme-light], body[data-theme-light-highcontrast] {
|
||||||
|
|
||||||
<?php if ($_['opener-only']) { ?>
|
<?php if ($_['opener-only']) { ?>
|
||||||
#nextcloud {
|
#nextcloud {
|
||||||
display: none;
|
display: none !important;
|
||||||
}
|
}
|
||||||
<?php } ?>
|
<?php } ?>
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -77,6 +77,10 @@ module.exports = {
|
||||||
new webpack.ProvidePlugin({
|
new webpack.ProvidePlugin({
|
||||||
Buffer: ['buffer', 'Buffer'],
|
Buffer: ['buffer', 'Buffer'],
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
new webpack.DefinePlugin({
|
||||||
|
appName: JSON.stringify(appName),
|
||||||
|
}),
|
||||||
],
|
],
|
||||||
|
|
||||||
resolve: {
|
resolve: {
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue