3 * Copyright (C) 2007 EDIT
4 * European Distributed Institute of Taxonomy
5 * http://www.e-taxonomy.eu
7 * The contents of this file are subject to the Mozilla Public License Version 1.1
8 * See LICENSE.TXT at the top of this package for the full license terms.
11 package eu
.etaxonomy
.taxeditor
.store
;
13 import java
.util
.ArrayList
;
14 import java
.util
.Arrays
;
15 import java
.util
.List
;
16 import java
.util
.UUID
;
18 import org
.eclipse
.jface
.dialogs
.MessageDialog
;
19 import org
.eclipse
.swt
.widgets
.Display
;
21 import eu
.etaxonomy
.cdm
.api
.service
.IAgentService
;
22 import eu
.etaxonomy
.cdm
.api
.service
.IGroupService
;
23 import eu
.etaxonomy
.cdm
.api
.service
.INameService
;
24 import eu
.etaxonomy
.cdm
.api
.service
.IOccurrenceService
;
25 import eu
.etaxonomy
.cdm
.api
.service
.IReferenceService
;
26 import eu
.etaxonomy
.cdm
.api
.service
.IService
;
27 import eu
.etaxonomy
.cdm
.api
.service
.ITaxonService
;
28 import eu
.etaxonomy
.cdm
.api
.service
.IUserService
;
29 import eu
.etaxonomy
.cdm
.api
.service
.config
.IFindTaxaAndNamesConfigurator
;
30 import eu
.etaxonomy
.cdm
.api
.service
.config
.IIdentifiableEntityServiceConfigurator
;
31 import eu
.etaxonomy
.cdm
.model
.agent
.AgentBase
;
32 import eu
.etaxonomy
.cdm
.model
.agent
.TeamOrPersonBase
;
33 import eu
.etaxonomy
.cdm
.model
.common
.Group
;
34 import eu
.etaxonomy
.cdm
.model
.common
.ICdmBase
;
35 import eu
.etaxonomy
.cdm
.model
.common
.IdentifiableEntity
;
36 import eu
.etaxonomy
.cdm
.model
.common
.RelationshipBase
;
37 import eu
.etaxonomy
.cdm
.model
.common
.User
;
38 import eu
.etaxonomy
.cdm
.model
.name
.NameRelationship
;
39 import eu
.etaxonomy
.cdm
.model
.name
.TaxonNameBase
;
40 import eu
.etaxonomy
.cdm
.model
.occurrence
.DerivedUnit
;
41 import eu
.etaxonomy
.cdm
.model
.occurrence
.SpecimenOrObservationBase
;
42 import eu
.etaxonomy
.cdm
.model
.reference
.Reference
;
43 import eu
.etaxonomy
.cdm
.model
.taxon
.TaxonBase
;
44 import eu
.etaxonomy
.cdm
.persistence
.dto
.UuidAndTitleCache
;
45 import eu
.etaxonomy
.taxeditor
.preference
.IPreferenceKeys
;
46 import eu
.etaxonomy
.taxeditor
.preference
.PreferencesUtil
;
50 * @created Dec 8, 2010
53 public class SearchManager
{
55 public static final List NO_RESULTS
= Arrays
.asList(new Object
[]{});
57 public static final String WILDCARD
= "*";
59 public static int NO_COUNT
= -1;
61 // TODO make this configurable via preferences
62 private static final int MAX_RESULTS_BEFORE_WARNING
= 500;
64 public List
<TaxonNameBase
> findNames(IIdentifiableEntityServiceConfigurator configurator
){
65 if(checkLargeResult(CdmStore
.getService(INameService
.class).countByTitle(configurator
))){
66 List
<TaxonNameBase
> records
= CdmStore
.getService(INameService
.class).findByTitle(configurator
).getRecords();
67 addUuidSearchResults(records
, configurator
, INameService
.class);
73 public List
<NameRelationship
> findNameRelationships(
74 IIdentifiableEntityServiceConfigurator configurator
) {
76 //if activated again remember to add uuid search results like in other searches
80 List
<NameRelationship
> relationships
= new ArrayList
<NameRelationship
>();
81 List
<RelationshipBase
> all
= CdmStore
.getService(INameService
.class).getAllRelationships(0, 0);
83 for (RelationshipBase relationship
: all
){
84 if(relationship
instanceof NameRelationship
){
85 relationships
.add((NameRelationship
) relationship
);
91 public List
<UuidAndTitleCache
<IdentifiableEntity
>> findTaxaAndNames(IFindTaxaAndNamesConfigurator
<TaxonBase
> configurator
){
92 return CdmStore
.getService(ITaxonService
.class).findTaxaAndNamesForEditor(configurator
);
95 public List
<Reference
> findReferences(IIdentifiableEntityServiceConfigurator configurator
){
96 if(checkLargeResult(CdmStore
.getService(IReferenceService
.class).countByTitle(configurator
))){
97 List
<Reference
> records
= CdmStore
.getService(IReferenceService
.class).findByTitle(configurator
).getRecords();
98 addUuidSearchResults(records
, configurator
, IReferenceService
.class);
104 public List
<AgentBase
> findAgents(IIdentifiableEntityServiceConfigurator configurator
){
105 if(checkLargeResult(CdmStore
.getService(IAgentService
.class).countByTitle(configurator
))){
106 List
<AgentBase
> records
= CdmStore
.getService(IAgentService
.class).findByTitle(configurator
).getRecords();
107 addUuidSearchResults(records
, configurator
, IAgentService
.class);
114 * Check search string if it is a {@link UUID} and, if <code>true</code>, search for the corresponding entity.
115 * @param records the list to which the search results are added
116 * @param configurator the configurator holding the search string
117 * @param service the service to use for searching
119 private <T
extends ICdmBase
> void addUuidSearchResults(List
<T
> records
, IIdentifiableEntityServiceConfigurator configurator
, Class
<?
extends IService
<T
>> service
) {
120 String titleSearchString
= configurator
.getTitleSearchString();
122 UUID uuid
= UUID
.fromString(titleSearchString
);
123 T foundRecord
= CdmStore
.getService(service
).find(uuid
);
124 if(foundRecord
!=null){
125 records
.add(foundRecord
);
127 } catch (IllegalArgumentException e
) {
128 //search string was no UUID
132 @SuppressWarnings("unchecked")
133 public List
<TeamOrPersonBase
> findTeamOrPersons(IIdentifiableEntityServiceConfigurator configurator
){
134 configurator
.setClazz(TeamOrPersonBase
.class);
135 return (List
)findAgents(configurator
);
139 * Searches for {@link SpecimenOrObservationBase} with the parameters specified in the
140 * {@link IIdentifiableEntityServiceConfigurator}<br>
142 * Note: FieldUnits are omitted by default. See {@link #findOccurrences(IIdentifiableEntityServiceConfigurator, boolean)}
144 * @param configurator the configurator to use for the search
145 * @return a list of the SpecimenOrObservationBases matching the search parameters found
147 public List
<SpecimenOrObservationBase
> findOccurrences(IIdentifiableEntityServiceConfigurator configurator
){
148 // by default we do not show field units. This may be configured via preferences
149 return findOccurrences(configurator
, PreferencesUtil
.getPreferenceStore().getBoolean(IPreferenceKeys
.BULK_EDITOR_OCCURRENCE_SHOW_FIELD_UNITS
));
154 * Searches for {@link SpecimenOrObservationBase} with the parameters specified in the
155 * {@link IIdentifiableEntityServiceConfigurator}
157 * @param configurator the configurator to use for the search
158 * @return a list of the SpecimenOrObservationBases found
159 * @param showFieldUnits if <code>true</code> then also FieldUnits are searched
162 public List
<SpecimenOrObservationBase
> findOccurrences(IIdentifiableEntityServiceConfigurator
<SpecimenOrObservationBase
> configurator
, boolean showFieldUnits
){
163 List
<SpecimenOrObservationBase
> records
= new ArrayList
<SpecimenOrObservationBase
>();
164 final List
<String
> BASE_OCCURRENCE_INIT_STRATEGY
= Arrays
.asList(new String
[] {
168 "derivationEvents.originals",
169 "derivedFrom.originals",
170 "gatheringEvent.country.representations",
171 "gatheringEvent.collector",
172 "gatheringEvent.locality",
173 "descriptions.descriptionElements",
175 "amplificationResults",
176 "sequences.singleReadAlignments",
180 if(configurator
.getClazz()==null){
182 configurator
.setClazz(SpecimenOrObservationBase
.class);
185 configurator
.setClazz(DerivedUnit
.class);
188 List
<String
> occurrencePropertyPaths
= new ArrayList
<String
>();
189 occurrencePropertyPaths
.addAll(BASE_OCCURRENCE_INIT_STRATEGY
);
190 for(String propertyPath
:BASE_OCCURRENCE_INIT_STRATEGY
) {
191 occurrencePropertyPaths
.add("derivationEvents.derivatives." + propertyPath
);
193 configurator
.setPropertyPaths(occurrencePropertyPaths
);
195 if(checkLargeResult(CdmStore
.getService(IOccurrenceService
.class).countOccurrences(configurator
))){
196 records
= CdmStore
.getService(IOccurrenceService
.class).findByTitle(configurator
).getRecords();
198 addUuidSearchResults(records
, configurator
, IOccurrenceService
.class);
202 public List
<User
> findUsers(IIdentifiableEntityServiceConfigurator configurator
){
203 String userNameSearchString
= sqlizeTitleSearchString(configurator
);
204 // TODO why are users not identifiable entities?
205 List
<User
> records
= CdmStore
.getService(IUserService
.class).listByUsername(userNameSearchString
, null, null, null, null, null, null);
206 addUuidSearchResults(records
, configurator
, IUserService
.class);
211 public List
<Group
> findGroups(IIdentifiableEntityServiceConfigurator configurator
){
212 String groupNameSearchString
= sqlizeTitleSearchString(configurator
);
213 // TODO why are groups not identifiable entities?
214 List
<Group
> records
= CdmStore
.getService(IGroupService
.class).listByName(groupNameSearchString
, null, null, null, null, null, null);
215 addUuidSearchResults(records
, configurator
, IGroupService
.class);
220 private boolean checkLargeResult(int count
) {
221 return checkLargeResult(count
, MAX_RESULTS_BEFORE_WARNING
);
224 private boolean checkLargeResult(int count
, int maxBeforWarning
) {
225 if(count
> maxBeforWarning
){
226 return MessageDialog
.openConfirm(Display
.getDefault().getActiveShell(), "Large result expected",
227 String
.format("The current search will return %s objects. This will " +
228 "take a long time and/or might render the editor unusable. Please consider refining your search.", count
));
234 private String
sqlizeTitleSearchString(IIdentifiableEntityServiceConfigurator configurator
){
235 return configurator
.getTitleSearchString().replace(WILDCARD
, "%");
238 public List
findTaxa(IIdentifiableEntityServiceConfigurator configurator
) {
239 if(checkLargeResult(CdmStore
.getService(ITaxonService
.class).countByTitle(configurator
))){
240 List
<TaxonBase
> records
= CdmStore
.getService(ITaxonService
.class).findByTitle(configurator
).getRecords();
241 addUuidSearchResults(records
, configurator
, ITaxonService
.class);