2 * Copyright (C) 2007 EDIT
3 * European Distributed Institute of Taxonomy
4 * http://www.e-taxonomy.eu
6 * The contents of this file are subject to the Mozilla Public License Version 1.1
7 * See LICENSE.TXT at the top of this package for the full license terms.
9 package eu
.etaxonomy
.cdm
.strategy
.cache
.name
;
11 import java
.util
.ArrayList
;
12 import java
.util
.Arrays
;
13 import java
.util
.Collection
;
14 import java
.util
.Iterator
;
15 import java
.util
.List
;
16 import java
.util
.UUID
;
18 import org
.apache
.logging
.log4j
.LogManager
;
19 import org
.apache
.logging
.log4j
.Logger
;
21 import eu
.etaxonomy
.cdm
.common
.CdmUtils
;
22 import eu
.etaxonomy
.cdm
.compare
.name
.NomenclaturalStatusComparator
;
23 import eu
.etaxonomy
.cdm
.format
.reference
.NomenclaturalSourceFormatter
;
24 import eu
.etaxonomy
.cdm
.model
.common
.CdmBase
;
25 import eu
.etaxonomy
.cdm
.model
.common
.Language
;
26 import eu
.etaxonomy
.cdm
.model
.name
.INonViralName
;
27 import eu
.etaxonomy
.cdm
.model
.name
.NomenclaturalStatus
;
28 import eu
.etaxonomy
.cdm
.model
.name
.NomenclaturalStatusType
;
29 import eu
.etaxonomy
.cdm
.model
.name
.TaxonName
;
30 import eu
.etaxonomy
.cdm
.model
.term
.Representation
;
31 import eu
.etaxonomy
.cdm
.ref
.TypedEntityReference
;
32 import eu
.etaxonomy
.cdm
.strategy
.StrategyBase
;
33 import eu
.etaxonomy
.cdm
.strategy
.cache
.HTMLTagRules
;
34 import eu
.etaxonomy
.cdm
.strategy
.cache
.TagEnum
;
35 import eu
.etaxonomy
.cdm
.strategy
.cache
.TaggedCacheHelper
;
36 import eu
.etaxonomy
.cdm
.strategy
.cache
.TaggedText
;
37 import eu
.etaxonomy
.cdm
.strategy
.parser
.NonViralNameParserImplRegExBase
;
42 public abstract class NameCacheStrategyBase
44 implements INameCacheStrategy
{
45 private static final long serialVersionUID
= -2322348388258675517L;
47 private static final Logger logger
= LogManager
.getLogger(NameCacheStrategyBase
.class);
49 final static UUID uuid
= UUID
.fromString("817ae5b5-3ac2-414b-a134-a9ae86cba040");
51 public NameCacheStrategyBase() {
56 public String
getFullTitleCache(TaxonName taxonName
, HTMLTagRules htmlTagRules
) {
57 List
<TaggedText
> tags
= getTaggedFullTitle(taxonName
);
61 String result
= createString(tags
, htmlTagRules
);
67 public String
getFullTitleCache(TaxonName taxonName
) {
68 return getFullTitleCache(taxonName
, null);
72 public List
<TaggedText
> getNomStatusTags(TaxonName taxonName
, boolean includeSeparatorBefore
,
73 boolean includeSeparatorAfter
) {
76 Collection
<NomenclaturalStatus
> ncStati
= taxonName
.getStatus();
77 if (ncStati
.size() > 1) {
78 //order to have defined behavior
79 ncStati
= new ArrayList
<>(ncStati
);
80 ((List
<NomenclaturalStatus
>)ncStati
).sort(NomenclaturalStatusComparator
.SINGLETON());
82 Iterator
<NomenclaturalStatus
> iterator
= ncStati
.iterator();
83 List
<TaggedText
> nomStatusTags
= new ArrayList
<>();
84 while (iterator
.hasNext()) {
85 NomenclaturalStatus ncStatus
= iterator
.next();
86 // since the NewInstance method of nomencatural status allows null as parameter
87 // we have to check for null values here
88 String nomStatusStr
= "not defined";
89 if(ncStatus
.getType() != null){
90 NomenclaturalStatusType statusType
= ncStatus
.getType();
91 List
<Language
> prefLangs
= Arrays
.asList(new Language
[]{Language
.LATIN(), Language
.DEFAULT()});
92 Representation repr
= statusType
.getPreferredRepresentation(prefLangs
);
94 if(!Language
.LATIN().equals(repr
.getLanguage())){
95 String message
= "No latin representation available for nom. status. " + statusType
.getTitleCache();
98 nomStatusStr
= repr
.getAbbreviatedLabel();
100 String message
= "No representation available for nom. status. " + statusType
.getTitleCache();
101 logger
.warn(message
);
102 nomStatusStr
= statusType
.getTitleCache();
104 }else if(isNotBlank(ncStatus
.getRuleConsidered())){
105 nomStatusStr
= ncStatus
.getRuleConsidered();
107 String statusSeparator
= ", ";
108 if (includeSeparatorBefore
){
109 nomStatusTags
.add(new TaggedText(TagEnum
.separator
, statusSeparator
));
111 nomStatusTags
.add(new TaggedText(TagEnum
.nomStatus
, nomStatusStr
, new TypedEntityReference
<>(ncStatus
.getClass(), ncStatus
.getUuid())));
112 if (includeSeparatorAfter
){
113 nomStatusTags
.add(new TaggedText(TagEnum
.postSeparator
, ","));
116 return nomStatusTags
;
120 public String
getNameCache(TaxonName nonViralName
) {
121 List
<TaggedText
> tags
= getTaggedName(nonViralName
);
125 String result
= createString(tags
);
131 public String
getNameCache(TaxonName nonViralName
, HTMLTagRules htmlTagRules
) {
132 List
<TaggedText
> tags
= getTaggedName(nonViralName
);
136 String result
= createString(tags
, htmlTagRules
);
142 * Generates and returns the title cache of the given name.
143 * The title cache in general includes the name and the authorship and year for some types of names.
145 * @see eu.etaxonomy.cdm.strategy.INameCacheStrategy#getTitleCache(eu.etaxonomy.cdm.model.common.CdmBase)
146 * @see eu.etaxonomy.cdm.strategy.cache.common.IIdentifiableEntityCacheStrategy#getTitleCache(eu.etaxonomy.cdm.model.common.IdentifiableEntity)
149 public String
getTitleCache(TaxonName taxonName
) {
150 return getTitleCache(taxonName
, null);
154 public String
getTitleCache(TaxonName taxonName
, HTMLTagRules htmlTagRules
) {
155 List
<TaggedText
> tags
= getTaggedTitle(taxonName
);
159 String result
= createString(tags
, htmlTagRules
);
165 public List
<TaggedText
> getTaggedTitle(TaxonName taxonName
) {
166 if (taxonName
== null){
169 //TODO how to handle protected fullTitleCache here?
171 if (taxonName
.isProtectedTitleCache()){
172 //protected title cache
173 List
<TaggedText
> tags
= new ArrayList
<>();
174 tags
.add(new TaggedText(TagEnum
.name
, taxonName
.getTitleCache()));
177 return doGetTaggedTitle(taxonName
);
181 protected abstract List
<TaggedText
> doGetTaggedTitle(TaxonName taxonName
);
184 public List
<TaggedText
> getTaggedFullTitle(TaxonName taxonName
) {
185 List
<TaggedText
> tags
= new ArrayList
<>();
188 if (taxonName
== null){
192 //protected full title cache
193 if (taxonName
.isProtectedFullTitleCache()){
194 tags
.add(new TaggedText(TagEnum
.fullName
, taxonName
.getFullTitleCache()));
199 // String titleCache = nonViralName.getTitleCache();
200 List
<TaggedText
> titleTags
= getTaggedTitle(taxonName
);
201 tags
.addAll(titleTags
);
204 String referenceCache
= NomenclaturalSourceFormatter
.INSTANCE().format(taxonName
.getNomenclaturalSource());
206 if (isNotBlank(referenceCache
)){
207 if (! referenceCache
.trim().startsWith("in ")){
208 String refConcat
= ", ";
209 tags
.add(new TaggedText(TagEnum
.separator
, refConcat
));
211 tags
.add(new TaggedText(TagEnum
.reference
, referenceCache
));
214 addOriginalSpelling(tags
, taxonName
);
216 //nomenclatural status
217 tags
.addAll(getNomStatusTags(taxonName
, true, false));
221 protected void addOriginalSpelling(List
<TaggedText
> tags
, TaxonName currentName
){
223 currentName
= CdmBase
.deproxy(currentName
);
224 //Hibernate.initialize(currentName.getRelationsToThisName());
225 TaxonName originalName
= currentName
.getOriginalSpelling();
226 if (originalName
!= null){
228 tags
.add(TaggedText
.NewSeparatorInstance(" [as \""));
229 if (!originalName
.isNonViral()){
230 originalInfo
= originalName
.getTitleCache();
231 tags
.add(new TaggedText(TagEnum
.name
, originalInfo
));
233 INonViralName originalNvName
= CdmBase
.deproxy(originalName
);
234 originalInfo
= makeOriginalInfo(originalNvName
, tags
);
235 for (String split
: originalInfo
.split(" ")){
236 if (split
.matches(NonViralNameParserImplRegExBase
.infraSpeciesMarker
)
237 || split
.matches(NonViralNameParserImplRegExBase
.oldInfraSpeciesMarker
)) {
238 tags
.add(new TaggedText(TagEnum
.rank
, split
));
240 tags
.add(new TaggedText(TagEnum
.name
, split
));
244 tags
.add(TaggedText
.NewSeparatorInstance("\"]"));
250 private String
makeOriginalInfo(INonViralName originalName
,
251 List
<TaggedText
> currentNameTags
) {
252 //use cache if necessary
253 String cacheToUse
= null;
254 if (originalName
.isProtectedNameCache() && isNotBlank(originalName
.getNameCache())){
255 cacheToUse
= originalName
.getNameCache();
256 }else if (originalName
.isProtectedTitleCache() && isNotBlank(originalName
.getTitleCache())){
257 cacheToUse
= originalName
.getTitleCache();
258 }else if (originalName
.isProtectedFullTitleCache() && isNotBlank(originalName
.getFullTitleCache())){
259 cacheToUse
= originalName
.getFullTitleCache();
261 if (cacheToUse
!= null){
265 //get originalNameParts array
266 String originalInfo
= originalName
.getNameCache();
267 if (originalInfo
== null){
268 originalInfo
= originalName
.getTitleCache();
270 if (originalInfo
== null){ //should not happen
271 originalInfo
= originalName
.getFullTitleCache();
273 String
[] originalNameSplit
= originalInfo
.split("\\s+");
275 //get current name parts
276 String currentNameString
= createString(currentNameTags
);
277 String
[] currentNameSplit
= currentNameString
.split("\\s+");
280 String result
= originalInfo
;
281 Integer firstDiff
= null;
282 Integer lastDiff
= -1;
283 for (int i
= 0; i
< Math
.min(originalNameSplit
.length
, currentNameSplit
.length
); i
++){
284 if (!originalNameSplit
[i
].equals(currentNameSplit
[i
])){
286 firstDiff
= (firstDiff
== null) ? i
: firstDiff
;
289 if (firstDiff
!= null){
290 result
= CdmUtils
.concat(" ", Arrays
.asList(originalNameSplit
).subList(firstDiff
, lastDiff
+1).toArray(new String
[0]));
296 protected String
createString(List
<TaggedText
> tags
) {
297 return TaggedCacheHelper
.createString(tags
);
300 protected String
createString(List
<TaggedText
> tags
, HTMLTagRules htmlTagRules
) {
301 return TaggedCacheHelper
.createString(tags
, htmlTagRules
);