Skip to content

Commit

Permalink
Merge pull request #953 from nextcloud/enh/noid/advanced-search
Browse files Browse the repository at this point in the history
adding options 'is:?' to search
  • Loading branch information
ArtificialOwl authored Jun 13, 2022
2 parents 0856963 + 876096e commit 1ef6aee
Show file tree
Hide file tree
Showing 5 changed files with 453 additions and 14 deletions.
4 changes: 1 addition & 3 deletions lib/AppInfo/Application.php
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@
use OCA\Circles\Notification\Notifier;
use OCA\Circles\Service\ConfigService;
use OCA\Circles\Service\DavService;
use OCA\Circles\UnifiedSearch\UnifiedSearchProvider;
use OCA\Circles\Search\UnifiedSearchProvider;
use OCP\App\ManagerEvent;
use OCP\AppFramework\App;
use OCP\AppFramework\Bootstrap\IBootContext;
Expand All @@ -90,8 +90,6 @@
use Symfony\Component\EventDispatcher\GenericEvent;
use Throwable;

//use OCA\Files\App as FilesApp;

/**
* Class Application
*
Expand Down
29 changes: 28 additions & 1 deletion lib/Db/CoreQueryBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -391,9 +391,36 @@ public function filterCircle(Circle $circle): void {
return;
}

$expr = $this->expr();
$orX = $expr->orX();
if ($circle->getDisplayName() !== '') {
$this->searchInDBField('display_name', '%' . $circle->getDisplayName() . '%');
$andX = $expr->andX();
foreach (explode(' ', $circle->getDisplayName()) as $word) {
$andX->add(
$expr->iLike(
$this->getDefaultSelectAlias() . '.' . 'display_name',
$this->createNamedParameter('%' . $word . '%')
)
);
}
$orX->add($andX);
}
if ($circle->getDescription() !== '') {
$orDescription = $expr->orX();
foreach (explode(' ', $circle->getDescription()) as $word) {
$orDescription->add(
$expr->iLike(
$this->getDefaultSelectAlias() . '.' . 'description',
$this->createNamedParameter('%' . $word . '%')
)
);
}
$orX->add($orDescription);
}
if ($orX->count() > 0) {
$this->andWhere($orX);
}

if ($circle->getSource() > 0) {
$this->limitInt('source', $circle->getSource());
}
Expand Down
153 changes: 153 additions & 0 deletions lib/Search/UnifiedSearchProvider.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
<?php

declare(strict_types=1);


/**
* Circles - Bring cloud-users closer together.
*
* This file is licensed under the Affero General Public License version 3 or
* later. See the COPYING file.
*
* @author Maxence Lange <maxence@artificial-owl.com>
* @copyright 2021
* @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\Circles\Search;

use Exception;
use OCA\Circles\Exceptions\ParseMemberLevelException;
use OCA\Circles\Model\Member;
use OCA\Circles\Service\FederatedUserService;
use OCA\Circles\Service\SearchService;
use OCP\IL10N;
use OCP\IUser;
use OCP\Search\IProvider;
use OCP\Search\ISearchQuery;
use OCP\Search\SearchResult;

class UnifiedSearchProvider implements IProvider {
public const PROVIDER_ID = 'circles';
public const ORDER = 9;


/** @var IL10N */
private $l10n;

/** @var FederatedUserService */
private $federatedUserService;

/** @var SearchService */
private $searchService;


/**
* @param IL10N $l10n
* @param FederatedUserService $federatedUserService
* @param SearchService $searchService
*/
public function __construct(
IL10N $l10n,
FederatedUserService $federatedUserService,
SearchService $searchService
) {
$this->l10n = $l10n;
$this->federatedUserService = $federatedUserService;
$this->searchService = $searchService;
}


/**
* return unique id of the provider
*/
public function getId(): string {
return self::PROVIDER_ID;
}


/**
* @return string
*/
public function getName(): string {
return $this->l10n->t('Circles');
}


/**
* @param string $route
* @param array $routeParameters
*
* @return int
*/
public function getOrder(string $route, array $routeParameters): int {
return self::ORDER;
}


/**
* @param IUser $user
* @param ISearchQuery $query
*
* @return SearchResult
*/
public function search(IUser $user, ISearchQuery $query): SearchResult {
$term = $query->getTerm();
$options = $this->extractOptions($term);

$result = [];
try {
$this->federatedUserService->setLocalCurrentUser($user);
$result = $this->searchService->unifiedSearch($term, $options);
} catch (Exception $e) {
}

return SearchResult::paginated(
$this->l10n->t('Circles'),
$result,
($query->getCursor() ?? 0) + $query->getLimit()
);
}


/**
* This is temporary, should be handled by core to extract Options from Term
*
* @param string $term
*
* @return array
*/
private function extractOptions(string &$term): array {
$new = $options = [];
foreach (explode(' ', $term) as $word) {
$word = trim($word);
if (strtolower(substr($word, 0, 3)) === 'is:') {
try {
$options['level'] = Member::parseLevelString(substr($word, 3));
} catch (ParseMemberLevelException $e) {
}
} else {
$new[] = $word;
}
}

$term = trim(implode(' ', $new));

return $options;
}
}
173 changes: 173 additions & 0 deletions lib/Search/UnifiedSearchResult.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,173 @@
<?php

declare(strict_types=1);


/**
* Circles - Bring cloud-users closer together.
*
* This file is licensed under the Affero General Public License version 3 or
* later. See the COPYING file.
*
* @author Maxence Lange <maxence@artificial-owl.com>
* @copyright 2021
* @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\Circles\Search;

use OCP\Search\SearchResultEntry;

class UnifiedSearchResult extends SearchResultEntry {


/**
* UnifiedSearchResult constructor.
*
* @param string $thumbnailUrl
* @param string $title
* @param string $subline
* @param string $resourceUrl
* @param string $icon
* @param bool $rounded
*/
public function __construct(
string $thumbnailUrl = '',
string $title = '',
string $subline = '',
string $resourceUrl = '',
string $icon = '',
bool $rounded = false
) {
parent::__construct($thumbnailUrl, $title, $subline, $resourceUrl, $icon, $rounded);
}


/**
* @return string
*/
public function getThumbnailUrl(): string {
return $this->thumbnailUrl;
}

/**
* @param string $thumbnailUrl
*
* @return UnifiedSearchResult
*/
public function setThumbnailUrl(string $thumbnailUrl): self {
$this->thumbnailUrl = $thumbnailUrl;

return $this;
}


/**
* @return string
*/
public function getTitle(): string {
return $this->title;
}

/**
* @param string $title
*
* @return UnifiedSearchResult
*/
public function setTitle(string $title): self {
$this->title = $title;

return $this;
}


/**
* @return string
*/
public function getSubline(): string {
return $this->subline;
}

/**
* @param string $subline
*
* @return UnifiedSearchResult
*/
public function setSubline(string $subline): self {
$this->subline = $subline;

return $this;
}


/**
* @return string
*/
public function getResourceUrl(): string {
return $this->resourceUrl;
}

/**
* @param string $resourceUrl
*
* @return UnifiedSearchResult
*/
public function setResourceUrl(string $resourceUrl): self {
$this->resourceUrl = $resourceUrl;

return $this;
}


/**
* @return string
*/
public function getIcon(): string {
return $this->icon;
}

/**
* @param string $icon
*
* @return UnifiedSearchResult
*/
public function setIcon(string $icon): self {
$this->icon = $icon;

return $this;
}


/**
* @return bool
*/
public function isRounded(): bool {
return $this->rounded;
}

/**
* @param bool $rounded
*
* @return UnifiedSearchResult
*/
public function setRounded(bool $rounded): self {
$this->rounded = $rounded;

return $this;
}
}
Loading

0 comments on commit 1ef6aee

Please sign in to comment.