refs #304 add unified search provider for projects

Signed-off-by: Julien Veyssier <julien-nc@posteo.net>
This commit is contained in:
Julien Veyssier
2024-10-27 17:38:05 +01:00
parent b24d720f59
commit 29a0db7433
5 changed files with 226 additions and 16 deletions

View File

@@ -14,18 +14,21 @@ namespace OCA\Cospend\AppInfo;
use OCA\Cospend\Capabilities;
use OCA\Cospend\Dashboard\CospendWidget;
use OCA\Cospend\Federation\CloudFederationProviderCospend;
use OCA\Cospend\Listener\BeforeTemplateRenderedListener;
use OCA\Cospend\Listener\CSPListener;
use OCA\Cospend\Middleware\FederationMiddleware;
use OCA\Cospend\Middleware\PublicAuthMiddleware;
use OCA\Cospend\Middleware\UserPermissionMiddleware;
use OCA\Cospend\Notification\Notifier;
use OCA\Cospend\Search\CospendSearchProvider;
use OCA\Cospend\Search\CospendSearchBillProvider;
use OCA\Cospend\Search\CospendSearchProjectProvider;
use OCA\Cospend\UserMigration\UserMigrator;
use OCP\AppFramework\App;
use OCP\AppFramework\Bootstrap\IBootContext;
use OCP\AppFramework\Bootstrap\IBootstrap;
use OCP\AppFramework\Bootstrap\IRegistrationContext;
use OCP\AppFramework\Http\Events\BeforeTemplateRenderedEvent;
use OCP\Federation\ICloudFederationProvider;
use OCP\Federation\ICloudFederationProviderManager;
use OCP\IConfig;
@@ -102,14 +105,13 @@ class Application extends App implements IBootstrap {
public function __construct(array $urlParams = []) {
parent::__construct(self::APP_ID, $urlParams);
// TODO
// - rename db columns with underscores, change new APIs param names, keep a second jsonSerialize method for old APIs
// - check if it makes sense to have a paypal integration
// - check how to switch to numerical project IDs (keep unique slug for client compatibility)
}
public function register(IRegistrationContext $context): void {
$context->registerNotifierService(Notifier::class);
$context->registerSearchProvider(CospendSearchProvider::class);
$context->registerSearchProvider(CospendSearchBillProvider::class);
$context->registerSearchProvider(CospendSearchProjectProvider::class);
$context->registerDashboardWidget(CospendWidget::class);
$context->registerUserMigrator(UserMigrator::class);
@@ -120,10 +122,10 @@ class Application extends App implements IBootstrap {
$context->registerCapability(Capabilities::class);
$context->registerEventListener(AddContentSecurityPolicyEvent::class, CSPListener::class);
$context->registerEventListener(BeforeTemplateRenderedEvent::class, BeforeTemplateRenderedListener::class);
}
public function boot(IBootContext $context): void {
Util::addStyle(self::APP_ID, 'cospend-search');
$context->injectFn([$this, 'registerCloudFederationProviderManager']);
}

View File

@@ -0,0 +1,42 @@
<?php
declare(strict_types=1);
namespace OCA\Cospend\Listener;
use OCA\Cospend\AppInfo\Application;
use OCP\AppFramework\Http\Events\BeforeTemplateRenderedEvent;
use OCP\AppFramework\Http\TemplateResponse;
use OCP\EventDispatcher\Event;
use OCP\EventDispatcher\IEventListener;
use OCP\IUser;
use OCP\IUserSession;
use OCP\Util;
/**
* @template-implements IEventListener<Event>
*/
class BeforeTemplateRenderedListener implements IEventListener {
public function __construct(
private IUserSession $userSession,
) {
}
public function handle(Event $event): void {
if (!($event instanceof BeforeTemplateRenderedEvent)) {
// Unrelated
return;
}
if ($event->getResponse()->getRenderAs() !== TemplateResponse::RENDER_AS_USER) {
return;
}
if (!$this->userSession->getUser() instanceof IUser) {
return;
}
Util::addStyle(Application::APP_ID, 'cospend-search');
}
}

View File

@@ -39,7 +39,7 @@ use OCP\Search\ISearchQuery;
use OCP\Search\SearchResult;
use OCP\Search\SearchResultEntry;
class CospendSearchProvider implements IProvider {
class CospendSearchBillProvider implements IProvider {
public function __construct(
private IAppManager $appManager,
@@ -55,21 +55,21 @@ class CospendSearchProvider implements IProvider {
* @inheritDoc
*/
public function getId(): string {
return 'cospend-search';
return 'cospend-search-bills';
}
/**
* @inheritDoc
*/
public function getName(): string {
return $this->l10n->t('Cospend');
return $this->l10n->t('Cospend bills');
}
/**
* @inheritDoc
*/
public function getOrder(string $route, array $routeParameters): int {
if (strpos($route, Application::APP_ID . '.') === 0) {
if (str_starts_with($route, Application::APP_ID . '.')) {
// Active app, prefer Cospend results
return -1;
}
@@ -81,7 +81,7 @@ class CospendSearchProvider implements IProvider {
* @inheritDoc
*/
public function search(IUser $user, ISearchQuery $query): SearchResult {
if (!$this->appManager->isEnabledForUser('cospend', $user)) {
if (!$this->appManager->isEnabledForUser(Application::APP_ID, $user)) {
return SearchResult::complete($this->getName(), []);
}
@@ -181,17 +181,20 @@ class CospendSearchProvider implements IProvider {
/**
* @param string $projectId
* @param int $billId
* @return string
*/
protected function getDeepLinkToCospendApp(string $projectId, int $billId): string {
return $this->urlGenerator->getAbsoluteURL(
$this->urlGenerator->linkToRoute('cospend.page.indexBill', [
'projectId' => $projectId,
'billId' => $billId,
])
);
return $this->urlGenerator->linkToRouteAbsolute('cospend.page.indexBill', [
'projectId' => $projectId,
'billId' => $billId,
]);
}
/**
* @param array $bill
* @return string
*/
protected function getThumbnailUrl(array $bill): string {
if ($bill['payer_user_id']) {
return $this->urlGenerator->linkToRouteAbsolute('core.avatar.getAvatar', ['userId' => $bill['payer_user_id'], 'size' => 44]);

View File

@@ -0,0 +1,160 @@
<?php
declare(strict_types=1);
/**
* @copyright Copyright (c) 2020, Julien Veyssier
*
* @author Julien Veyssier <julien-nc@posteo.net>
*
* @license AGPL-3.0
*
* This code is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License, version 3,
* as published by the Free Software Foundation.
*
* 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, version 3,
* along with this program. If not, see <http://www.gnu.org/licenses/>
*
*/
namespace OCA\Cospend\Search;
use OCA\Cospend\AppInfo\Application;
use OCA\Cospend\Service\LocalProjectService;
use OCP\App\IAppManager;
use OCP\IL10N;
use OCP\IURLGenerator;
use OCP\IUser;
use OCP\IUserManager;
use OCP\Search\IProvider;
use OCP\Search\ISearchQuery;
use OCP\Search\SearchResult;
use OCP\Search\SearchResultEntry;
class CospendSearchProjectProvider implements IProvider {
public function __construct(
private IAppManager $appManager,
private IL10N $l10n,
private IURLGenerator $urlGenerator,
private LocalProjectService $projectService,
private IUserManager $userManager,
) {
}
/**
* @inheritDoc
*/
public function getId(): string {
return 'cospend-search-projects';
}
/**
* @inheritDoc
*/
public function getName(): string {
return $this->l10n->t('Cospend projects');
}
/**
* @inheritDoc
*/
public function getOrder(string $route, array $routeParameters): int {
if (str_starts_with($route, Application::APP_ID . '.')) {
// Active app, prefer Cospend results
return -1;
}
return 20;
}
/**
* @inheritDoc
*/
public function search(IUser $user, ISearchQuery $query): SearchResult {
if (!$this->appManager->isEnabledForUser(Application::APP_ID, $user)) {
return SearchResult::complete($this->getName(), []);
}
$limit = $query->getLimit();
$term = $query->getTerm();
$offset = $query->getCursor();
$offset = $offset ? (int)$offset : 0;
$resultBills = [];
// get user's projects
$projects = $this->projectService->getLocalProjects($user->getUID());
$resultProjects = array_filter($projects, static function(array $project) use ($term) {
$projectName = $project['name'];
return str_contains(strtolower($projectName), strtolower($term));
});
$resultProjects = array_slice($resultProjects, $offset, $limit);
// build formatted
$formattedResults = array_map(function (array $project): SearchResultEntry {
$thumbnailUrl = $this->getThumbnailUrl($project);
return new SearchResultEntry(
$thumbnailUrl,
$this->getMainText($project),
$this->getSubline($project),
$this->getDeepLinkToCospendApp($project['id']),
$thumbnailUrl === '' ? 'icon-cospend-search-fallback' : '',
true
);
}, $resultProjects);
return SearchResult::paginated(
$this->getName(),
$formattedResults,
$offset + $limit
);
}
/**
* @param array $project
* @return string
*/
protected function getMainText(array $project): string {
return $project['name'];
}
/**
* @param array $project
* @return string
*/
protected function getSubline(array $project): string {
$ownerId = $project['userid'];
$owner = $this->userManager->get($ownerId);
if ($owner === null) {
return '';
}
return $this->l10n->t('Owned by %1$s', [$owner->getDisplayName()]);
}
/**
* @param string $projectId
* @return string
*/
protected function getDeepLinkToCospendApp(string $projectId): string {
return $this->urlGenerator->linkToRouteAbsolute('cospend.page.indexProject', [
'projectId' => $projectId,
]);
}
/**
* @param array $project
* @return string
*/
protected function getThumbnailUrl(array $project): string {
return '';
// return $this->urlGenerator->imagePath(Application::APP_ID, '');
}
}

View File

@@ -1759,6 +1759,9 @@ class LocalProjectService implements IProjectService {
*
* @param string $userId
* @return array
* @throws CospendBasicException
* @throws DoesNotExistException
* @throws MultipleObjectsReturnedException
* @throws \OCP\DB\Exception
*/
public function getLocalProjects(string $userId): array {