001package org.hl7.fhir.dstu3.utils;
002
003/*
004  Copyright (c) 2011+, HL7, Inc.
005  All rights reserved.
006  
007  Redistribution and use in source and binary forms, with or without modification, 
008  are permitted provided that the following conditions are met:
009    
010   * Redistributions of source code must retain the above copyright notice, this 
011     list of conditions and the following disclaimer.
012   * Redistributions in binary form must reproduce the above copyright notice, 
013     this list of conditions and the following disclaimer in the documentation 
014     and/or other materials provided with the distribution.
015   * Neither the name of HL7 nor the names of its contributors may be used to 
016     endorse or promote products derived from this software without specific 
017     prior written permission.
018  
019  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND 
020  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 
021  WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 
022  IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, 
023  INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 
024  NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
025  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 
026  WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 
027  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 
028  POSSIBILITY OF SUCH DAMAGE.
029  
030 */
031
032
033
034import java.io.IOException;
035import java.io.UnsupportedEncodingException;
036import java.text.ParseException;
037import java.text.SimpleDateFormat;
038
039/*
040Copyright (c) 2011+, HL7, Inc
041  All rights reserved.
042
043  Redistribution and use in source and binary forms, with or without modification,
044  are permitted provided that the following conditions are met:
045
046   * Redistributions of source code must retain the above copyright notice, this
047     list of conditions and the following disclaimer.
048   * Redistributions in binary form must reproduce the above copyright notice,
049     this list of conditions and the following disclaimer in the documentation
050     and/or other materials provided with the distribution.
051   * Neither the name of HL7 nor the names of its contributors may be used to
052     endorse or promote products derived from this software without specific
053     prior written permission.
054
055  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
056  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
057  WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
058  IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
059  INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
060  NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
061  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
062  WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
063  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
064  POSSIBILITY OF SUCH DAMAGE.
065
066*/
067
068import java.util.ArrayList;
069import java.util.Collections;
070import java.util.Date;
071import java.util.HashMap;
072import java.util.HashSet;
073import java.util.List;
074import java.util.Map;
075
076import org.apache.commons.codec.binary.Base64;
077import org.apache.commons.io.output.ByteArrayOutputStream;
078import org.apache.commons.lang3.NotImplementedException;
079import org.hl7.fhir.dstu3.conformance.ProfileUtilities;
080import org.hl7.fhir.dstu3.conformance.ProfileUtilities.ProfileKnowledgeProvider;
081import org.hl7.fhir.dstu3.context.IWorkerContext;
082import org.hl7.fhir.dstu3.context.IWorkerContext.ValidationResult;
083import org.hl7.fhir.dstu3.formats.FormatUtilities;
084import org.hl7.fhir.dstu3.formats.IParser.OutputStyle;
085import org.hl7.fhir.dstu3.model.Address;
086import org.hl7.fhir.dstu3.model.Annotation;
087import org.hl7.fhir.dstu3.model.Attachment;
088import org.hl7.fhir.dstu3.model.Base;
089import org.hl7.fhir.dstu3.model.Base64BinaryType;
090import org.hl7.fhir.dstu3.model.BooleanType;
091import org.hl7.fhir.dstu3.model.Bundle;
092import org.hl7.fhir.dstu3.model.Bundle.BundleEntryComponent;
093import org.hl7.fhir.dstu3.model.Bundle.BundleEntryRequestComponent;
094import org.hl7.fhir.dstu3.model.Bundle.BundleEntryResponseComponent;
095import org.hl7.fhir.dstu3.model.Bundle.BundleEntrySearchComponent;
096import org.hl7.fhir.dstu3.model.Bundle.BundleType;
097import org.hl7.fhir.dstu3.model.CapabilityStatement;
098import org.hl7.fhir.dstu3.model.CapabilityStatement.CapabilityStatementRestComponent;
099import org.hl7.fhir.dstu3.model.CapabilityStatement.CapabilityStatementRestResourceComponent;
100import org.hl7.fhir.dstu3.model.CapabilityStatement.ResourceInteractionComponent;
101import org.hl7.fhir.dstu3.model.CapabilityStatement.SystemInteractionComponent;
102import org.hl7.fhir.dstu3.model.CapabilityStatement.SystemRestfulInteraction;
103import org.hl7.fhir.dstu3.model.CapabilityStatement.TypeRestfulInteraction;
104import org.hl7.fhir.dstu3.model.CodeSystem;
105import org.hl7.fhir.dstu3.model.CodeSystem.CodeSystemContentMode;
106import org.hl7.fhir.dstu3.model.CodeSystem.CodeSystemFilterComponent;
107import org.hl7.fhir.dstu3.model.CodeSystem.ConceptDefinitionComponent;
108import org.hl7.fhir.dstu3.model.CodeSystem.ConceptDefinitionDesignationComponent;
109import org.hl7.fhir.dstu3.model.CodeSystem.PropertyComponent;
110import org.hl7.fhir.dstu3.model.CodeType;
111import org.hl7.fhir.dstu3.model.CodeableConcept;
112import org.hl7.fhir.dstu3.model.Coding;
113import org.hl7.fhir.dstu3.model.CompartmentDefinition;
114import org.hl7.fhir.dstu3.model.CompartmentDefinition.CompartmentDefinitionResourceComponent;
115import org.hl7.fhir.dstu3.model.Composition;
116import org.hl7.fhir.dstu3.model.Composition.SectionComponent;
117import org.hl7.fhir.dstu3.model.ConceptMap;
118import org.hl7.fhir.dstu3.model.ConceptMap.ConceptMapGroupComponent;
119import org.hl7.fhir.dstu3.model.ConceptMap.OtherElementComponent;
120import org.hl7.fhir.dstu3.model.ConceptMap.SourceElementComponent;
121import org.hl7.fhir.dstu3.model.ConceptMap.TargetElementComponent;
122import org.hl7.fhir.dstu3.model.ContactDetail;
123import org.hl7.fhir.dstu3.model.ContactPoint;
124import org.hl7.fhir.dstu3.model.ContactPoint.ContactPointSystem;
125import org.hl7.fhir.dstu3.model.DateTimeType;
126import org.hl7.fhir.dstu3.model.DiagnosticReport;
127import org.hl7.fhir.dstu3.model.DomainResource;
128import org.hl7.fhir.dstu3.model.Dosage;
129import org.hl7.fhir.dstu3.model.ElementDefinition;
130import org.hl7.fhir.dstu3.model.ElementDefinition.TypeRefComponent;
131import org.hl7.fhir.dstu3.model.Enumeration;
132import org.hl7.fhir.dstu3.model.Enumerations.ConceptMapEquivalence;
133import org.hl7.fhir.dstu3.model.Extension;
134import org.hl7.fhir.dstu3.model.ExtensionHelper;
135import org.hl7.fhir.dstu3.model.HumanName;
136import org.hl7.fhir.dstu3.model.HumanName.NameUse;
137import org.hl7.fhir.dstu3.model.IdType;
138import org.hl7.fhir.dstu3.model.Identifier;
139import org.hl7.fhir.dstu3.model.ImplementationGuide;
140import org.hl7.fhir.dstu3.model.InstantType;
141import org.hl7.fhir.dstu3.model.Meta;
142import org.hl7.fhir.dstu3.model.MetadataResource;
143import org.hl7.fhir.dstu3.model.Narrative;
144import org.hl7.fhir.dstu3.model.Narrative.NarrativeStatus;
145import org.hl7.fhir.dstu3.model.OperationDefinition;
146import org.hl7.fhir.dstu3.model.OperationDefinition.OperationDefinitionParameterComponent;
147import org.hl7.fhir.dstu3.model.OperationOutcome;
148import org.hl7.fhir.dstu3.model.OperationOutcome.IssueSeverity;
149import org.hl7.fhir.dstu3.model.OperationOutcome.OperationOutcomeIssueComponent;
150import org.hl7.fhir.dstu3.model.Period;
151import org.hl7.fhir.dstu3.model.PrimitiveType;
152import org.hl7.fhir.dstu3.model.Property;
153import org.hl7.fhir.dstu3.model.Quantity;
154import org.hl7.fhir.dstu3.model.Questionnaire;
155import org.hl7.fhir.dstu3.model.Range;
156import org.hl7.fhir.dstu3.model.Ratio;
157import org.hl7.fhir.dstu3.model.Reference;
158import org.hl7.fhir.dstu3.model.Resource;
159import org.hl7.fhir.dstu3.model.SampledData;
160import org.hl7.fhir.dstu3.model.Signature;
161import org.hl7.fhir.dstu3.model.StringType;
162import org.hl7.fhir.dstu3.model.StructureDefinition;
163import org.hl7.fhir.dstu3.model.StructureDefinition.StructureDefinitionKind;
164import org.hl7.fhir.dstu3.model.Timing;
165import org.hl7.fhir.dstu3.model.Timing.EventTiming;
166import org.hl7.fhir.dstu3.model.Timing.TimingRepeatComponent;
167import org.hl7.fhir.dstu3.model.Timing.UnitsOfTime;
168import org.hl7.fhir.dstu3.model.Type;
169import org.hl7.fhir.dstu3.model.UriType;
170import org.hl7.fhir.dstu3.model.UsageContext;
171import org.hl7.fhir.dstu3.model.ValueSet;
172import org.hl7.fhir.dstu3.model.ValueSet.ConceptReferenceComponent;
173import org.hl7.fhir.dstu3.model.ValueSet.ConceptReferenceDesignationComponent;
174import org.hl7.fhir.dstu3.model.ValueSet.ConceptSetComponent;
175import org.hl7.fhir.dstu3.model.ValueSet.ConceptSetFilterComponent;
176import org.hl7.fhir.dstu3.model.ValueSet.FilterOperator;
177import org.hl7.fhir.dstu3.model.ValueSet.ValueSetExpansionComponent;
178import org.hl7.fhir.dstu3.model.ValueSet.ValueSetExpansionContainsComponent;
179import org.hl7.fhir.dstu3.model.ValueSet.ValueSetExpansionParameterComponent;
180import org.hl7.fhir.dstu3.terminologies.CodeSystemUtilities;
181import org.hl7.fhir.dstu3.terminologies.ValueSetExpander.ValueSetExpansionOutcome;
182import org.hl7.fhir.exceptions.DefinitionException;
183import org.hl7.fhir.exceptions.FHIRException;
184import org.hl7.fhir.exceptions.FHIRFormatError;
185import org.hl7.fhir.exceptions.TerminologyServiceException;
186import org.hl7.fhir.utilities.CommaSeparatedStringBuilder;
187import org.hl7.fhir.utilities.LoincLinker;
188import org.hl7.fhir.utilities.MarkDownProcessor;
189import org.hl7.fhir.utilities.MarkDownProcessor.Dialect;
190import org.hl7.fhir.utilities.Utilities;
191import org.hl7.fhir.utilities.xhtml.NodeType;
192import org.hl7.fhir.utilities.xhtml.XhtmlComposer;
193import org.hl7.fhir.utilities.xhtml.XhtmlNode;
194import org.hl7.fhir.utilities.xhtml.XhtmlParser;
195import org.hl7.fhir.utilities.xml.XMLUtil;
196import org.hl7.fhir.utilities.xml.XmlGenerator;
197import org.w3c.dom.Element;
198
199public class NarrativeGenerator implements INarrativeGenerator {
200
201  public class ResourceContext {
202    Bundle bundleResource;
203    
204    DomainResource resourceResource;
205    
206    public ResourceContext(Bundle bundle, DomainResource dr) {
207      super();
208      this.bundleResource = bundle;
209      this.resourceResource = dr;
210    }
211
212    public ResourceContext(Element bundle, Element doc) {
213    }
214
215    public ResourceContext(org.hl7.fhir.dstu3.elementmodel.Element bundle, org.hl7.fhir.dstu3.elementmodel.Element er) {
216    }
217
218    public Resource resolve(String value) {
219      if (value.startsWith("#")) {
220        for (Resource r : resourceResource.getContained()) {
221          if (r.getId().equals(value.substring(1)))
222            return r;
223        }
224        return null;
225      }
226      if (bundleResource != null) {
227        for (BundleEntryComponent be : bundleResource.getEntry()) {
228          if (be.getFullUrl().equals(value))
229            return be.getResource();
230          if (value.equals(be.getResource().fhirType()+"/"+be.getResource().getId()))
231            return be.getResource();
232        }
233      }
234      return null;
235    }
236
237  }
238
239  private static final String ABSTRACT_CODE_HINT = "This code is not selectable ('Abstract')";
240
241  public interface IReferenceResolver {
242
243    ResourceWithReference resolve(String url);
244
245  }
246
247  private Bundle bundle;
248  private String definitionsTarget;
249  private String corePath;
250  private String destDir;
251  private ProfileKnowledgeProvider pkp;
252  
253  public boolean generate(Bundle b, boolean evenIfAlreadyHasNarrative) throws EOperationOutcome, FHIRException, IOException {
254    boolean res = false;
255    this.bundle = b;
256    for (BundleEntryComponent be : b.getEntry()) {
257      if (be.hasResource() && be.getResource() instanceof DomainResource) {
258        DomainResource dr = (DomainResource) be.getResource();
259        if (evenIfAlreadyHasNarrative || !dr.getText().hasDiv())
260          res = generate(new ResourceContext(b, dr), dr) || res;
261      }
262    }
263    return res;
264  }
265
266  public boolean generate(DomainResource r) throws EOperationOutcome, FHIRException, IOException {
267    return generate(null, r);
268  }
269  
270  public boolean generate(ResourceContext rcontext, DomainResource r) throws EOperationOutcome, FHIRException, IOException {
271    if (rcontext == null)
272      rcontext = new ResourceContext(null, r);
273    
274    if (r instanceof ConceptMap) {
275      return generate(rcontext, (ConceptMap) r); // Maintainer = Grahame
276    } else if (r instanceof ValueSet) {
277      return generate(rcontext, (ValueSet) r, true); // Maintainer = Grahame
278    } else if (r instanceof CodeSystem) {
279      return generate(rcontext, (CodeSystem) r, true); // Maintainer = Grahame
280    } else if (r instanceof OperationOutcome) {
281      return generate(rcontext, (OperationOutcome) r); // Maintainer = Grahame
282    } else if (r instanceof CapabilityStatement) {
283      return generate(rcontext, (CapabilityStatement) r);   // Maintainer = Grahame
284    } else if (r instanceof CompartmentDefinition) {
285      return generate(rcontext, (CompartmentDefinition) r);   // Maintainer = Grahame
286    } else if (r instanceof OperationDefinition) {
287      return generate(rcontext, (OperationDefinition) r);   // Maintainer = Grahame
288    } else if (r instanceof StructureDefinition) {
289      return generate(rcontext, (StructureDefinition) r);   // Maintainer = Grahame
290    } else if (r instanceof ImplementationGuide) {
291      return generate(rcontext, (ImplementationGuide) r);   // Maintainer = Lloyd (until Grahame wants to take over . . . :))
292    } else if (r instanceof DiagnosticReport) {
293      inject(r, generateDiagnosticReport(new ResourceWrapperDirect(r)),  NarrativeStatus.GENERATED);   // Maintainer = Grahame
294      return true;
295    } else {
296      StructureDefinition p = null;
297      if (r.hasMeta())
298        for (UriType pu : r.getMeta().getProfile())
299          if (p == null)
300            p = context.fetchResource(StructureDefinition.class, pu.getValue());
301      if (p == null)
302        p = context.fetchResource(StructureDefinition.class, r.getResourceType().toString());
303      if (p == null)
304        p = context.fetchTypeDefinition(r.getResourceType().toString().toLowerCase());
305      if (p != null)
306        return generateByProfile(r, p, true);
307      else
308        return false;
309    }
310  }
311
312  private interface PropertyWrapper {
313    public String getName();
314    public boolean hasValues();
315    public List<BaseWrapper> getValues();
316    public String getTypeCode();
317    public String getDefinition();
318    public int getMinCardinality();
319    public int getMaxCardinality();
320    public StructureDefinition getStructure();
321    public BaseWrapper value();
322  }
323
324  private interface ResourceWrapper {
325    public List<ResourceWrapper> getContained();
326    public String getId();
327    public XhtmlNode getNarrative() throws FHIRFormatError, IOException, FHIRException;
328    public String getName();
329    public List<PropertyWrapper> children();
330  }
331
332  private interface BaseWrapper {
333    public Base getBase() throws UnsupportedEncodingException, IOException, FHIRException;
334    public List<PropertyWrapper> children();
335    public PropertyWrapper getChildByName(String tail);
336  }
337
338  private class BaseWrapperElement implements BaseWrapper {
339    private Element element;
340    private String type;
341    private StructureDefinition structure;
342    private ElementDefinition definition;
343    private List<ElementDefinition> children;
344    private List<PropertyWrapper> list;
345
346    public BaseWrapperElement(Element element, String type, StructureDefinition structure, ElementDefinition definition) {
347      this.element = element;
348      this.type = type;
349      this.structure = structure;
350      this.definition = definition;
351    }
352
353    @Override
354    public Base getBase() throws UnsupportedEncodingException, IOException, FHIRException {
355      if (type == null || type.equals("Resource") || type.equals("BackboneElement") || type.equals("Element"))
356        return null;
357
358      String xml;
359                try {
360                        xml = new XmlGenerator().generate(element);
361                } catch (org.hl7.fhir.exceptions.FHIRException e) {
362                        throw new FHIRException(e.getMessage(), e);
363                }
364      return context.newXmlParser().setOutputStyle(OutputStyle.PRETTY).parseType(xml, type);
365    }
366
367    @Override
368    public List<PropertyWrapper> children() {
369      if (list == null) {
370        children = ProfileUtilities.getChildList(structure, definition);
371        list = new ArrayList<NarrativeGenerator.PropertyWrapper>();
372        for (ElementDefinition child : children) {
373          List<Element> elements = new ArrayList<Element>();
374          XMLUtil.getNamedChildrenWithWildcard(element, tail(child.getPath()), elements);
375          list.add(new PropertyWrapperElement(structure, child, elements));
376        }
377      }
378      return list;
379    }
380
381    @Override
382    public PropertyWrapper getChildByName(String name) {
383      for (PropertyWrapper p : children())
384        if (p.getName().equals(name))
385          return p;
386      return null;
387    }
388
389  }
390
391  private class PropertyWrapperElement implements PropertyWrapper {
392
393    private StructureDefinition structure;
394    private ElementDefinition definition;
395    private List<Element> values;
396    private List<BaseWrapper> list;
397
398    public PropertyWrapperElement(StructureDefinition structure, ElementDefinition definition, List<Element> values) {
399      this.structure = structure;
400      this.definition = definition;
401      this.values = values;
402    }
403
404    @Override
405    public String getName() {
406      return tail(definition.getPath());
407    }
408
409    @Override
410    public boolean hasValues() {
411      return values.size() > 0;
412    }
413
414    @Override
415    public List<BaseWrapper> getValues() {
416      if (list == null) {
417        list = new ArrayList<NarrativeGenerator.BaseWrapper>();
418        for (Element e : values)
419          list.add(new BaseWrapperElement(e, determineType(e), structure, definition));
420      }
421      return list;
422    }
423    private String determineType(Element e) {
424      if (definition.getType().isEmpty())
425        return null;
426      if (definition.getType().size() == 1) {
427        if (definition.getType().get(0).getCode().equals("Element") || definition.getType().get(0).getCode().equals("BackboneElement"))
428          return null;
429        return definition.getType().get(0).getCode();
430      }
431      String t = e.getNodeName().substring(tail(definition.getPath()).length()-3);
432      boolean allReference = true;
433      for (TypeRefComponent tr : definition.getType()) {
434        if (!tr.getCode().equals("Reference"))
435          allReference = false;
436      }
437      if (allReference)
438        return "Reference";
439
440      if (isPrimitive(Utilities.uncapitalize(t)))
441        return Utilities.uncapitalize(t);
442      else
443        return t;
444    }
445
446    private boolean isPrimitive(String code) {
447      StructureDefinition sd = context.fetchTypeDefinition(code);
448      return sd != null && sd.getKind() == StructureDefinitionKind.PRIMITIVETYPE;
449    }
450
451    @Override
452    public String getTypeCode() {
453      if (definition == null || definition.getType().size() != 1)
454        throw new Error("not handled");
455      return definition.getType().get(0).getCode();
456    }
457
458    @Override
459    public String getDefinition() {
460      if (definition == null)
461        throw new Error("not handled");
462      return definition.getDefinition();
463    }
464
465    @Override
466    public int getMinCardinality() {
467      if (definition == null)
468        throw new Error("not handled");
469      return definition.getMin();
470    }
471
472    @Override
473    public int getMaxCardinality() {
474      if (definition == null)
475        throw new Error("not handled");
476      return definition.getMax().equals("*") ? Integer.MAX_VALUE : Integer.parseInt(definition.getMax());
477    }
478
479    @Override
480    public StructureDefinition getStructure() {
481      return structure;
482    }
483
484    @Override
485    public BaseWrapper value() {
486      if (getValues().size() != 1)
487        throw new Error("Access single value, but value count is "+getValues().size());
488      return getValues().get(0);
489    }
490
491  }
492
493  private class BaseWrapperMetaElement implements BaseWrapper {
494    private org.hl7.fhir.dstu3.elementmodel.Element element;
495    private String type;
496    private StructureDefinition structure;
497    private ElementDefinition definition;
498    private List<ElementDefinition> children;
499    private List<PropertyWrapper> list;
500
501    public BaseWrapperMetaElement(org.hl7.fhir.dstu3.elementmodel.Element element, String type, StructureDefinition structure, ElementDefinition definition) {
502      this.element = element;
503      this.type = type;
504      this.structure = structure;
505      this.definition = definition;
506    }
507
508    @Override
509    public Base getBase() throws UnsupportedEncodingException, IOException, FHIRException {
510      if (type == null || type.equals("Resource") || type.equals("BackboneElement") || type.equals("Element"))
511        return null;
512
513      if (element.hasElementProperty())
514        return null;
515      ByteArrayOutputStream xml = new ByteArrayOutputStream();
516      try {
517        new org.hl7.fhir.dstu3.elementmodel.XmlParser(context).compose(element, xml, OutputStyle.PRETTY, null);
518      } catch (Exception e) {
519        throw new FHIRException(e.getMessage(), e);
520      }
521      return context.newXmlParser().setOutputStyle(OutputStyle.PRETTY).parseType(xml.toString(), type);
522    }
523
524    @Override
525    public List<PropertyWrapper> children() {
526      if (list == null) {
527        children = ProfileUtilities.getChildList(structure, definition);
528        list = new ArrayList<NarrativeGenerator.PropertyWrapper>();
529        for (ElementDefinition child : children) {
530          List<org.hl7.fhir.dstu3.elementmodel.Element> elements = new ArrayList<org.hl7.fhir.dstu3.elementmodel.Element>();
531          String name = tail(child.getPath());
532          if (name.endsWith("[x]"))
533            element.getNamedChildrenWithWildcard(name, elements);
534          else
535            element.getNamedChildren(name, elements);
536          list.add(new PropertyWrapperMetaElement(structure, child, elements));
537        }
538      }
539      return list;
540    }
541
542    @Override
543    public PropertyWrapper getChildByName(String name) {
544      for (PropertyWrapper p : children())
545        if (p.getName().equals(name))
546          return p;
547      return null;
548    }
549
550  }
551  public class ResurceWrapperMetaElement implements ResourceWrapper {
552    private org.hl7.fhir.dstu3.elementmodel.Element wrapped;
553    private List<ResourceWrapper> list;
554    private List<PropertyWrapper> list2;
555    private StructureDefinition definition;
556    public ResurceWrapperMetaElement(org.hl7.fhir.dstu3.elementmodel.Element wrapped) {
557      this.wrapped = wrapped;
558      this.definition = wrapped.getProperty().getStructure();
559    }
560
561    @Override
562    public List<ResourceWrapper> getContained() {
563      if (list == null) {
564        List<org.hl7.fhir.dstu3.elementmodel.Element> children = wrapped.getChildrenByName("contained");
565        list = new ArrayList<NarrativeGenerator.ResourceWrapper>();
566        for (org.hl7.fhir.dstu3.elementmodel.Element e : children) {
567          list.add(new ResurceWrapperMetaElement(e));
568        }
569      }
570      return list;
571    }
572
573    @Override
574    public String getId() {
575      return wrapped.getNamedChildValue("id");
576    }
577
578    @Override
579    public XhtmlNode getNarrative() throws FHIRFormatError, IOException, FHIRException {
580      org.hl7.fhir.dstu3.elementmodel.Element txt = wrapped.getNamedChild("text");
581      if (txt == null)
582        return null;
583      org.hl7.fhir.dstu3.elementmodel.Element div = txt.getNamedChild("div");
584      if (div == null)
585        return null;
586      else
587        return div.getXhtml();
588    }
589
590    @Override
591    public String getName() {
592      return wrapped.getName();
593    }
594
595    @Override
596    public List<PropertyWrapper> children() {
597      if (list2 == null) {
598        List<ElementDefinition> children = ProfileUtilities.getChildList(definition, definition.getSnapshot().getElement().get(0));
599        list2 = new ArrayList<NarrativeGenerator.PropertyWrapper>();
600        for (ElementDefinition child : children) {
601          List<org.hl7.fhir.dstu3.elementmodel.Element> elements = new ArrayList<org.hl7.fhir.dstu3.elementmodel.Element>();
602          if (child.getPath().endsWith("[x]"))
603            wrapped.getNamedChildrenWithWildcard(tail(child.getPath()), elements);
604          else
605            wrapped.getNamedChildren(tail(child.getPath()), elements);
606          list2.add(new PropertyWrapperMetaElement(definition, child, elements));
607        }
608      }
609      return list2;
610    }
611  }
612
613  private class PropertyWrapperMetaElement implements PropertyWrapper {
614
615    private StructureDefinition structure;
616    private ElementDefinition definition;
617    private List<org.hl7.fhir.dstu3.elementmodel.Element> values;
618    private List<BaseWrapper> list;
619
620    public PropertyWrapperMetaElement(StructureDefinition structure, ElementDefinition definition, List<org.hl7.fhir.dstu3.elementmodel.Element> values) {
621      this.structure = structure;
622      this.definition = definition;
623      this.values = values;
624    }
625
626    @Override
627    public String getName() {
628      return tail(definition.getPath());
629    }
630
631    @Override
632    public boolean hasValues() {
633      return values.size() > 0;
634    }
635
636    @Override
637    public List<BaseWrapper> getValues() {
638      if (list == null) {
639        list = new ArrayList<NarrativeGenerator.BaseWrapper>();
640        for (org.hl7.fhir.dstu3.elementmodel.Element e : values)
641          list.add(new BaseWrapperMetaElement(e, e.fhirType(), structure, definition));
642      }
643      return list;
644    }
645
646    @Override
647    public String getTypeCode() {
648      throw new Error("todo");
649    }
650
651    @Override
652    public String getDefinition() {
653      throw new Error("todo");
654    }
655
656    @Override
657    public int getMinCardinality() {
658      throw new Error("todo");
659      //      return definition.getMin();
660    }
661
662    @Override
663    public int getMaxCardinality() {
664      throw new Error("todo");
665    }
666
667    @Override
668    public StructureDefinition getStructure() {
669      return structure;
670    }
671
672    @Override
673    public BaseWrapper value() {
674      if (getValues().size() != 1)
675        throw new Error("Access single value, but value count is "+getValues().size());
676      return getValues().get(0);
677    }
678
679  }
680
681  private class ResurceWrapperElement implements ResourceWrapper {
682
683    private Element wrapped;
684    private StructureDefinition definition;
685    private List<ResourceWrapper> list;
686    private List<PropertyWrapper> list2;
687
688    public ResurceWrapperElement(Element wrapped, StructureDefinition definition) {
689      this.wrapped = wrapped;
690      this.definition = definition;
691    }
692
693    @Override
694    public List<ResourceWrapper> getContained() {
695      if (list == null) {
696        List<Element> children = new ArrayList<Element>();
697        XMLUtil.getNamedChildren(wrapped, "contained", children);
698        list = new ArrayList<NarrativeGenerator.ResourceWrapper>();
699        for (Element e : children) {
700          Element c = XMLUtil.getFirstChild(e);
701          list.add(new ResurceWrapperElement(c, context.fetchTypeDefinition(c.getNodeName())));
702        }
703      }
704      return list;
705    }
706
707    @Override
708    public String getId() {
709      return XMLUtil.getNamedChildValue(wrapped, "id");
710    }
711
712    @Override
713    public XhtmlNode getNarrative() throws FHIRFormatError, IOException, FHIRException {
714      Element txt = XMLUtil.getNamedChild(wrapped, "text");
715      if (txt == null)
716        return null;
717      Element div = XMLUtil.getNamedChild(txt, "div");
718      if (div == null)
719        return null;
720      try {
721                        return new XhtmlParser().parse(new XmlGenerator().generate(div), "div");
722                } catch (org.hl7.fhir.exceptions.FHIRFormatError e) {
723                        throw new FHIRFormatError(e.getMessage(), e);
724                } catch (org.hl7.fhir.exceptions.FHIRException e) {
725                        throw new FHIRException(e.getMessage(), e);
726                }
727    }
728
729    @Override
730    public String getName() {
731      return wrapped.getNodeName();
732    }
733
734    @Override
735    public List<PropertyWrapper> children() {
736      if (list2 == null) {
737        List<ElementDefinition> children = ProfileUtilities.getChildList(definition, definition.getSnapshot().getElement().get(0));
738        list2 = new ArrayList<NarrativeGenerator.PropertyWrapper>();
739        for (ElementDefinition child : children) {
740          List<Element> elements = new ArrayList<Element>();
741          XMLUtil.getNamedChildrenWithWildcard(wrapped, tail(child.getPath()), elements);
742          list2.add(new PropertyWrapperElement(definition, child, elements));
743        }
744      }
745      return list2;
746    }
747  }
748
749  private class PropertyWrapperDirect implements PropertyWrapper {
750    private Property wrapped;
751    private List<BaseWrapper> list;
752
753    private PropertyWrapperDirect(Property wrapped) {
754      super();
755      if (wrapped == null)
756        throw new Error("wrapped == null");
757      this.wrapped = wrapped;
758    }
759
760    @Override
761    public String getName() {
762      return wrapped.getName();
763    }
764
765    @Override
766    public boolean hasValues() {
767      return wrapped.hasValues();
768    }
769
770    @Override
771    public List<BaseWrapper> getValues() {
772      if (list == null) {
773        list = new ArrayList<NarrativeGenerator.BaseWrapper>();
774        for (Base b : wrapped.getValues())
775          list.add(b == null ? null : new BaseWrapperDirect(b));
776      }
777      return list;
778    }
779
780    @Override
781    public String getTypeCode() {
782      return wrapped.getTypeCode();
783    }
784
785    @Override
786    public String getDefinition() {
787      return wrapped.getDefinition();
788    }
789
790    @Override
791    public int getMinCardinality() {
792      return wrapped.getMinCardinality();
793    }
794
795    @Override
796    public int getMaxCardinality() {
797      return wrapped.getMinCardinality();
798    }
799
800    @Override
801    public StructureDefinition getStructure() {
802      return wrapped.getStructure();
803    }
804
805    @Override
806    public BaseWrapper value() {
807      if (getValues().size() != 1)
808        throw new Error("Access single value, but value count is "+getValues().size());
809      return getValues().get(0);
810    }
811  }
812
813  private class BaseWrapperDirect implements BaseWrapper {
814    private Base wrapped;
815    private List<PropertyWrapper> list;
816
817    private BaseWrapperDirect(Base wrapped) {
818      super();
819      if (wrapped == null)
820        throw new Error("wrapped == null");
821      this.wrapped = wrapped;
822    }
823
824    @Override
825    public Base getBase() {
826      return wrapped;
827    }
828
829    @Override
830    public List<PropertyWrapper> children() {
831      if (list == null) {
832        list = new ArrayList<NarrativeGenerator.PropertyWrapper>();
833        for (Property p : wrapped.children())
834          list.add(new PropertyWrapperDirect(p));
835      }
836      return list;
837
838    }
839
840    @Override
841    public PropertyWrapper getChildByName(String name) {
842      Property p = wrapped.getChildByName(name);
843      if (p == null)
844        return null;
845      else
846        return new PropertyWrapperDirect(p);
847    }
848
849  }
850
851  public class ResourceWrapperDirect implements ResourceWrapper {
852    private Resource wrapped;
853
854    public ResourceWrapperDirect(Resource wrapped) {
855      super();
856      if (wrapped == null)
857        throw new Error("wrapped == null");
858      this.wrapped = wrapped;
859    }
860
861    @Override
862    public List<ResourceWrapper> getContained() {
863      List<ResourceWrapper> list = new ArrayList<NarrativeGenerator.ResourceWrapper>();
864      if (wrapped instanceof DomainResource) {
865        DomainResource dr = (DomainResource) wrapped;
866        for (Resource c : dr.getContained()) {
867          list.add(new ResourceWrapperDirect(c));
868        }
869      }
870      return list;
871    }
872
873    @Override
874    public String getId() {
875      return wrapped.getId();
876    }
877
878    @Override
879    public XhtmlNode getNarrative() {
880      if (wrapped instanceof DomainResource) {
881        DomainResource dr = (DomainResource) wrapped;
882        if (dr.hasText() && dr.getText().hasDiv())
883          return dr.getText().getDiv();
884      }
885      return null;
886    }
887
888    @Override
889    public String getName() {
890      return wrapped.getResourceType().toString();
891    }
892
893    @Override
894    public List<PropertyWrapper> children() {
895      List<PropertyWrapper> list = new ArrayList<PropertyWrapper>();
896      for (Property c : wrapped.children())
897        list.add(new PropertyWrapperDirect(c));
898      return list;
899    }
900  }
901
902  public static class ResourceWithReference {
903
904    private String reference;
905    private ResourceWrapper resource;
906
907    public ResourceWithReference(String reference, ResourceWrapper resource) {
908      this.reference = reference;
909      this.resource = resource;
910    }
911
912    public String getReference() {
913      return reference;
914    }
915
916    public ResourceWrapper getResource() {
917      return resource;
918    }
919  }
920
921  private String prefix;
922  private IWorkerContext context;
923  private String basePath;
924  private String tooCostlyNoteEmpty;
925  private String tooCostlyNoteNotEmpty;
926  private IReferenceResolver resolver;
927  private int headerLevelContext;
928
929  public NarrativeGenerator(String prefix, String basePath, IWorkerContext context) {
930    super();
931    this.prefix = prefix;
932    this.context = context;
933    this.basePath = basePath;
934  }
935
936  public NarrativeGenerator(String prefix, String basePath, IWorkerContext context, IReferenceResolver resolver) {
937    super();
938    this.prefix = prefix;
939    this.context = context;
940    this.basePath = basePath;
941    this.resolver = resolver;
942  }
943
944
945  public int getHeaderLevelContext() {
946    return headerLevelContext;
947  }
948
949  public NarrativeGenerator setHeaderLevelContext(int headerLevelContext) {
950    this.headerLevelContext = headerLevelContext;
951    return this;
952  }
953
954  public String getTooCostlyNoteEmpty() {
955    return tooCostlyNoteEmpty;
956  }
957
958
959  public NarrativeGenerator setTooCostlyNoteEmpty(String tooCostlyNoteEmpty) {
960    this.tooCostlyNoteEmpty = tooCostlyNoteEmpty;
961    return this;
962  }
963
964
965  public String getTooCostlyNoteNotEmpty() {
966    return tooCostlyNoteNotEmpty;
967  }
968
969
970  public NarrativeGenerator setTooCostlyNoteNotEmpty(String tooCostlyNoteNotEmpty) {
971    this.tooCostlyNoteNotEmpty = tooCostlyNoteNotEmpty;
972    return this;
973  }
974
975
976  // dom based version, for build program
977  public String generate(Element doc) throws IOException, org.hl7.fhir.exceptions.FHIRException {
978    return generate(null, doc);
979  }
980  public String generate(ResourceContext rcontext, Element doc) throws IOException, org.hl7.fhir.exceptions.FHIRException {
981    if (rcontext == null)
982      rcontext = new ResourceContext(null, doc);
983    String rt = "http://hl7.org/fhir/StructureDefinition/"+doc.getNodeName();
984    StructureDefinition p = context.fetchResource(StructureDefinition.class, rt);
985    return generateByProfile(doc, p, true);
986  }
987
988  // dom based version, for build program
989  public String generate(org.hl7.fhir.dstu3.elementmodel.Element er, boolean showCodeDetails) throws IOException, DefinitionException {
990    return generate(null, er, showCodeDetails);
991  }
992  
993  public String generate(ResourceContext rcontext, org.hl7.fhir.dstu3.elementmodel.Element er, boolean showCodeDetails) throws IOException, DefinitionException {
994    if (rcontext == null)
995      rcontext = new ResourceContext(null, er);
996    
997    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
998    x.para().b().tx("Generated Narrative"+(showCodeDetails ? " with Details" : ""));
999    try {
1000      ResurceWrapperMetaElement resw = new ResurceWrapperMetaElement(er);
1001      BaseWrapperMetaElement base = new BaseWrapperMetaElement(er, null, er.getProperty().getStructure(), er.getProperty().getDefinition());
1002      base.children();
1003      generateByProfile(resw, er.getProperty().getStructure(), base, er.getProperty().getStructure().getSnapshot().getElement(), er.getProperty().getDefinition(), base.children, x, er.fhirType(), showCodeDetails);
1004
1005    } catch (Exception e) {
1006      e.printStackTrace();
1007      x.para().b().setAttribute("style", "color: maroon").tx("Exception generating Narrative: "+e.getMessage());
1008    }
1009    inject(er, x,  NarrativeStatus.GENERATED);
1010    return new XhtmlComposer(XhtmlComposer.XML).compose(x);
1011  }
1012
1013  private boolean generateByProfile(DomainResource r, StructureDefinition profile, boolean showCodeDetails) {
1014    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
1015    x.para().b().tx("Generated Narrative"+(showCodeDetails ? " with Details" : ""));
1016    try {
1017      generateByProfile(r, profile, r, profile.getSnapshot().getElement(), profile.getSnapshot().getElement().get(0), getChildrenForPath(profile.getSnapshot().getElement(), r.getResourceType().toString()), x, r.getResourceType().toString(), showCodeDetails);
1018    } catch (Exception e) {
1019      e.printStackTrace();
1020      x.para().b().setAttribute("style", "color: maroon").tx("Exception generating Narrative: "+e.getMessage());
1021    }
1022    inject(r, x,  NarrativeStatus.GENERATED);
1023    return true;
1024  }
1025
1026  private String generateByProfile(Element er, StructureDefinition profile, boolean showCodeDetails) throws IOException, org.hl7.fhir.exceptions.FHIRException {
1027    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
1028    x.para().b().tx("Generated Narrative"+(showCodeDetails ? " with Details" : ""));
1029    try {
1030      generateByProfile(er, profile, er, profile.getSnapshot().getElement(), profile.getSnapshot().getElement().get(0), getChildrenForPath(profile.getSnapshot().getElement(), er.getLocalName()), x, er.getLocalName(), showCodeDetails);
1031    } catch (Exception e) {
1032      e.printStackTrace();
1033      x.para().b().setAttribute("style", "color: maroon").tx("Exception generating Narrative: "+e.getMessage());
1034    }
1035    inject(er, x,  NarrativeStatus.GENERATED);
1036    return new XhtmlComposer(XhtmlComposer.XML).compose(x);
1037  }
1038
1039  private void generateByProfile(Element eres, StructureDefinition profile, Element ee, List<ElementDefinition> allElements, ElementDefinition defn, List<ElementDefinition> children,  XhtmlNode x, String path, boolean showCodeDetails) throws FHIRException, UnsupportedEncodingException, IOException {
1040
1041    ResurceWrapperElement resw = new ResurceWrapperElement(eres, profile);
1042    BaseWrapperElement base = new BaseWrapperElement(ee, null, profile, profile.getSnapshot().getElement().get(0));
1043    generateByProfile(resw, profile, base, allElements, defn, children, x, path, showCodeDetails);
1044  }
1045
1046
1047  private void generateByProfile(Resource res, StructureDefinition profile, Base e, List<ElementDefinition> allElements, ElementDefinition defn, List<ElementDefinition> children,  XhtmlNode x, String path, boolean showCodeDetails) throws FHIRException, UnsupportedEncodingException, IOException {
1048    generateByProfile(new ResourceWrapperDirect(res), profile, new BaseWrapperDirect(e), allElements, defn, children, x, path, showCodeDetails);
1049  }
1050
1051  private void generateByProfile(ResourceWrapper res, StructureDefinition profile, BaseWrapper e, List<ElementDefinition> allElements, ElementDefinition defn, List<ElementDefinition> children,  XhtmlNode x, String path, boolean showCodeDetails) throws FHIRException, UnsupportedEncodingException, IOException {
1052    if (children.isEmpty()) {
1053      renderLeaf(res, e, defn, x, false, showCodeDetails, readDisplayHints(defn), path);
1054    } else {
1055      for (PropertyWrapper p : splitExtensions(profile, e.children())) {
1056        if (p.hasValues()) {
1057          ElementDefinition child = getElementDefinition(children, path+"."+p.getName(), p);
1058          if (child != null) {
1059            Map<String, String> displayHints = readDisplayHints(child);
1060            if (!exemptFromRendering(child)) {
1061              List<ElementDefinition> grandChildren = getChildrenForPath(allElements, path+"."+p.getName());
1062            filterGrandChildren(grandChildren, path+"."+p.getName(), p);
1063              if (p.getValues().size() > 0 && child != null) {
1064                if (isPrimitive(child)) {
1065                  XhtmlNode para = x.para();
1066                  String name = p.getName();
1067                  if (name.endsWith("[x]"))
1068                    name = name.substring(0, name.length() - 3);
1069                  if (showCodeDetails || !isDefaultValue(displayHints, p.getValues())) {
1070                    para.b().addText(name);
1071                    para.tx(": ");
1072                    if (renderAsList(child) && p.getValues().size() > 1) {
1073                      XhtmlNode list = x.ul();
1074                      for (BaseWrapper v : p.getValues())
1075                        renderLeaf(res, v, child, list.li(), false, showCodeDetails, displayHints, path);
1076                    } else {
1077                      boolean first = true;
1078                      for (BaseWrapper v : p.getValues()) {
1079                        if (first)
1080                          first = false;
1081                        else
1082                          para.tx(", ");
1083                        renderLeaf(res, v, child, para, false, showCodeDetails, displayHints, path);
1084                      }
1085                    }
1086                  }
1087                } else if (canDoTable(path, p, grandChildren)) {
1088                  x.addTag(getHeader()).addText(Utilities.capitalize(Utilities.camelCase(Utilities.pluralizeMe(p.getName()))));
1089                  XhtmlNode tbl = x.table( "grid");
1090                  XhtmlNode tr = tbl.tr();
1091                  tr.td().tx("-"); // work around problem with empty table rows
1092                  addColumnHeadings(tr, grandChildren);
1093                  for (BaseWrapper v : p.getValues()) {
1094                    if (v != null) {
1095                      tr = tbl.tr();
1096                      tr.td().tx("*"); // work around problem with empty table rows
1097                      addColumnValues(res, tr, grandChildren, v, showCodeDetails, displayHints, path);
1098                    }
1099                  }
1100                } else {
1101                  for (BaseWrapper v : p.getValues()) {
1102                    if (v != null) {
1103                      XhtmlNode bq = x.addTag("blockquote");
1104                      bq.para().b().addText(p.getName());
1105                      generateByProfile(res, profile, v, allElements, child, grandChildren, bq, path+"."+p.getName(), showCodeDetails);
1106                    }
1107                  }
1108                }
1109              }
1110            }
1111          }
1112        }
1113      }
1114    }
1115  }
1116
1117  private String getHeader() {
1118    int i = 3;
1119    while (i <= headerLevelContext)
1120      i++;
1121    if (i > 6)
1122      i = 6;
1123    return "h"+Integer.toString(i);
1124  }
1125
1126  private void filterGrandChildren(List<ElementDefinition> grandChildren,  String string, PropertyWrapper prop) {
1127        List<ElementDefinition> toRemove = new ArrayList<ElementDefinition>();
1128        toRemove.addAll(grandChildren);
1129        for (BaseWrapper b : prop.getValues()) {
1130        List<ElementDefinition> list = new ArrayList<ElementDefinition>();
1131                for (ElementDefinition ed : toRemove) {
1132                        PropertyWrapper p = b.getChildByName(tail(ed.getPath()));
1133                        if (p != null && p.hasValues())
1134                                list.add(ed);
1135                }
1136                toRemove.removeAll(list);
1137        }
1138        grandChildren.removeAll(toRemove);
1139  }
1140
1141  private List<PropertyWrapper> splitExtensions(StructureDefinition profile, List<PropertyWrapper> children) throws UnsupportedEncodingException, IOException, FHIRException {
1142    List<PropertyWrapper> results = new ArrayList<PropertyWrapper>();
1143    Map<String, PropertyWrapper> map = new HashMap<String, PropertyWrapper>();
1144    for (PropertyWrapper p : children)
1145      if (p.getName().equals("extension") || p.getName().equals("modifierExtension")) {
1146        // we're going to split these up, and create a property for each url
1147        if (p.hasValues()) {
1148          for (BaseWrapper v : p.getValues()) {
1149            Extension ex  = (Extension) v.getBase();
1150            String url = ex.getUrl();
1151            StructureDefinition ed = context.fetchResource(StructureDefinition.class, url);
1152            if (p.getName().equals("modifierExtension") && ed == null)
1153              throw new DefinitionException("Unknown modifier extension "+url);
1154            PropertyWrapper pe = map.get(p.getName()+"["+url+"]");
1155            if (pe == null) {
1156              if (ed == null) {
1157                if (url.startsWith("http://hl7.org/fhir"))
1158                  throw new DefinitionException("unknown extension "+url);
1159                // System.out.println("unknown extension "+url);
1160                pe = new PropertyWrapperDirect(new Property(p.getName()+"["+url+"]", p.getTypeCode(), p.getDefinition(), p.getMinCardinality(), p.getMaxCardinality(), ex));
1161              } else {
1162                ElementDefinition def = ed.getSnapshot().getElement().get(0);
1163                pe = new PropertyWrapperDirect(new Property(p.getName()+"["+url+"]", "Extension", def.getDefinition(), def.getMin(), def.getMax().equals("*") ? Integer.MAX_VALUE : Integer.parseInt(def.getMax()), ex));
1164                ((PropertyWrapperDirect) pe).wrapped.setStructure(ed);
1165              }
1166              results.add(pe);
1167            } else
1168              pe.getValues().add(v);
1169          }
1170        }
1171      } else
1172        results.add(p);
1173    return results;
1174  }
1175
1176  @SuppressWarnings("rawtypes")
1177  private boolean isDefaultValue(Map<String, String> displayHints, List<BaseWrapper> list) throws UnsupportedEncodingException, IOException, FHIRException {
1178    if (list.size() != 1)
1179      return false;
1180    if (list.get(0).getBase() instanceof PrimitiveType)
1181      return isDefault(displayHints, (PrimitiveType) list.get(0).getBase());
1182    else
1183      return false;
1184  }
1185
1186  private boolean isDefault(Map<String, String> displayHints, PrimitiveType primitiveType) {
1187    String v = primitiveType.asStringValue();
1188    if (!Utilities.noString(v) && displayHints.containsKey("default") && v.equals(displayHints.get("default")))
1189        return true;
1190    return false;
1191  }
1192
1193  private boolean exemptFromRendering(ElementDefinition child) {
1194    if (child == null)
1195      return false;
1196    if ("Composition.subject".equals(child.getPath()))
1197      return true;
1198    if ("Composition.section".equals(child.getPath()))
1199      return true;
1200    return false;
1201  }
1202
1203  private boolean renderAsList(ElementDefinition child) {
1204    if (child.getType().size() == 1) {
1205      String t = child.getType().get(0).getCode();
1206      if (t.equals("Address") || t.equals("Reference"))
1207        return true;
1208    }
1209    return false;
1210  }
1211
1212  private void addColumnHeadings(XhtmlNode tr, List<ElementDefinition> grandChildren) {
1213    for (ElementDefinition e : grandChildren)
1214      tr.td().b().addText(Utilities.capitalize(tail(e.getPath())));
1215  }
1216
1217  private void addColumnValues(ResourceWrapper res, XhtmlNode tr, List<ElementDefinition> grandChildren, BaseWrapper v, boolean showCodeDetails, Map<String, String> displayHints, String path) throws FHIRException, UnsupportedEncodingException, IOException {
1218    for (ElementDefinition e : grandChildren) {
1219      PropertyWrapper p = v.getChildByName(e.getPath().substring(e.getPath().lastIndexOf(".")+1));
1220      if (p == null || p.getValues().size() == 0 || p.getValues().get(0) == null)
1221        tr.td().tx(" ");
1222      else
1223        renderLeaf(res, p.getValues().get(0), e, tr.td(), false, showCodeDetails, displayHints, path);
1224    }
1225  }
1226
1227  private String tail(String path) {
1228    return path.substring(path.lastIndexOf(".")+1);
1229  }
1230
1231  private boolean canDoTable(String path, PropertyWrapper p, List<ElementDefinition> grandChildren) {
1232    for (ElementDefinition e : grandChildren) {
1233      List<PropertyWrapper> values = getValues(path, p, e);
1234      if (values.size() > 1 || !isPrimitive(e) || !canCollapse(e))
1235        return false;
1236    }
1237    return true;
1238  }
1239
1240  private List<PropertyWrapper> getValues(String path, PropertyWrapper p, ElementDefinition e) {
1241    List<PropertyWrapper> res = new ArrayList<PropertyWrapper>();
1242    for (BaseWrapper v : p.getValues()) {
1243      for (PropertyWrapper g : v.children()) {
1244        if ((path+"."+p.getName()+"."+g.getName()).equals(e.getPath()))
1245          res.add(p);
1246      }
1247    }
1248    return res;
1249  }
1250
1251  private boolean canCollapse(ElementDefinition e) {
1252    // we can collapse any data type
1253    return !e.getType().isEmpty();
1254  }
1255
1256  private boolean isPrimitive(ElementDefinition e) {
1257    //we can tell if e is a primitive because it has types
1258    if (e.getType().isEmpty())
1259      return false;
1260    if (e.getType().size() == 1 && isBase(e.getType().get(0).getCode()))
1261      return false;
1262    return true;
1263//    return !e.getType().isEmpty()
1264  }
1265
1266  private boolean isBase(String code) {
1267    return code.equals("Element") || code.equals("BackboneElement");
1268  }
1269
1270  private ElementDefinition getElementDefinition(List<ElementDefinition> elements, String path, PropertyWrapper p) {
1271    for (ElementDefinition element : elements)
1272      if (element.getPath().equals(path))
1273        return element;
1274    if (path.endsWith("\"]") && p.getStructure() != null)
1275      return p.getStructure().getSnapshot().getElement().get(0);
1276    return null;
1277  }
1278
1279  private void renderLeaf(ResourceWrapper res, BaseWrapper ew, ElementDefinition defn, XhtmlNode x, boolean title, boolean showCodeDetails, Map<String, String> displayHints, String path) throws FHIRException, UnsupportedEncodingException, IOException {
1280    if (ew == null)
1281      return;
1282
1283
1284    Base e = ew.getBase();
1285
1286    if (e instanceof StringType)
1287      x.addText(((StringType) e).getValue());
1288    else if (e instanceof CodeType)
1289      x.addText(((CodeType) e).getValue());
1290    else if (e instanceof IdType)
1291      x.addText(((IdType) e).getValue());
1292    else if (e instanceof Extension)
1293      return;
1294    else if (e instanceof InstantType)
1295      x.addText(((InstantType) e).toHumanDisplay());
1296    else if (e instanceof DateTimeType)
1297      x.addText(((DateTimeType) e).toHumanDisplay());
1298    else if (e instanceof Base64BinaryType)
1299      x.addText(new Base64().encodeAsString(((Base64BinaryType) e).getValue()));
1300    else if (e instanceof org.hl7.fhir.dstu3.model.DateType)
1301      x.addText(((org.hl7.fhir.dstu3.model.DateType) e).toHumanDisplay());
1302    else if (e instanceof Enumeration) {
1303      Object ev = ((Enumeration<?>) e).getValue();
1304                        x.addText(ev == null ? "" : ev.toString()); // todo: look up a display name if there is one
1305    } else if (e instanceof BooleanType)
1306      x.addText(((BooleanType) e).getValue().toString());
1307    else if (e instanceof CodeableConcept) {
1308      renderCodeableConcept((CodeableConcept) e, x, showCodeDetails);
1309    } else if (e instanceof Coding) {
1310      renderCoding((Coding) e, x, showCodeDetails);
1311    } else if (e instanceof Annotation) {
1312      renderAnnotation((Annotation) e, x);
1313    } else if (e instanceof Identifier) {
1314      renderIdentifier((Identifier) e, x);
1315    } else if (e instanceof org.hl7.fhir.dstu3.model.IntegerType) {
1316      x.addText(Integer.toString(((org.hl7.fhir.dstu3.model.IntegerType) e).getValue()));
1317    } else if (e instanceof org.hl7.fhir.dstu3.model.DecimalType) {
1318      x.addText(((org.hl7.fhir.dstu3.model.DecimalType) e).getValue().toString());
1319    } else if (e instanceof HumanName) {
1320      renderHumanName((HumanName) e, x);
1321    } else if (e instanceof SampledData) {
1322      renderSampledData((SampledData) e, x);
1323    } else if (e instanceof Address) {
1324      renderAddress((Address) e, x);
1325    } else if (e instanceof ContactPoint) {
1326      renderContactPoint((ContactPoint) e, x);
1327    } else if (e instanceof UriType) {
1328      renderUri((UriType) e, x);
1329    } else if (e instanceof Timing) {
1330      renderTiming((Timing) e, x);
1331    } else if (e instanceof Range) {
1332      renderRange((Range) e, x);
1333    } else if (e instanceof Quantity) {
1334      renderQuantity((Quantity) e, x, showCodeDetails);
1335    } else if (e instanceof Ratio) {
1336      renderQuantity(((Ratio) e).getNumerator(), x, showCodeDetails);
1337      x.tx("/");
1338      renderQuantity(((Ratio) e).getDenominator(), x, showCodeDetails);
1339    } else if (e instanceof Period) {
1340      Period p = (Period) e;
1341      x.addText(!p.hasStart() ? "??" : p.getStartElement().toHumanDisplay());
1342      x.tx(" --> ");
1343      x.addText(!p.hasEnd() ? "(ongoing)" : p.getEndElement().toHumanDisplay());
1344    } else if (e instanceof Reference) {
1345      Reference r = (Reference) e;
1346      XhtmlNode c = x;
1347      ResourceWithReference tr = null;
1348      if (r.hasReferenceElement()) {
1349        tr = resolveReference(res, r.getReference());
1350        if (!r.getReference().startsWith("#")) {
1351          if (tr != null && tr.getReference() != null)
1352            c = x.ah(tr.getReference());
1353          else
1354            c = x.ah(r.getReference());
1355        }
1356      }
1357      // what to display: if text is provided, then that. if the reference was resolved, then show the generated narrative
1358      if (r.hasDisplayElement()) {
1359        c.addText(r.getDisplay());
1360        if (tr != null && tr.getResource() != null) {
1361          c.tx(". Generated Summary: ");
1362          generateResourceSummary(c, tr.getResource(), true, r.getReference().startsWith("#"));
1363        }
1364      } else if (tr != null && tr.getResource() != null) {
1365        generateResourceSummary(c, tr.getResource(), r.getReference().startsWith("#"), r.getReference().startsWith("#"));
1366      } else {
1367        c.addText(r.getReference());
1368      }
1369    } else if (e instanceof Resource) {
1370      return;
1371    } else if (e instanceof ElementDefinition) {
1372      x.tx("todo-bundle");
1373    } else if (e != null && !(e instanceof Attachment) && !(e instanceof Narrative) && !(e instanceof Meta)) {
1374      StructureDefinition sd = context.fetchTypeDefinition(e.fhirType());
1375      if (sd == null)
1376        throw new NotImplementedException("type "+e.getClass().getName()+" not handled yet, and no structure found");
1377      else
1378        generateByProfile(res, sd, ew, sd.getSnapshot().getElement(), sd.getSnapshot().getElementFirstRep(),
1379            getChildrenForPath(sd.getSnapshot().getElement(), sd.getSnapshot().getElementFirstRep().getPath()), x, path, showCodeDetails);
1380    }
1381  }
1382
1383  private boolean displayLeaf(ResourceWrapper res, BaseWrapper ew, ElementDefinition defn, XhtmlNode x, String name, boolean showCodeDetails) throws FHIRException, UnsupportedEncodingException, IOException {
1384    if (ew == null)
1385      return false;
1386    Base e = ew.getBase();
1387    if (e == null)
1388      return false;
1389
1390    Map<String, String> displayHints = readDisplayHints(defn);
1391
1392    if (name.endsWith("[x]"))
1393      name = name.substring(0, name.length() - 3);
1394
1395    if (!showCodeDetails && e instanceof PrimitiveType && isDefault(displayHints, ((PrimitiveType) e)))
1396        return false;
1397
1398    if (e instanceof StringType) {
1399      x.addText(name+": "+((StringType) e).getValue());
1400      return true;
1401    } else if (e instanceof CodeType) {
1402      x.addText(name+": "+((CodeType) e).getValue());
1403      return true;
1404    } else if (e instanceof IdType) {
1405      x.addText(name+": "+((IdType) e).getValue());
1406      return true;
1407    } else if (e instanceof UriType) {
1408      x.addText(name+": "+((UriType) e).getValue());
1409      return true;
1410    } else if (e instanceof DateTimeType) {
1411      x.addText(name+": "+((DateTimeType) e).toHumanDisplay());
1412      return true;
1413    } else if (e instanceof InstantType) {
1414      x.addText(name+": "+((InstantType) e).toHumanDisplay());
1415      return true;
1416    } else if (e instanceof Extension) {
1417//      x.tx("Extensions: todo");
1418      return false;
1419    } else if (e instanceof org.hl7.fhir.dstu3.model.DateType) {
1420      x.addText(name+": "+((org.hl7.fhir.dstu3.model.DateType) e).toHumanDisplay());
1421      return true;
1422    } else if (e instanceof Enumeration) {
1423      x.addText(((Enumeration<?>) e).getValue().toString()); // todo: look up a display name if there is one
1424      return true;
1425    } else if (e instanceof BooleanType) {
1426      if (((BooleanType) e).getValue()) {
1427        x.addText(name);
1428          return true;
1429      }
1430    } else if (e instanceof CodeableConcept) {
1431      renderCodeableConcept((CodeableConcept) e, x, showCodeDetails);
1432      return true;
1433    } else if (e instanceof Coding) {
1434      renderCoding((Coding) e, x, showCodeDetails);
1435      return true;
1436    } else if (e instanceof Annotation) {
1437      renderAnnotation((Annotation) e, x, showCodeDetails);
1438      return true;
1439    } else if (e instanceof org.hl7.fhir.dstu3.model.IntegerType) {
1440      x.addText(Integer.toString(((org.hl7.fhir.dstu3.model.IntegerType) e).getValue()));
1441      return true;
1442    } else if (e instanceof org.hl7.fhir.dstu3.model.DecimalType) {
1443      x.addText(((org.hl7.fhir.dstu3.model.DecimalType) e).getValue().toString());
1444      return true;
1445    } else if (e instanceof Identifier) {
1446      renderIdentifier((Identifier) e, x);
1447      return true;
1448    } else if (e instanceof HumanName) {
1449      renderHumanName((HumanName) e, x);
1450      return true;
1451    } else if (e instanceof SampledData) {
1452      renderSampledData((SampledData) e, x);
1453      return true;
1454    } else if (e instanceof Address) {
1455      renderAddress((Address) e, x);
1456      return true;
1457    } else if (e instanceof ContactPoint) {
1458      renderContactPoint((ContactPoint) e, x);
1459      return true;
1460    } else if (e instanceof Timing) {
1461      renderTiming((Timing) e, x);
1462      return true;
1463    } else if (e instanceof Quantity) {
1464      renderQuantity((Quantity) e, x, showCodeDetails);
1465      return true;
1466    } else if (e instanceof Ratio) {
1467      renderQuantity(((Ratio) e).getNumerator(), x, showCodeDetails);
1468      x.tx("/");
1469      renderQuantity(((Ratio) e).getDenominator(), x, showCodeDetails);
1470      return true;
1471    } else if (e instanceof Period) {
1472      Period p = (Period) e;
1473      x.addText(name+": ");
1474      x.addText(!p.hasStart() ? "??" : p.getStartElement().toHumanDisplay());
1475      x.tx(" --> ");
1476      x.addText(!p.hasEnd() ? "(ongoing)" : p.getEndElement().toHumanDisplay());
1477      return true;
1478    } else if (e instanceof Reference) {
1479      Reference r = (Reference) e;
1480      if (r.hasDisplayElement())
1481        x.addText(r.getDisplay());
1482      else if (r.hasReferenceElement()) {
1483        ResourceWithReference tr = resolveReference(res, r.getReference());
1484        x.addText(tr == null ? r.getReference() : "????"); // getDisplayForReference(tr.getReference()));
1485      } else
1486        x.tx("??");
1487      return true;
1488    } else if (e instanceof Narrative) {
1489      return false;
1490    } else if (e instanceof Resource) {
1491      return false;
1492    } else if (e instanceof ContactDetail) {
1493      return false;
1494    } else if (e instanceof Range) {
1495      return false;
1496    } else if (e instanceof Meta) {
1497      return false;
1498    } else if (e instanceof Dosage) {
1499      return false;
1500    } else if (e instanceof Signature) {
1501      return false;
1502    } else if (e instanceof UsageContext) {
1503      return false;
1504    } else if (e instanceof ElementDefinition) {
1505      return false;
1506    } else if (!(e instanceof Attachment))
1507      throw new NotImplementedException("type "+e.getClass().getName()+" not handled yet");
1508    return false;
1509  }
1510
1511
1512  private Map<String, String> readDisplayHints(ElementDefinition defn) throws DefinitionException {
1513    Map<String, String> hints = new HashMap<String, String>();
1514    if (defn != null) {
1515      String displayHint = ToolingExtensions.getDisplayHint(defn);
1516      if (!Utilities.noString(displayHint)) {
1517        String[] list = displayHint.split(";");
1518        for (String item : list) {
1519          String[] parts = item.split(":");
1520          if (parts.length != 2)
1521            throw new DefinitionException("error reading display hint: '"+displayHint+"'");
1522          hints.put(parts[0].trim(), parts[1].trim());
1523        }
1524      }
1525    }
1526    return hints;
1527  }
1528
1529  public static String displayPeriod(Period p) {
1530    String s = !p.hasStart() ? "??" : p.getStartElement().toHumanDisplay();
1531    s = s + " --> ";
1532    return s + (!p.hasEnd() ? "(ongoing)" : p.getEndElement().toHumanDisplay());
1533  }
1534
1535  private void generateResourceSummary(XhtmlNode x, ResourceWrapper res, boolean textAlready, boolean showCodeDetails) throws FHIRException, UnsupportedEncodingException, IOException {
1536    if (!textAlready) {
1537      XhtmlNode div = res.getNarrative();
1538      if (div != null) {
1539        if (div.allChildrenAreText())
1540          x.addChildNodes(div.getChildNodes());
1541        if (div.getChildNodes().size() == 1 && div.getChildNodes().get(0).allChildrenAreText())
1542          x.addChildNodes(div.getChildNodes().get(0).getChildNodes());
1543      }
1544      x.tx("Generated Summary: ");
1545    }
1546    String path = res.getName();
1547    StructureDefinition profile = context.fetchResource(StructureDefinition.class, path);
1548    if (profile == null)
1549      x.tx("unknown resource " +path);
1550    else {
1551      boolean firstElement = true;
1552      boolean last = false;
1553      for (PropertyWrapper p : res.children()) {
1554        ElementDefinition child = getElementDefinition(profile.getSnapshot().getElement(), path+"."+p.getName(), p);
1555        if (p.getValues().size() > 0 && p.getValues().get(0) != null && child != null && isPrimitive(child) && includeInSummary(child)) {
1556          if (firstElement)
1557            firstElement = false;
1558          else if (last)
1559            x.tx("; ");
1560          boolean first = true;
1561          last = false;
1562          for (BaseWrapper v : p.getValues()) {
1563            if (first)
1564              first = false;
1565            else if (last)
1566              x.tx(", ");
1567            last = displayLeaf(res, v, child, x, p.getName(), showCodeDetails) || last;
1568          }
1569        }
1570      }
1571    }
1572  }
1573
1574
1575  private boolean includeInSummary(ElementDefinition child) {
1576    if (child.getIsModifier())
1577      return true;
1578    if (child.getMustSupport())
1579      return true;
1580    if (child.getType().size() == 1) {
1581      String t = child.getType().get(0).getCode();
1582      if (t.equals("Address") || t.equals("Contact") || t.equals("Reference") || t.equals("Uri"))
1583        return false;
1584    }
1585    return true;
1586  }
1587
1588  private ResourceWithReference resolveReference(ResourceWrapper res, String url) {
1589    if (url == null)
1590      return null;
1591    if (url.startsWith("#")) {
1592      for (ResourceWrapper r : res.getContained()) {
1593        if (r.getId().equals(url.substring(1)))
1594          return new ResourceWithReference(null, r);
1595      }
1596      return null;
1597    }
1598
1599    Resource ae = context.fetchResource(null, url);
1600    if (ae != null)
1601      return new ResourceWithReference(url, new ResourceWrapperDirect(ae));
1602    else if (resolver != null) {
1603      return resolver.resolve(url);
1604    } else
1605      return null;
1606  }
1607
1608  private void renderCodeableConcept(CodeableConcept cc, XhtmlNode x, boolean showCodeDetails) {
1609    String s = cc.getText();
1610    if (Utilities.noString(s)) {
1611      for (Coding c : cc.getCoding()) {
1612        if (c.hasDisplayElement()) {
1613          s = c.getDisplay();
1614          break;
1615        }
1616      }
1617    }
1618    if (Utilities.noString(s)) {
1619      // still? ok, let's try looking it up
1620      for (Coding c : cc.getCoding()) {
1621        if (c.hasCodeElement() && c.hasSystemElement()) {
1622          s = lookupCode(c.getSystem(), c.getCode());
1623          if (!Utilities.noString(s))
1624            break;
1625        }
1626      }
1627    }
1628
1629    if (Utilities.noString(s)) {
1630      if (cc.getCoding().isEmpty())
1631        s = "";
1632      else
1633        s = cc.getCoding().get(0).getCode();
1634    }
1635
1636    if (showCodeDetails) {
1637      x.addText(s+" ");
1638      XhtmlNode sp = x.span("background: LightGoldenRodYellow", null);
1639      sp.tx("(Details ");
1640      boolean first = true;
1641      for (Coding c : cc.getCoding()) {
1642        if (first) {
1643          sp.tx(": ");
1644          first = false;
1645        } else
1646          sp.tx("; ");
1647        sp.tx("{"+describeSystem(c.getSystem())+" code '"+c.getCode()+"' = '"+lookupCode(c.getSystem(), c.getCode())+(c.hasDisplay() ? "', given as '"+c.getDisplay()+"'}" : ""));
1648      }
1649      sp.tx(")");
1650    } else {
1651
1652    CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder();
1653    for (Coding c : cc.getCoding()) {
1654      if (c.hasCodeElement() && c.hasSystemElement()) {
1655        b.append("{"+c.getSystem()+" "+c.getCode()+"}");
1656      }
1657    }
1658
1659    x.span(null, "Codes: "+b.toString()).addText(s);
1660    }
1661  }
1662
1663  private void renderAnnotation(Annotation a, XhtmlNode x, boolean showCodeDetails) throws FHIRException {
1664    StringBuilder s = new StringBuilder();
1665    if (a.hasAuthor()) {
1666      s.append("Author: ");
1667
1668      if (a.hasAuthorReference())
1669        s.append(a.getAuthorReference().getReference());
1670      else if (a.hasAuthorStringType())
1671        s.append(a.getAuthorStringType().getValue());
1672    }
1673
1674
1675    if (a.hasTimeElement()) {
1676      if (s.length() > 0)
1677        s.append("; ");
1678
1679      s.append("Made: ").append(a.getTimeElement().toHumanDisplay());
1680    }
1681
1682    if (a.hasText()) {
1683      if (s.length() > 0)
1684        s.append("; ");
1685
1686      s.append("Annotation: ").append(a.getText());
1687    }
1688
1689    x.addText(s.toString());
1690  }
1691
1692  private void renderCoding(Coding c, XhtmlNode x, boolean showCodeDetails) {
1693    String s = "";
1694    if (c.hasDisplayElement())
1695      s = c.getDisplay();
1696    if (Utilities.noString(s))
1697      s = lookupCode(c.getSystem(), c.getCode());
1698
1699    if (Utilities.noString(s))
1700      s = c.getCode();
1701
1702    if (showCodeDetails) {
1703      x.addText(s+" (Details: "+describeSystem(c.getSystem())+" code "+c.getCode()+" = '"+lookupCode(c.getSystem(), c.getCode())+"', stated as '"+c.getDisplay()+"')");
1704    } else
1705      x.span(null, "{"+c.getSystem()+" "+c.getCode()+"}").addText(s);
1706  }
1707
1708  public static String describeSystem(String system) {
1709    if (system == null)
1710      return "[not stated]";
1711    if (system.equals("http://loinc.org"))
1712      return "LOINC";
1713    if (system.startsWith("http://snomed.info"))
1714      return "SNOMED CT";
1715    if (system.equals("http://www.nlm.nih.gov/research/umls/rxnorm"))
1716      return "RxNorm";
1717    if (system.equals("http://hl7.org/fhir/sid/icd-9"))
1718      return "ICD-9";
1719    if (system.equals("http://dicom.nema.org/resources/ontology/DCM"))
1720      return "DICOM";
1721    if (system.equals("http://unitsofmeasure.org"))
1722      return "UCUM";
1723
1724    return system;
1725  }
1726
1727  private String lookupCode(String system, String code) {
1728    ValidationResult t = context.validateCode(system, code, null);
1729
1730    if (t != null && t.getDisplay() != null)
1731        return t.getDisplay();
1732    else
1733      return code;
1734
1735  }
1736
1737  private ConceptDefinitionComponent findCode(String code, List<ConceptDefinitionComponent> list) {
1738    for (ConceptDefinitionComponent t : list) {
1739      if (code.equals(t.getCode()))
1740        return t;
1741      ConceptDefinitionComponent c = findCode(code, t.getConcept());
1742      if (c != null)
1743        return c;
1744    }
1745    return null;
1746  }
1747
1748  public String displayCodeableConcept(CodeableConcept cc) {
1749    String s = cc.getText();
1750    if (Utilities.noString(s)) {
1751      for (Coding c : cc.getCoding()) {
1752        if (c.hasDisplayElement()) {
1753          s = c.getDisplay();
1754          break;
1755        }
1756      }
1757    }
1758    if (Utilities.noString(s)) {
1759      // still? ok, let's try looking it up
1760      for (Coding c : cc.getCoding()) {
1761        if (c.hasCode() && c.hasSystem()) {
1762          s = lookupCode(c.getSystem(), c.getCode());
1763          if (!Utilities.noString(s))
1764            break;
1765        }
1766      }
1767    }
1768
1769    if (Utilities.noString(s)) {
1770      if (cc.getCoding().isEmpty())
1771        s = "";
1772      else
1773        s = cc.getCoding().get(0).getCode();
1774    }
1775    return s;
1776  }
1777
1778  private void renderIdentifier(Identifier ii, XhtmlNode x) {
1779    x.addText(displayIdentifier(ii));
1780  }
1781
1782  private void renderTiming(Timing s, XhtmlNode x) throws FHIRException {
1783    x.addText(displayTiming(s));
1784  }
1785
1786  private void renderQuantity(Quantity q, XhtmlNode x, boolean showCodeDetails) {
1787    if (q.hasComparator())
1788      x.addText(q.getComparator().toCode());
1789    x.addText(q.getValue().toString());
1790    if (q.hasUnit())
1791      x.tx(" "+q.getUnit());
1792    else if (q.hasCode())
1793      x.tx(" "+q.getCode());
1794    if (showCodeDetails && q.hasCode()) {
1795      x.span("background: LightGoldenRodYellow", null).tx(" (Details: "+describeSystem(q.getSystem())+" code "+q.getCode()+" = '"+lookupCode(q.getSystem(), q.getCode())+"')");
1796    }
1797  }
1798
1799  private void renderRange(Range q, XhtmlNode x) {
1800    if (q.hasLow())
1801      x.addText(q.getLow().getValue().toString());
1802    else
1803      x.tx("?");
1804    x.tx("-");
1805    if (q.hasHigh())
1806      x.addText(q.getHigh().getValue().toString());
1807    else
1808      x.tx("?");
1809    if (q.getLow().hasUnit())
1810      x.tx(" "+q.getLow().getUnit());
1811  }
1812
1813  public String displayRange(Range q) {
1814    StringBuilder b = new StringBuilder();
1815    if (q.hasLow())
1816      b.append(q.getLow().getValue().toString());
1817    else
1818      b.append("?");
1819    b.append("-");
1820    if (q.hasHigh())
1821      b.append(q.getHigh().getValue().toString());
1822    else
1823      b.append("?");
1824    if (q.getLow().hasUnit())
1825      b.append(" "+q.getLow().getUnit());
1826    return b.toString();
1827  }
1828
1829  private void renderHumanName(HumanName name, XhtmlNode x) {
1830    x.addText(displayHumanName(name));
1831  }
1832
1833  private void renderAnnotation(Annotation annot, XhtmlNode x) {
1834    x.addText(annot.getText());
1835  }
1836
1837  private void renderAddress(Address address, XhtmlNode x) {
1838    x.addText(displayAddress(address));
1839  }
1840
1841  private void renderContactPoint(ContactPoint contact, XhtmlNode x) {
1842    x.addText(displayContactPoint(contact));
1843  }
1844
1845  private void renderUri(UriType uri, XhtmlNode x) {
1846    x.ah(uri.getValue()).addText(uri.getValue());
1847  }
1848
1849  private void renderSampledData(SampledData sampledData, XhtmlNode x) {
1850    x.addText(displaySampledData(sampledData));
1851  }
1852
1853  private String displaySampledData(SampledData s) {
1854    CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder();
1855    if (s.hasOrigin())
1856      b.append("Origin: "+displayQuantity(s.getOrigin()));
1857
1858    if (s.hasPeriod())
1859      b.append("Period: "+s.getPeriod().toString());
1860
1861    if (s.hasFactor())
1862      b.append("Factor: "+s.getFactor().toString());
1863
1864    if (s.hasLowerLimit())
1865      b.append("Lower: "+s.getLowerLimit().toString());
1866
1867    if (s.hasUpperLimit())
1868      b.append("Upper: "+s.getUpperLimit().toString());
1869
1870    if (s.hasDimensions())
1871      b.append("Dimensions: "+s.getDimensions());
1872
1873    if (s.hasData())
1874      b.append("Data: "+s.getData());
1875
1876    return b.toString();
1877  }
1878
1879  private String displayQuantity(Quantity q) {
1880    StringBuilder s = new StringBuilder();
1881
1882    s.append("(system = '").append(describeSystem(q.getSystem()))
1883        .append("' code ").append(q.getCode())
1884        .append(" = '").append(lookupCode(q.getSystem(), q.getCode())).append("')");
1885
1886    return s.toString();
1887  }
1888
1889  private String displayTiming(Timing s) throws FHIRException {
1890    CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder();
1891    if (s.hasCode())
1892        b.append("Code: "+displayCodeableConcept(s.getCode()));
1893
1894    if (s.getEvent().size() > 0) {
1895      CommaSeparatedStringBuilder c = new CommaSeparatedStringBuilder();
1896      for (DateTimeType p : s.getEvent()) {
1897        c.append(p.toHumanDisplay());
1898      }
1899      b.append("Events: "+ c.toString());
1900    }
1901
1902    if (s.hasRepeat()) {
1903      TimingRepeatComponent rep = s.getRepeat();
1904      if (rep.hasBoundsPeriod() && rep.getBoundsPeriod().hasStart())
1905        b.append("Starting "+rep.getBoundsPeriod().getStartElement().toHumanDisplay());
1906      if (rep.hasCount())
1907        b.append("Count "+Integer.toString(rep.getCount())+" times");
1908      if (rep.hasDuration())
1909        b.append("Duration "+rep.getDuration().toPlainString()+displayTimeUnits(rep.getPeriodUnit()));
1910
1911      if (rep.hasWhen()) {
1912        String st = "";
1913        if (rep.hasOffset()) {
1914          st = Integer.toString(rep.getOffset())+"min ";
1915        }
1916        b.append("Do "+st);
1917        for (Enumeration<EventTiming> wh : rep.getWhen())
1918          b.append(displayEventCode(wh.getValue()));
1919      } else {
1920        String st = "";
1921        if (!rep.hasFrequency() || (!rep.hasFrequencyMax() && rep.getFrequency() == 1) )
1922          st = "Once";
1923        else {
1924          st = Integer.toString(rep.getFrequency());
1925          if (rep.hasFrequencyMax())
1926            st = st + "-"+Integer.toString(rep.getFrequency());
1927        }
1928        if (rep.hasPeriod()) {
1929        st = st + " per "+rep.getPeriod().toPlainString();
1930        if (rep.hasPeriodMax())
1931          st = st + "-"+rep.getPeriodMax().toPlainString();
1932                st = st + " "+displayTimeUnits(rep.getPeriodUnit());
1933        }
1934        b.append("Do "+st);
1935      }
1936      if (rep.hasBoundsPeriod() && rep.getBoundsPeriod().hasEnd())
1937        b.append("Until "+rep.getBoundsPeriod().getEndElement().toHumanDisplay());
1938    }
1939    return b.toString();
1940  }
1941
1942  private String displayEventCode(EventTiming when) {
1943    switch (when) {
1944    case C: return "at meals";
1945    case CD: return "at lunch";
1946    case CM: return "at breakfast";
1947    case CV: return "at dinner";
1948    case AC: return "before meals";
1949    case ACD: return "before lunch";
1950    case ACM: return "before breakfast";
1951    case ACV: return "before dinner";
1952    case HS: return "before sleeping";
1953    case PC: return "after meals";
1954    case PCD: return "after lunch";
1955    case PCM: return "after breakfast";
1956    case PCV: return "after dinner";
1957    case WAKE: return "after waking";
1958    default: return "??";
1959    }
1960  }
1961
1962  private String displayTimeUnits(UnitsOfTime units) {
1963        if (units == null)
1964                return "??";
1965    switch (units) {
1966    case A: return "years";
1967    case D: return "days";
1968    case H: return "hours";
1969    case MIN: return "minutes";
1970    case MO: return "months";
1971    case S: return "seconds";
1972    case WK: return "weeks";
1973    default: return "??";
1974    }
1975  }
1976
1977  public static String displayHumanName(HumanName name) {
1978    StringBuilder s = new StringBuilder();
1979    if (name.hasText())
1980      s.append(name.getText());
1981    else {
1982      for (StringType p : name.getGiven()) {
1983        s.append(p.getValue());
1984        s.append(" ");
1985      }
1986      if (name.hasFamily()) {
1987        s.append(name.getFamily());
1988        s.append(" ");
1989      }
1990    }
1991    if (name.hasUse() && name.getUse() != NameUse.USUAL)
1992      s.append("("+name.getUse().toString()+")");
1993    return s.toString();
1994  }
1995
1996  private String displayAddress(Address address) {
1997    StringBuilder s = new StringBuilder();
1998    if (address.hasText())
1999      s.append(address.getText());
2000    else {
2001      for (StringType p : address.getLine()) {
2002        s.append(p.getValue());
2003        s.append(" ");
2004      }
2005      if (address.hasCity()) {
2006        s.append(address.getCity());
2007        s.append(" ");
2008      }
2009      if (address.hasState()) {
2010        s.append(address.getState());
2011        s.append(" ");
2012      }
2013
2014      if (address.hasPostalCode()) {
2015        s.append(address.getPostalCode());
2016        s.append(" ");
2017      }
2018
2019      if (address.hasCountry()) {
2020        s.append(address.getCountry());
2021        s.append(" ");
2022      }
2023    }
2024    if (address.hasUse())
2025      s.append("("+address.getUse().toString()+")");
2026    return s.toString();
2027  }
2028
2029  public static String displayContactPoint(ContactPoint contact) {
2030    StringBuilder s = new StringBuilder();
2031    s.append(describeSystem(contact.getSystem()));
2032    if (Utilities.noString(contact.getValue()))
2033      s.append("-unknown-");
2034    else
2035      s.append(contact.getValue());
2036    if (contact.hasUse())
2037      s.append("("+contact.getUse().toString()+")");
2038    return s.toString();
2039  }
2040
2041  private static String describeSystem(ContactPointSystem system) {
2042    if (system == null)
2043      return "";
2044    switch (system) {
2045    case PHONE: return "ph: ";
2046    case FAX: return "fax: ";
2047    default:
2048      return "";
2049    }
2050  }
2051
2052  private String displayIdentifier(Identifier ii) {
2053    String s = Utilities.noString(ii.getValue()) ? "??" : ii.getValue();
2054
2055    if (ii.hasType()) {
2056        if (ii.getType().hasText())
2057                s = ii.getType().getText()+" = "+s;
2058        else if (ii.getType().hasCoding() && ii.getType().getCoding().get(0).hasDisplay())
2059                s = ii.getType().getCoding().get(0).getDisplay()+" = "+s;
2060        else if (ii.getType().hasCoding() && ii.getType().getCoding().get(0).hasCode())
2061                s = lookupCode(ii.getType().getCoding().get(0).getSystem(), ii.getType().getCoding().get(0).getCode())+" = "+s;
2062    }
2063
2064    if (ii.hasUse())
2065      s = s + " ("+ii.getUse().toString()+")";
2066    return s;
2067  }
2068
2069  private List<ElementDefinition> getChildrenForPath(List<ElementDefinition> elements, String path) throws DefinitionException {
2070    // do we need to do a name reference substitution?
2071    for (ElementDefinition e : elements) {
2072      if (e.getPath().equals(path) && e.hasContentReference()) {
2073        String ref = e.getContentReference();
2074        ElementDefinition t = null;
2075        // now, resolve the name
2076        for (ElementDefinition e1 : elements) {
2077                if (ref.equals("#"+e1.getId()))
2078                        t = e1;
2079        }
2080        if (t == null)
2081                throw new DefinitionException("Unable to resolve content reference "+ref+" trying to resolve "+path);
2082        path = t.getPath();
2083        break;
2084      }
2085    }
2086
2087    List<ElementDefinition> results = new ArrayList<ElementDefinition>();
2088    for (ElementDefinition e : elements) {
2089      if (e.getPath().startsWith(path+".") && !e.getPath().substring(path.length()+1).contains("."))
2090        results.add(e);
2091    }
2092    return results;
2093  }
2094
2095
2096  public boolean generate(ResourceContext rcontext, ConceptMap cm) throws FHIRFormatError, DefinitionException, IOException {
2097    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
2098    x.h2().addText(cm.getName()+" ("+cm.getUrl()+")");
2099
2100    XhtmlNode p = x.para();
2101    if (cm.hasSource() || cm.hasTarget())
2102      p.tx("Mapping from ");
2103    if (!cm.hasSource())
2104      p.tx("(unspecified)");
2105    else
2106      AddVsRef(rcontext, cm.getSource() instanceof Reference ? ((Reference) cm.getSource()).getReference() : ((UriType) cm.getSource()).asStringValue(), p);
2107    p.tx(" to ");
2108    if (!cm.hasTarget())
2109      p.tx("(unspecified)");
2110    else 
2111      AddVsRef(rcontext, cm.getTarget() instanceof Reference ? ((Reference) cm.getTarget()).getReference() : ((UriType) cm.getTarget()).asStringValue(), p);
2112    
2113    p = x.para();
2114    if (cm.getExperimental())
2115      p.addText(Utilities.capitalize(cm.getStatus().toString())+" (not intended for production usage). ");
2116    else
2117      p.addText(Utilities.capitalize(cm.getStatus().toString())+". ");
2118    p.tx("Published on "+(cm.hasDate() ? cm.getDateElement().toHumanDisplay() : "??")+" by "+cm.getPublisher());
2119    if (!cm.getContact().isEmpty()) {
2120      p.tx(" (");
2121      boolean firsti = true;
2122      for (ContactDetail ci : cm.getContact()) {
2123        if (firsti)
2124          firsti = false;
2125        else
2126          p.tx(", ");
2127        if (ci.hasName())
2128          p.addText(ci.getName()+": ");
2129        boolean first = true;
2130        for (ContactPoint c : ci.getTelecom()) {
2131          if (first)
2132            first = false;
2133          else
2134            p.tx(", ");
2135          addTelecom(p, c);
2136        }
2137      }
2138      p.tx(")");
2139    }
2140    p.tx(". ");
2141    p.addText(cm.getCopyright());
2142    if (!Utilities.noString(cm.getDescription()))
2143      addMarkdown(x, cm.getDescription());
2144
2145    x.br();
2146
2147    for (ConceptMapGroupComponent grp : cm.getGroup()) {
2148      String src = grp.getSource();
2149      boolean comment = false;
2150      boolean ok = true;
2151    Map<String, HashSet<String>> sources = new HashMap<String, HashSet<String>>();
2152    Map<String, HashSet<String>> targets = new HashMap<String, HashSet<String>>();
2153      sources.put("code", new HashSet<String>());
2154    targets.put("code", new HashSet<String>());
2155      SourceElementComponent cc = grp.getElement().get(0);
2156      String dst = grp.getTarget();
2157      sources.get("code").add(grp.getSource());
2158      targets.get("code").add(grp.getTarget());
2159      for (SourceElementComponent ccl : grp.getElement()) {
2160        ok = ok && ccl.getTarget().size() == 1 && ccl.getTarget().get(0).getDependsOn().isEmpty() && ccl.getTarget().get(0).getProduct().isEmpty();
2161        for (TargetElementComponent ccm : ccl.getTarget()) {
2162                comment = comment || !Utilities.noString(ccm.getComment());
2163                for (OtherElementComponent d : ccm.getDependsOn()) {
2164            if (!sources.containsKey(d.getProperty()))
2165              sources.put(d.getProperty(), new HashSet<String>());
2166            sources.get(d.getProperty()).add(d.getSystem());
2167                }
2168                for (OtherElementComponent d : ccm.getProduct()) {
2169            if (!targets.containsKey(d.getProperty()))
2170              targets.put(d.getProperty(), new HashSet<String>());
2171            targets.get(d.getProperty()).add(d.getSystem());
2172            }
2173
2174                }
2175        }
2176
2177      String display;
2178      if (ok) {
2179        // simple
2180        XhtmlNode tbl = x.table( "grid");
2181        XhtmlNode tr = tbl.tr();
2182        tr.td().b().tx("Source Code");
2183        tr.td().b().tx("Equivalence");
2184        tr.td().b().tx("Destination Code");
2185        if (comment)
2186          tr.td().b().tx("Comment");
2187        for (SourceElementComponent ccl : grp.getElement()) {
2188          tr = tbl.tr();
2189          XhtmlNode td = tr.td();
2190          td.addText(ccl.getCode());
2191          display = getDisplayForConcept(grp.getSource(), ccl.getCode());
2192          if (display != null)
2193            td.tx(" ("+display+")");
2194          TargetElementComponent ccm = ccl.getTarget().get(0);
2195          tr.td().addText(!ccm.hasEquivalence() ? "" : ccm.getEquivalence().toCode());
2196          td = tr.td();
2197          td.addText(ccm.getCode());
2198          display = getDisplayForConcept(grp.getTarget(), ccm.getCode());
2199          if (display != null)
2200            td.tx(" ("+display+")");
2201          if (comment)
2202            tr.td().addText(ccm.getComment());
2203        }
2204      } else {
2205        XhtmlNode tbl = x.table( "grid");
2206        XhtmlNode tr = tbl.tr();
2207        XhtmlNode td;
2208        tr.td().colspan(Integer.toString(sources.size())).b().tx("Source Concept");
2209        tr.td().b().tx("Equivalence");
2210        tr.td().colspan(Integer.toString(targets.size())).b().tx("Destination Concept");
2211        if (comment)
2212          tr.td().b().tx("Comment");
2213        tr = tbl.tr();
2214        if (sources.get("code").size() == 1)
2215          tr.td().b().tx("Code "+sources.get("code").toString()+"");
2216        else
2217          tr.td().b().tx("Code");
2218        for (String s : sources.keySet()) {
2219          if (!s.equals("code")) {
2220            if (sources.get(s).size() == 1)
2221              tr.td().b().addText(getDescForConcept(s) +" "+sources.get(s).toString());
2222            else
2223              tr.td().b().addText(getDescForConcept(s));
2224          }
2225        }
2226        tr.td();
2227        if (targets.get("code").size() == 1)
2228          tr.td().b().tx("Code "+targets.get("code").toString());
2229        else
2230          tr.td().b().tx("Code");
2231        for (String s : targets.keySet()) {
2232          if (!s.equals("code")) {
2233            if (targets.get(s).size() == 1)
2234              tr.td().b().addText(getDescForConcept(s) +" "+targets.get(s).toString()+"");
2235            else
2236              tr.td().b().addText(getDescForConcept(s));
2237          }
2238        }
2239        if (comment)
2240          tr.td();
2241
2242        for (SourceElementComponent ccl : grp.getElement()) {
2243          tr = tbl.tr();
2244          td = tr.td();
2245          if (sources.get("code").size() == 1)
2246            td.addText(ccl.getCode());
2247          else
2248            td.addText(grp.getSource()+" / "+ccl.getCode());
2249          display = getDisplayForConcept(grp.getSource(), ccl.getCode());
2250          if (display != null)
2251            td.tx(" ("+display+")");
2252
2253          TargetElementComponent ccm = ccl.getTarget().get(0);
2254          for (String s : sources.keySet()) {
2255            if (!s.equals("code")) {
2256              td = tr.td();
2257              td.addText(getCode(ccm.getDependsOn(), s, sources.get(s).size() != 1));
2258              display = getDisplay(ccm.getDependsOn(), s);
2259              if (display != null)
2260                td.tx(" ("+display+")");
2261            }
2262          }
2263          if (!ccm.hasEquivalence())
2264            tr.td().tx(":"+"("+ConceptMapEquivalence.EQUIVALENT.toCode()+")");
2265          else
2266            tr.td().tx(":"+ccm.getEquivalence().toCode());
2267          td = tr.td();
2268          if (targets.get("code").size() == 1)
2269            td.addText(ccm.getCode());
2270          else
2271            td.addText(grp.getTarget()+" / "+ccm.getCode());
2272          display = getDisplayForConcept(grp.getTarget(), ccm.getCode());
2273          if (display != null)
2274            td.tx(" ("+display+")");
2275
2276          for (String s : targets.keySet()) {
2277            if (!s.equals("code")) {
2278              td = tr.td();
2279              td.addText(getCode(ccm.getProduct(), s, targets.get(s).size() != 1));
2280              display = getDisplay(ccm.getProduct(), s);
2281              if (display != null)
2282                td.tx(" ("+display+")");
2283            }
2284          }
2285          if (comment)
2286            tr.td().addText(ccm.getComment());
2287        }
2288      }
2289    }
2290
2291    inject(cm, x, NarrativeStatus.GENERATED);
2292    return true;
2293  }
2294
2295
2296
2297  private void inject(DomainResource r, XhtmlNode x, NarrativeStatus status) {
2298    if (!x.hasAttribute("xmlns"))
2299      x.setAttribute("xmlns", "http://www.w3.org/1999/xhtml");
2300    if (!r.hasText() || !r.getText().hasDiv() || r.getText().getDiv().getChildNodes().isEmpty()) {
2301      r.setText(new Narrative());
2302      r.getText().setDiv(x);
2303      r.getText().setStatus(status);
2304    } else {
2305      XhtmlNode n = r.getText().getDiv();
2306      n.hr();
2307      n.addChildNodes(x.getChildNodes());
2308    }
2309  }
2310
2311  public Element getNarrative(Element er) {
2312    Element txt = XMLUtil.getNamedChild(er, "text");
2313    if (txt == null)
2314      return null;
2315    return XMLUtil.getNamedChild(txt, "div");
2316  }
2317
2318
2319  private void inject(Element er, XhtmlNode x, NarrativeStatus status) {
2320    if (!x.hasAttribute("xmlns"))
2321      x.setAttribute("xmlns", "http://www.w3.org/1999/xhtml");
2322    Element txt = XMLUtil.getNamedChild(er, "text");
2323    if (txt == null) {
2324      txt = er.getOwnerDocument().createElementNS(FormatUtilities.FHIR_NS, "text");
2325      Element n = XMLUtil.getFirstChild(er);
2326      while (n != null && (n.getNodeName().equals("id") || n.getNodeName().equals("meta") || n.getNodeName().equals("implicitRules") || n.getNodeName().equals("language")))
2327        n = XMLUtil.getNextSibling(n);
2328      if (n == null)
2329        er.appendChild(txt);
2330      else
2331        er.insertBefore(txt, n);
2332    }
2333    Element st = XMLUtil.getNamedChild(txt, "status");
2334    if (st == null) {
2335      st = er.getOwnerDocument().createElementNS(FormatUtilities.FHIR_NS, "status");
2336      Element n = XMLUtil.getFirstChild(txt);
2337      if (n == null)
2338        txt.appendChild(st);
2339      else
2340        txt.insertBefore(st, n);
2341    }
2342    st.setAttribute("value", status.toCode());
2343    Element div = XMLUtil.getNamedChild(txt, "div");
2344    if (div == null) {
2345      div = er.getOwnerDocument().createElementNS(FormatUtilities.XHTML_NS, "div");
2346      div.setAttribute("xmlns", FormatUtilities.XHTML_NS);
2347      txt.appendChild(div);
2348    }
2349    if (div.hasChildNodes())
2350      div.appendChild(er.getOwnerDocument().createElementNS(FormatUtilities.XHTML_NS, "hr"));
2351    new XhtmlComposer(XhtmlComposer.XML).compose(div, x);
2352  }
2353
2354  private void inject(org.hl7.fhir.dstu3.elementmodel.Element er, XhtmlNode x, NarrativeStatus status) throws DefinitionException, IOException {
2355    if (!x.hasAttribute("xmlns"))
2356      x.setAttribute("xmlns", "http://www.w3.org/1999/xhtml");
2357    org.hl7.fhir.dstu3.elementmodel.Element txt = er.getNamedChild("text");
2358    if (txt == null) {
2359      txt = new org.hl7.fhir.dstu3.elementmodel.Element("text", er.getProperty().getChild(null, "text"));
2360      int i = 0;
2361      while (i < er.getChildren().size() && (er.getChildren().get(i).getName().equals("id") || er.getChildren().get(i).getName().equals("meta") || er.getChildren().get(i).getName().equals("implicitRules") || er.getChildren().get(i).getName().equals("language")))
2362        i++;
2363      if (i >= er.getChildren().size())
2364        er.getChildren().add(txt);
2365      else
2366        er.getChildren().add(i, txt);
2367    }
2368    org.hl7.fhir.dstu3.elementmodel.Element st = txt.getNamedChild("status");
2369    if (st == null) {
2370      st = new org.hl7.fhir.dstu3.elementmodel.Element("status", txt.getProperty().getChild(null, "status"));
2371      txt.getChildren().add(0, st);
2372    }
2373    st.setValue(status.toCode());
2374    org.hl7.fhir.dstu3.elementmodel.Element div = txt.getNamedChild("div");
2375    if (div == null) {
2376      div = new org.hl7.fhir.dstu3.elementmodel.Element("div", txt.getProperty().getChild(null, "div"));
2377      txt.getChildren().add(div);
2378      div.setValue(new XhtmlComposer(XhtmlComposer.XML).compose(x));
2379    }
2380    div.setXhtml(x);
2381  }
2382
2383  private String getDisplay(List<OtherElementComponent> list, String s) {
2384    for (OtherElementComponent c : list) {
2385      if (s.equals(c.getProperty()))
2386        return getDisplayForConcept(c.getSystem(), c.getCode());
2387    }
2388    return null;
2389  }
2390
2391  private String getDisplayForConcept(String system, String code) {
2392    if (code == null)
2393      return null;
2394    ValidationResult cl = context.validateCode(system, code, null);
2395    return cl == null ? null : cl.getDisplay();
2396  }
2397
2398
2399
2400  private String getDescForConcept(String s) {
2401    if (s.startsWith("http://hl7.org/fhir/v2/element/"))
2402        return "v2 "+s.substring("http://hl7.org/fhir/v2/element/".length());
2403    return s;
2404  }
2405
2406  private String getCode(List<OtherElementComponent> list, String s, boolean withSystem) {
2407    for (OtherElementComponent c : list) {
2408      if (s.equals(c.getProperty()))
2409        if (withSystem)
2410          return c.getSystem()+" / "+c.getCode();
2411        else
2412          return c.getCode();
2413    }
2414    return null;
2415  }
2416
2417  private void addTelecom(XhtmlNode p, ContactPoint c) {
2418    if (c.getSystem() == ContactPointSystem.PHONE) {
2419      p.tx("Phone: "+c.getValue());
2420    } else if (c.getSystem() == ContactPointSystem.FAX) {
2421      p.tx("Fax: "+c.getValue());
2422    } else if (c.getSystem() == ContactPointSystem.EMAIL) {
2423      p.ah( "mailto:"+c.getValue()).addText(c.getValue());
2424    } else if (c.getSystem() == ContactPointSystem.URL) {
2425      if (c.getValue().length() > 30)
2426        p.ah(c.getValue()).addText(c.getValue().substring(0, 30)+"...");
2427      else
2428        p.ah(c.getValue()).addText(c.getValue());
2429    }
2430  }
2431
2432  /**
2433   * This generate is optimised for the FHIR build process itself in as much as it
2434   * generates hyperlinks in the narrative that are only going to be correct for
2435   * the purposes of the build. This is to be reviewed in the future.
2436   *
2437   * @param vs
2438   * @param codeSystems
2439   * @throws IOException
2440   * @throws DefinitionException
2441   * @throws FHIRFormatError
2442   * @throws Exception
2443   */
2444  public boolean generate(ResourceContext rcontext, CodeSystem cs, boolean header) throws FHIRFormatError, DefinitionException, IOException {
2445    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
2446    boolean hasExtensions = false;
2447    hasExtensions = generateDefinition(x, cs, header);
2448    inject(cs, x, hasExtensions ? NarrativeStatus.EXTENSIONS :  NarrativeStatus.GENERATED);
2449    return true;
2450  }
2451
2452  private boolean generateDefinition(XhtmlNode x, CodeSystem cs, boolean header) throws FHIRFormatError, DefinitionException, IOException {
2453    boolean hasExtensions = false;
2454    Map<ConceptMap, String> mymaps = new HashMap<ConceptMap, String>();
2455//    for (ConceptMap a : context.findMapsForSource(cs.getValueSet())) {
2456//      String url = "";
2457//      ValueSet vsr = context.fetchResource(ValueSet.class, ((Reference) a.getTarget()).getReference());
2458//      if (vsr != null)
2459//        url = (String) vsr.getUserData("filename");
2460//      mymaps.put(a, url);
2461//    }
2462    // also, look in the contained resources for a concept map
2463    for (Resource r : cs.getContained()) {
2464      if (r instanceof ConceptMap) {
2465        ConceptMap cm = (ConceptMap) r;
2466        if (((Reference) cm.getSource()).getReference().equals(cs.getValueSet())) {
2467          String url = "";
2468          ValueSet vsr = context.fetchResource(ValueSet.class, ((Reference) cm.getTarget()).getReference());
2469          if (vsr != null)
2470              url = (String) vsr.getUserData("filename");
2471        mymaps.put(cm, url);
2472        }
2473      }
2474    }
2475    List<String> langs = new ArrayList<String>();
2476
2477    if (header) {
2478      XhtmlNode h = x.h2();
2479      h.addText(cs.hasTitle() ? cs.getTitle() : cs.getName());
2480      addMarkdown(x, cs.getDescription());
2481      if (cs.hasCopyright())
2482        generateCopyright(x, cs);
2483    }
2484
2485    generateProperties(x, cs);
2486    generateFilters(x, cs);
2487    hasExtensions = generateCodeSystemContent(x, cs, hasExtensions, mymaps, langs);
2488
2489    return hasExtensions;
2490  }
2491
2492  private void generateFilters(XhtmlNode x, CodeSystem cs) {
2493    if (cs.hasFilter()) {
2494      x.para().b().tx("Filters");
2495      XhtmlNode tbl = x.table("grid");
2496      XhtmlNode tr = tbl.tr();
2497      tr.td().b().tx("Code");
2498      tr.td().b().tx("Description");
2499      tr.td().b().tx("operator");
2500      tr.td().b().tx("Value");
2501      for (CodeSystemFilterComponent f : cs.getFilter()) {
2502        tr = tbl.tr();
2503        tr.td().tx(f.getCode());
2504        tr.td().tx(f.getDescription());
2505        XhtmlNode td = tr.td();
2506        for (Enumeration<org.hl7.fhir.dstu3.model.CodeSystem.FilterOperator> t : f.getOperator())
2507          td.tx(t.asStringValue()+" ");
2508        tr.td().tx(f.getValue());
2509      }
2510    }
2511  }
2512
2513  private void generateProperties(XhtmlNode x, CodeSystem cs) {
2514    if (cs.hasProperty()) {
2515      x.para().b().tx("Properties");
2516      XhtmlNode tbl = x.table("grid");
2517      XhtmlNode tr = tbl.tr();
2518      tr.td().b().tx("Code");
2519      tr.td().b().tx("URL");
2520      tr.td().b().tx("Description");
2521      tr.td().b().tx("Type");
2522      for (PropertyComponent p : cs.getProperty()) {
2523        tr = tbl.tr();
2524        tr.td().tx(p.getCode());
2525        tr.td().tx(p.getUri());
2526        tr.td().tx(p.getDescription());
2527        tr.td().tx(p.hasType() ? p.getType().toCode() : "");
2528      }
2529    }
2530  }
2531
2532  private boolean generateCodeSystemContent(XhtmlNode x, CodeSystem cs, boolean hasExtensions, Map<ConceptMap, String> mymaps, List<String> langs)
2533      throws FHIRFormatError, DefinitionException, IOException {
2534    XhtmlNode p = x.para();
2535    if (cs.getContent() == CodeSystemContentMode.COMPLETE)
2536      p.tx("This code system "+cs.getUrl()+" defines the following codes:");
2537    else if (cs.getContent() == CodeSystemContentMode.EXAMPLE)
2538        p.tx("This code system "+cs.getUrl()+" defines many codes, of which the following are some examples:");
2539    else if (cs.getContent() == CodeSystemContentMode.FRAGMENT )
2540      p.tx("This code system "+cs.getUrl()+" defines many codes, of which the following are a subset:");
2541    else if (cs.getContent() == CodeSystemContentMode.NOTPRESENT ) {
2542      p.tx("This code system "+cs.getUrl()+" defines many codes, but they are not represented here");
2543      return false;
2544    }
2545    XhtmlNode t = x.table( "codes");
2546    boolean commentS = false;
2547    boolean deprecated = false;
2548    boolean display = false;
2549    boolean hierarchy = false;
2550    for (ConceptDefinitionComponent c : cs.getConcept()) {
2551      commentS = commentS || conceptsHaveComments(c);
2552      deprecated = deprecated || conceptsHaveDeprecated(cs, c);
2553      display = display || conceptsHaveDisplay(c);
2554      hierarchy = hierarchy || c.hasConcept();
2555      scanLangs(c, langs);
2556    }
2557    addMapHeaders(addTableHeaderRowStandard(t, hierarchy, display, true, commentS, deprecated), mymaps);
2558    for (ConceptDefinitionComponent c : cs.getConcept()) {
2559      hasExtensions = addDefineRowToTable(t, c, 0, hierarchy, display, commentS, deprecated, mymaps, cs.getUrl(), cs) || hasExtensions;
2560    }
2561    if (langs.size() > 0) {
2562      Collections.sort(langs);
2563      x.para().b().tx("Additional Language Displays");
2564      t = x.table( "codes");
2565      XhtmlNode tr = t.tr();
2566      tr.td().b().tx("Code");
2567      for (String lang : langs)
2568        tr.td().b().addText(describeLang(lang));
2569      for (ConceptDefinitionComponent c : cs.getConcept()) {
2570        addLanguageRow(c, t, langs);
2571      }
2572    }
2573    return hasExtensions;
2574  }
2575
2576  private int countConcepts(List<ConceptDefinitionComponent> list) {
2577    int count = list.size();
2578    for (ConceptDefinitionComponent c : list)
2579      if (c.hasConcept())
2580        count = count + countConcepts(c.getConcept());
2581    return count;
2582  }
2583
2584  private void generateCopyright(XhtmlNode x, CodeSystem cs) {
2585    XhtmlNode p = x.para();
2586    p.b().tx("Copyright Statement:");
2587    smartAddText(p, " " + cs.getCopyright());
2588  }
2589
2590
2591  /**
2592   * This generate is optimised for the FHIR build process itself in as much as it
2593   * generates hyperlinks in the narrative that are only going to be correct for
2594   * the purposes of the build. This is to be reviewed in the future.
2595   *
2596   * @param vs
2597   * @param codeSystems
2598   * @throws FHIRException
2599   * @throws IOException
2600   * @throws Exception
2601   */
2602  public boolean generate(ResourceContext rcontext, ValueSet vs, boolean header) throws FHIRException, IOException {
2603    generate(rcontext, vs, null, header);
2604    return true;
2605  }
2606
2607  public void generate(ResourceContext rcontext, ValueSet vs, ValueSet src, boolean header) throws FHIRException, IOException {
2608    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
2609    boolean hasExtensions;
2610    if (vs.hasExpansion()) {
2611      // for now, we just accept an expansion if there is one
2612      hasExtensions = generateExpansion(x, vs, src, header);
2613    } else {
2614      hasExtensions = generateComposition(rcontext, x, vs, header);
2615    }
2616    inject(vs, x, hasExtensions ? NarrativeStatus.EXTENSIONS :  NarrativeStatus.GENERATED);
2617  }
2618
2619  private Integer countMembership(ValueSet vs) {
2620    int count = 0;
2621    if (vs.hasExpansion())
2622      count = count + conceptCount(vs.getExpansion().getContains());
2623    else {
2624      if (vs.hasCompose()) {
2625        if (vs.getCompose().hasExclude()) {
2626          try {
2627            ValueSetExpansionOutcome vse = context.expandVS(vs, true, false);
2628            count = 0;
2629            count += conceptCount(vse.getValueset().getExpansion().getContains());
2630            return count;
2631          } catch (Exception e) {
2632            return null;
2633          }
2634        }
2635        for (ConceptSetComponent inc : vs.getCompose().getInclude()) {
2636          if (inc.hasFilter())
2637            return null;
2638          if (!inc.hasConcept())
2639            return null;
2640          count = count + inc.getConcept().size();
2641        }
2642      }
2643    }
2644    return count;
2645  }
2646
2647  private int conceptCount(List<ValueSetExpansionContainsComponent> list) {
2648    int count = 0;
2649    for (ValueSetExpansionContainsComponent c : list) {
2650      if (!c.getAbstract())
2651        count++;
2652      count = count + conceptCount(c.getContains());
2653    }
2654    return count;
2655  }
2656
2657  private boolean generateExpansion(XhtmlNode x, ValueSet vs, ValueSet src, boolean header) throws FHIRFormatError, DefinitionException, IOException {
2658    boolean hasExtensions = false;
2659    List<String> langs = new ArrayList<String>();
2660
2661    Map<ConceptMap, String> mymaps = new HashMap<ConceptMap, String>();
2662//    for (ConceptMap a : context.findMapsForSource(vs.getUrl())) {
2663//      String url = "";
2664//      ValueSet vsr = context.fetchResource(ValueSet.class, ((Reference) a.getTarget()).getReference());
2665//      if (vsr != null)
2666//        url = (String) vsr.getUserData("filename");
2667//      mymaps.put(a, url);
2668//    }
2669
2670    if (header) {
2671      XhtmlNode h = x.addTag(getHeader());
2672      h.tx("Value Set Contents");
2673      if (IsNotFixedExpansion(vs))
2674        addMarkdown(x, vs.getDescription());
2675      if (vs.hasCopyright())
2676        generateCopyright(x, vs);
2677    }
2678    if (ToolingExtensions.hasExtension(vs.getExpansion(), "http://hl7.org/fhir/StructureDefinition/valueset-toocostly"))
2679      x.para().setAttribute("style", "border: maroon 1px solid; background-color: #FFCCCC; font-weight: bold; padding: 8px").addText(vs.getExpansion().getContains().isEmpty() ? tooCostlyNoteEmpty : tooCostlyNoteNotEmpty );
2680    else {
2681      Integer count = countMembership(vs);
2682      if (count == null)
2683        x.para().tx("This value set does not contain a fixed number of concepts");
2684      else
2685        x.para().tx("This value set contains "+count.toString()+" concepts");
2686    }
2687
2688    generateVersionNotice(x, vs.getExpansion());
2689
2690    CodeSystem allCS = null;
2691    boolean doSystem = true; // checkDoSystem(vs, src);
2692    boolean doDefinition = checkDoDefinition(vs.getExpansion().getContains());
2693    if (doSystem && allFromOneSystem(vs)) {
2694      doSystem = false;
2695      XhtmlNode p = x.para();
2696      p.tx("All codes from system ");
2697      allCS = context.fetchCodeSystem(vs.getExpansion().getContains().get(0).getSystem());
2698      String ref = null;
2699      if (allCS != null)
2700        ref = getCsRef(allCS);
2701      if (ref == null)
2702        p.code(vs.getExpansion().getContains().get(0).getSystem());
2703      else
2704        p.ah(prefix+ref).code(vs.getExpansion().getContains().get(0).getSystem());
2705    }
2706    XhtmlNode t = x.table( "codes");
2707    XhtmlNode tr = t.tr();
2708    tr.td().b().tx("Code");
2709    if (doSystem)
2710      tr.td().b().tx("System");
2711    tr.td().b().tx("Display");
2712    if (doDefinition)
2713      tr.td().b().tx("Definition");
2714
2715    addMapHeaders(tr, mymaps);
2716    for (ValueSetExpansionContainsComponent c : vs.getExpansion().getContains()) {
2717      addExpansionRowToTable(t, c, 0, doSystem, doDefinition, mymaps, allCS, langs);
2718    }
2719
2720    // now, build observed languages
2721
2722    if (langs.size() > 0) {
2723      Collections.sort(langs);
2724      x.para().b().tx("Additional Language Displays");
2725      t = x.table( "codes");
2726      tr = t.tr();
2727      tr.td().b().tx("Code");
2728      for (String lang : langs)
2729        tr.td().b().addText(describeLang(lang));
2730      for (ValueSetExpansionContainsComponent c : vs.getExpansion().getContains()) {
2731        addLanguageRow(c, t, langs);
2732      }
2733    }
2734
2735    return hasExtensions;
2736  }
2737
2738  @SuppressWarnings("rawtypes")
2739  private void generateVersionNotice(XhtmlNode x, ValueSetExpansionComponent expansion) {
2740    Map<String, String> versions = new HashMap<String, String>();
2741    boolean firstVersion = true;
2742    for (ValueSetExpansionParameterComponent p : expansion.getParameter()) {
2743      if (p.getName().equals("version")) {
2744        String[] parts = ((PrimitiveType) p.getValue()).asStringValue().split("\\|");
2745        if (parts.length == 2)
2746          versions.put(parts[0], parts[1]);
2747        if (!versions.isEmpty()) {
2748          StringBuilder b = new StringBuilder();
2749          if (firstVersion) {
2750            // the first version
2751            // set the <p> tag and style attribute
2752            x.para().setAttribute("style", "border: black 1px dotted; background-color: #EEEEEE; padding: 8px");
2753                  firstVersion = false;
2754          } else {
2755            // the second (or greater) version
2756            x.br(); // add line break before the version text
2757          }
2758          b.append("Expansion based on ");
2759          boolean firstPart = true;
2760          for (String s : versions.keySet()) {
2761            if (firstPart)
2762              firstPart = false;
2763            else
2764              b.append(", ");
2765            if (!s.equals("http://snomed.info/sct"))
2766              b.append(describeSystem(s)+" version "+versions.get(s));
2767            else {
2768              parts = versions.get(s).split("\\/");
2769              if (parts.length >= 5) {
2770                String m = describeModule(parts[4]);
2771                if (parts.length == 7)
2772                  b.append("SNOMED CT "+m+" edition "+formatSCTDate(parts[6]));
2773                else
2774                  b.append("SNOMED CT "+m+" edition");
2775              } else
2776                b.append(describeSystem(s)+" version "+versions.get(s));
2777            }
2778          }
2779          x.addText(b.toString()); // add the version text
2780        }
2781      }
2782    }
2783  }
2784
2785  private String formatSCTDate(String ds) {
2786    SimpleDateFormat format = new SimpleDateFormat("yyyyMMdd");
2787    Date date;
2788    try {
2789      date = format.parse(ds);
2790    } catch (ParseException e) {
2791      return ds;
2792    }
2793    return new SimpleDateFormat("dd-MMM yyyy").format(date);
2794  }
2795
2796  private String describeModule(String module) {
2797    if ("900000000000207008".equals(module))
2798      return "International";
2799    if ("731000124108".equals(module))
2800      return "United States";
2801    if ("32506021000036107".equals(module))
2802      return "Australian";
2803    if ("449081005".equals(module))
2804      return "Spanish";
2805    if ("554471000005108".equals(module))
2806      return "Danish";
2807    if ("11000146104".equals(module))
2808      return "Dutch";
2809    if ("45991000052106".equals(module))
2810      return "Swedish";
2811    if ("999000041000000102".equals(module))
2812      return "United Kingdon";
2813    return module;
2814  }
2815
2816  private boolean hasVersionParameter(ValueSetExpansionComponent expansion) {
2817    for (ValueSetExpansionParameterComponent p : expansion.getParameter()) {
2818      if (p.getName().equals("version"))
2819        return true;
2820    }
2821    return false;
2822  }
2823
2824  private void addLanguageRow(ValueSetExpansionContainsComponent c, XhtmlNode t, List<String> langs) {
2825    XhtmlNode tr = t.tr();
2826    tr.td().addText(c.getCode());
2827    for (String lang : langs) {
2828      String d = null;
2829      for (Extension ext : c.getExtension()) {
2830        if (ToolingExtensions.EXT_TRANSLATION.equals(ext.getUrl())) {
2831          String l = ToolingExtensions.readStringExtension(ext, "lang");
2832          if (lang.equals(l))
2833            d = ToolingExtensions.readStringExtension(ext, "content");;
2834        }
2835      }
2836      tr.td().addText(d == null ? "" : d);
2837    }
2838    for (ValueSetExpansionContainsComponent cc : c.getContains()) {
2839      addLanguageRow(cc, t, langs);
2840    }
2841  }
2842
2843
2844  private String describeLang(String lang) {
2845    ValueSet v = context.fetchResource(ValueSet.class, "http://hl7.org/fhir/ValueSet/languages");
2846    if (v != null) {
2847      ConceptReferenceComponent l = null;
2848      for (ConceptReferenceComponent cc : v.getCompose().getIncludeFirstRep().getConcept()) {
2849        if (cc.getCode().equals(lang))
2850          l = cc;
2851      }
2852      if (l == null) {
2853        if (lang.contains("-"))
2854          lang = lang.substring(0, lang.indexOf("-"));
2855        for (ConceptReferenceComponent cc : v.getCompose().getIncludeFirstRep().getConcept()) {
2856          if (cc.getCode().equals(lang) || cc.getCode().startsWith(lang+"-"))
2857            l = cc;
2858        }
2859      }
2860      if (l != null) {
2861        if (lang.contains("-"))
2862          lang = lang.substring(0, lang.indexOf("-"));
2863        String en = l.getDisplay();
2864        String nativelang = null;
2865        for (ConceptReferenceDesignationComponent cd : l.getDesignation()) {
2866          if (cd.getLanguage().equals(lang))
2867            nativelang = cd.getValue();
2868        }
2869        if (nativelang == null)
2870          return en+" ("+lang+")";
2871        else
2872          return nativelang+" ("+en+", "+lang+")";
2873      }
2874    }
2875    return lang;
2876  }
2877
2878
2879  private boolean checkDoDefinition(List<ValueSetExpansionContainsComponent> contains) {
2880    for (ValueSetExpansionContainsComponent c : contains) {
2881      CodeSystem cs = context.fetchCodeSystem(c.getSystem());
2882      if (cs != null)
2883        return true;
2884      if (checkDoDefinition(c.getContains()))
2885        return true;
2886    }
2887    return false;
2888  }
2889
2890
2891  private boolean allFromOneSystem(ValueSet vs) {
2892    if (vs.getExpansion().getContains().isEmpty())
2893      return false;
2894    String system = vs.getExpansion().getContains().get(0).getSystem();
2895    for (ValueSetExpansionContainsComponent cc : vs.getExpansion().getContains()) {
2896      if (!checkSystemMatches(system, cc))
2897        return false;
2898    }
2899    return true;
2900  }
2901
2902
2903  private boolean checkSystemMatches(String system, ValueSetExpansionContainsComponent cc) {
2904    if (!system.equals(cc.getSystem()))
2905      return false;
2906    for (ValueSetExpansionContainsComponent cc1 : cc.getContains()) {
2907      if (!checkSystemMatches(system, cc1))
2908        return false;
2909    }
2910     return true;
2911  }
2912
2913
2914  private boolean checkDoSystem(ValueSet vs, ValueSet src) {
2915    if (src != null)
2916      vs = src;
2917    if (vs.hasCompose())
2918      return true;
2919    return false;
2920  }
2921
2922  private boolean IsNotFixedExpansion(ValueSet vs) {
2923    if (vs.hasCompose())
2924      return false;
2925
2926
2927    // it's not fixed if it has any includes that are not version fixed
2928    for (ConceptSetComponent cc : vs.getCompose().getInclude()) {
2929      if (cc.hasValueSet())
2930        return true;
2931      if (!cc.hasVersion())
2932        return true;
2933    }
2934    return false;
2935  }
2936
2937
2938  private void addLanguageRow(ConceptDefinitionComponent c, XhtmlNode t, List<String> langs) {
2939    XhtmlNode tr = t.tr();
2940    tr.td().addText(c.getCode());
2941    for (String lang : langs) {
2942      ConceptDefinitionDesignationComponent d = null;
2943      for (ConceptDefinitionDesignationComponent designation : c.getDesignation()) {
2944        if (lang.equals(designation.getLanguage()))
2945          d = designation;
2946      }
2947      tr.td().addText(d == null ? "" : d.getValue());
2948    }
2949  }
2950
2951  private void scanLangs(ConceptDefinitionComponent c, List<String> langs) {
2952    for (ConceptDefinitionDesignationComponent designation : c.getDesignation()) {
2953      String lang = designation.getLanguage();
2954      if (langs != null && !langs.contains(lang))
2955        langs.add(lang);
2956    }
2957    for (ConceptDefinitionComponent g : c.getConcept())
2958      scanLangs(g, langs);
2959  }
2960
2961  private void addMapHeaders(XhtmlNode tr, Map<ConceptMap, String> mymaps) throws FHIRFormatError, DefinitionException, IOException {
2962          for (ConceptMap m : mymaps.keySet()) {
2963                XhtmlNode td = tr.td();
2964                XhtmlNode b = td.b();
2965                XhtmlNode a = b.ah(prefix+mymaps.get(m));
2966      a.addText(m.getName());
2967      if (m.hasDescription())
2968        addMarkdown(td, m.getDescription());
2969          }
2970  }
2971
2972        private void smartAddText(XhtmlNode p, String text) {
2973          if (text == null)
2974            return;
2975
2976    String[] lines = text.split("\\r\\n");
2977    for (int i = 0; i < lines.length; i++) {
2978      if (i > 0)
2979        p.br();
2980      p.addText(lines[i]);
2981    }
2982  }
2983
2984  private boolean conceptsHaveComments(ConceptDefinitionComponent c) {
2985    if (ToolingExtensions.hasCSComment(c))
2986      return true;
2987    for (ConceptDefinitionComponent g : c.getConcept())
2988      if (conceptsHaveComments(g))
2989        return true;
2990    return false;
2991  }
2992
2993  private boolean conceptsHaveDisplay(ConceptDefinitionComponent c) {
2994    if (c.hasDisplay())
2995      return true;
2996    for (ConceptDefinitionComponent g : c.getConcept())
2997      if (conceptsHaveDisplay(g))
2998        return true;
2999    return false;
3000  }
3001
3002  private boolean conceptsHaveDeprecated(CodeSystem cs, ConceptDefinitionComponent c) {
3003    if (CodeSystemUtilities.isDeprecated(cs, c))
3004      return true;
3005    for (ConceptDefinitionComponent g : c.getConcept())
3006      if (conceptsHaveDeprecated(cs, g))
3007        return true;
3008    return false;
3009  }
3010
3011  private void generateCopyright(XhtmlNode x, ValueSet vs) {
3012    XhtmlNode p = x.para();
3013    p.b().tx("Copyright Statement:");
3014    smartAddText(p, " " + vs.getCopyright());
3015  }
3016
3017
3018  private XhtmlNode addTableHeaderRowStandard(XhtmlNode t, boolean hasHierarchy, boolean hasDisplay, boolean definitions, boolean comments, boolean deprecated) {
3019    XhtmlNode tr = t.tr();
3020    if (hasHierarchy)
3021      tr.td().b().tx("Lvl");
3022    tr.td().b().tx("Code");
3023    if (hasDisplay)
3024      tr.td().b().tx("Display");
3025    if (definitions)
3026      tr.td().b().tx("Definition");
3027    if (deprecated)
3028      tr.td().b().tx("Deprecated");
3029    if (comments)
3030      tr.td().b().tx("Comments");
3031    return tr;
3032  }
3033
3034  private void addExpansionRowToTable(XhtmlNode t, ValueSetExpansionContainsComponent c, int i, boolean doSystem, boolean doDefinition, Map<ConceptMap, String> mymaps, CodeSystem allCS, List<String> langs) {
3035    XhtmlNode tr = t.tr();
3036    XhtmlNode td = tr.td();
3037
3038    String tgt = makeAnchor(c.getSystem(), c.getCode());
3039    td.an(tgt);
3040
3041    String s = Utilities.padLeft("", '\u00A0', i*2);
3042
3043    td.addText(s);
3044    addCodeToTable(c.getAbstract(), c.getSystem(), c.getCode(), c.getDisplay(), td);
3045    if (doSystem) {
3046      td = tr.td();
3047      td.addText(c.getSystem());
3048    }
3049    td = tr.td();
3050    if (c.hasDisplayElement())
3051      td.addText(c.getDisplay());
3052
3053    if (doDefinition) {
3054      CodeSystem cs = allCS;
3055      if (cs == null)
3056        cs = context.fetchCodeSystem(c.getSystem());
3057      td = tr.td();
3058      if (cs != null)
3059        td.addText(CodeSystemUtilities.getCodeDefinition(cs, c.getCode()));
3060    }
3061    for (ConceptMap m : mymaps.keySet()) {
3062      td = tr.td();
3063      List<TargetElementComponentWrapper> mappings = findMappingsForCode(c.getCode(), m);
3064      boolean first = true;
3065      for (TargetElementComponentWrapper mapping : mappings) {
3066        if (!first)
3067            td.br();
3068        first = false;
3069        XhtmlNode span = td.span(null, mapping.comp.getEquivalence().toString());
3070        span.addText(getCharForEquivalence(mapping.comp));
3071        XhtmlNode a = td.ah(prefix+mymaps.get(m)+"#"+mapping.comp.getCode());
3072        a.addText(mapping.comp.getCode());
3073        if (!Utilities.noString(mapping.comp.getComment()))
3074          td.i().tx("("+mapping.comp.getComment()+")");
3075      }
3076    }
3077    for (Extension ext : c.getExtension()) {
3078      if (ToolingExtensions.EXT_TRANSLATION.equals(ext.getUrl())) {
3079        String lang = ToolingExtensions.readStringExtension(ext,  "lang");
3080        if (!Utilities.noString(lang) && !langs.contains(lang))
3081          langs.add(lang);
3082      }
3083    }
3084    for (ValueSetExpansionContainsComponent cc : c.getContains()) {
3085      addExpansionRowToTable(t, cc, i+1, doSystem, doDefinition, mymaps, allCS, langs);
3086    }
3087  }
3088
3089  private void addCodeToTable(boolean isAbstract, String system, String code, String display, XhtmlNode td) {
3090    CodeSystem e = context.fetchCodeSystem(system);
3091    if (e == null || e.getContent() != org.hl7.fhir.dstu3.model.CodeSystem.CodeSystemContentMode.COMPLETE) {
3092      if (isAbstract)
3093        td.i().setAttribute("title", ABSTRACT_CODE_HINT).addText(code);
3094      else if ("http://snomed.info/sct".equals(system)) {
3095        td.ah("http://browser.ihtsdotools.org/?perspective=full&conceptId1="+code).addText(code);
3096      } else if ("http://loinc.org".equals(system)) {
3097          td.ah(LoincLinker.getLinkForCode(code)).addText(code);
3098      } else        
3099        td.addText(code);
3100    } else {
3101      String href = prefix+getCsRef(e);
3102      if (href.contains("#"))
3103        href = href + "-"+Utilities.nmtokenize(code);
3104      else
3105        href = href + "#"+e.getId()+"-"+Utilities.nmtokenize(code);
3106      if (isAbstract)
3107        td.ah(href).setAttribute("title", ABSTRACT_CODE_HINT).i().addText(code);
3108      else
3109        td.ah(href).addText(code);
3110    }
3111  }
3112
3113  private class TargetElementComponentWrapper {
3114    private ConceptMapGroupComponent group;
3115    private TargetElementComponent comp;
3116    public TargetElementComponentWrapper(ConceptMapGroupComponent group, TargetElementComponent comp) {
3117      super();
3118      this.group = group;
3119      this.comp = comp;
3120    }
3121
3122  }
3123
3124  private boolean addDefineRowToTable(XhtmlNode t, ConceptDefinitionComponent c, int i, boolean hasHierarchy, boolean hasDisplay, boolean comment, boolean deprecated, Map<ConceptMap, String> maps, String system, CodeSystem cs) {
3125    boolean hasExtensions = false;
3126    XhtmlNode tr = t.tr();
3127    XhtmlNode td = tr.td();
3128    if (hasHierarchy) {
3129      td.addText(Integer.toString(i+1));
3130      td = tr.td();
3131      String s = Utilities.padLeft("", '\u00A0', i*2);
3132      td.addText(s);
3133    }
3134    td.addText(c.getCode());
3135    XhtmlNode a;
3136    if (c.hasCodeElement()) {
3137      td.an(cs.getId()+"-" + Utilities.nmtokenize(c.getCode()));
3138    }
3139
3140    if (hasDisplay) {
3141      td = tr.td();
3142      if (c.hasDisplayElement())
3143        td.addText(c.getDisplay());
3144    }
3145    td = tr.td();
3146    if (c != null)
3147      smartAddText(td, c.getDefinition());
3148    if (deprecated) {
3149      td = tr.td();
3150      Boolean b = CodeSystemUtilities.isDeprecated(cs, c);
3151      if (b !=  null && b) {
3152        smartAddText(td, "Deprecated");
3153        hasExtensions = true;
3154        if (ToolingExtensions.hasExtension(c, ToolingExtensions.EXT_REPLACED_BY)) {
3155          Coding cc = (Coding) ToolingExtensions.getExtension(c, ToolingExtensions.EXT_REPLACED_BY).getValue();
3156          td.tx(" (replaced by ");
3157          String url = getCodingReference(cc, system);
3158          if (url != null) {
3159            td.ah(url).addText(cc.getCode());
3160            td.tx(": "+cc.getDisplay()+")");
3161          } else
3162            td.addText(cc.getCode()+" '"+cc.getDisplay()+"' in "+cc.getSystem()+")");
3163        }
3164      }
3165    }
3166    if (comment) {
3167      td = tr.td();
3168      String s = ToolingExtensions.getCSComment(c);
3169      if (s != null) {
3170        smartAddText(td, s);
3171        hasExtensions = true;
3172      }
3173    }
3174    for (ConceptMap m : maps.keySet()) {
3175      td = tr.td();
3176      List<TargetElementComponentWrapper> mappings = findMappingsForCode(c.getCode(), m);
3177      boolean first = true;
3178      for (TargetElementComponentWrapper mapping : mappings) {
3179        if (!first)
3180                  td.br();
3181        first = false;
3182        XhtmlNode span = td.span(null, mapping.comp.hasEquivalence() ?  mapping.comp.getEquivalence().toCode() : "");
3183        span.addText(getCharForEquivalence(mapping.comp));
3184        a = td.ah(prefix+maps.get(m)+"#"+makeAnchor(mapping.group.getTarget(), mapping.comp.getCode()));
3185        a.addText(mapping.comp.getCode());
3186        if (!Utilities.noString(mapping.comp.getComment()))
3187          td.i().tx("("+mapping.comp.getComment()+")");
3188      }
3189    }
3190    for (CodeType e : ToolingExtensions.getSubsumes(c)) {
3191      hasExtensions = true;
3192      tr = t.tr();
3193      td = tr.td();
3194      String s = Utilities.padLeft("", '.', i*2);
3195      td.addText(s);
3196      a = td.ah("#"+Utilities.nmtokenize(e.getValue()));
3197      a.addText(c.getCode());
3198    }
3199    for (ConceptDefinitionComponent cc : c.getConcept()) {
3200      hasExtensions = addDefineRowToTable(t, cc, i+1, hasHierarchy, hasDisplay, comment, deprecated, maps, system, cs) || hasExtensions;
3201    }
3202    return hasExtensions;
3203  }
3204
3205
3206  private String makeAnchor(String codeSystem, String code) {
3207    String s = codeSystem+'-'+code;
3208    StringBuilder b = new StringBuilder();
3209    for (char c : s.toCharArray()) {
3210      if (Character.isAlphabetic(c) || Character.isDigit(c) || c == '.')
3211        b.append(c);
3212      else
3213        b.append('-');
3214    }
3215    return b.toString();
3216  }
3217
3218  private String getCodingReference(Coding cc, String system) {
3219    if (cc.getSystem().equals(system))
3220      return "#"+cc.getCode();
3221    if (cc.getSystem().equals("http://snomed.info/sct"))
3222      return "http://snomed.info/sct/"+cc.getCode();
3223    if (cc.getSystem().equals("http://loinc.org"))
3224      return LoincLinker.getLinkForCode(cc.getCode());
3225    return null;
3226  }
3227
3228  private String getCharForEquivalence(TargetElementComponent mapping) {
3229    if (!mapping.hasEquivalence())
3230      return "";
3231          switch (mapping.getEquivalence()) {
3232          case EQUAL : return "=";
3233          case EQUIVALENT : return "~";
3234          case WIDER : return "<";
3235          case NARROWER : return ">";
3236          case INEXACT : return "><";
3237          case UNMATCHED : return "-";
3238          case DISJOINT : return "!=";
3239    case NULL: return null;
3240            default: return "?";
3241          }
3242  }
3243
3244  private List<TargetElementComponentWrapper> findMappingsForCode(String code, ConceptMap map) {
3245    List<TargetElementComponentWrapper> mappings = new ArrayList<TargetElementComponentWrapper>();
3246
3247    for (ConceptMapGroupComponent g : map.getGroup()) {
3248      for (SourceElementComponent c : g.getElement()) {
3249                if (c.getCode().equals(code))
3250          for (TargetElementComponent cc : c.getTarget())
3251            mappings.add(new TargetElementComponentWrapper(g, cc));
3252      }
3253          }
3254          return mappings;
3255  }
3256
3257  private boolean generateComposition(ResourceContext rcontext, XhtmlNode x, ValueSet vs, boolean header) throws FHIRException, IOException {
3258          boolean hasExtensions = false;
3259    List<String> langs = new ArrayList<String>();
3260
3261    if (header) {
3262      XhtmlNode h = x.h2();
3263      h.addText(vs.getName());
3264      addMarkdown(x, vs.getDescription());
3265      if (vs.hasCopyrightElement())
3266        generateCopyright(x, vs);
3267    }
3268    XhtmlNode p = x.para();
3269    p.tx("This value set includes codes from the following code systems:");
3270
3271    XhtmlNode ul = x.ul();
3272    XhtmlNode li;
3273    for (ConceptSetComponent inc : vs.getCompose().getInclude()) {
3274      hasExtensions = genInclude(rcontext, ul, inc, "Include", langs) || hasExtensions;
3275    }
3276    for (ConceptSetComponent exc : vs.getCompose().getExclude()) {
3277      hasExtensions = genInclude(rcontext, ul, exc, "Exclude", langs) || hasExtensions;
3278    }
3279
3280    // now, build observed languages
3281
3282    if (langs.size() > 0) {
3283      Collections.sort(langs);
3284      x.para().b().tx("Additional Language Displays");
3285      XhtmlNode t = x.table( "codes");
3286      XhtmlNode tr = t.tr();
3287      tr.td().b().tx("Code");
3288      for (String lang : langs)
3289        tr.td().b().addText(describeLang(lang));
3290      for (ConceptSetComponent c : vs.getCompose().getInclude()) {
3291        for (ConceptReferenceComponent cc : c.getConcept()) {
3292          addLanguageRow(cc, t, langs);
3293        }
3294      }
3295    }
3296
3297    return hasExtensions;
3298  }
3299
3300    private void addLanguageRow(ConceptReferenceComponent c, XhtmlNode t, List<String> langs) {
3301      XhtmlNode tr = t.tr();
3302      tr.td().addText(c.getCode());
3303      for (String lang : langs) {
3304        String d = null;
3305        for (ConceptReferenceDesignationComponent cd : c.getDesignation()) {
3306          String l = cd.getLanguage();
3307          if (lang.equals(l))
3308            d = cd.getValue();
3309        }
3310        tr.td().addText(d == null ? "" : d);
3311      }
3312    }
3313
3314  private void AddVsRef(ResourceContext rcontext, String value, XhtmlNode li) {
3315    Resource res = rcontext == null ? null : rcontext.resolve(value); 
3316    if (res != null && !(res instanceof MetadataResource)) {
3317      li.addText(value);
3318      return;      
3319    }      
3320    MetadataResource vs = (MetadataResource) res;
3321    if (vs == null)
3322                vs = context.fetchResource(ValueSet.class, value);
3323    if (vs == null)
3324                vs = context.fetchResource(StructureDefinition.class, value);
3325//    if (vs == null)
3326        //      vs = context.fetchResource(DataElement.class, value);
3327    if (vs == null)
3328                vs = context.fetchResource(Questionnaire.class, value);
3329    if (vs != null) {
3330      String ref = (String) vs.getUserData("path");
3331      
3332      ref = adjustForPath(ref);
3333      XhtmlNode a = li.ah(ref == null ? "??" : ref.replace("\\", "/"));
3334      a.addText(value);
3335    } else {
3336        CodeSystem cs = context.fetchCodeSystem(value);
3337        if (cs != null) {
3338        String ref = (String) cs.getUserData("path");
3339        ref = adjustForPath(ref);
3340        XhtmlNode a = li.ah(ref == null ? "??" : ref.replace("\\", "/"));
3341        a.addText(value);
3342            } else if (value.equals("http://snomed.info/sct") || value.equals("http://snomed.info/id")) {
3343              XhtmlNode a = li.ah(value);
3344              a.tx("SNOMED-CT");
3345            }
3346            else {
3347              if (value.startsWith("http://hl7.org") && !Utilities.existsInList(value, "http://hl7.org/fhir/sid/icd-10-us"))
3348                System.out.println("Unable to resolve value set "+value);
3349              li.addText(value);
3350    }
3351  }
3352        }
3353
3354  private String adjustForPath(String ref) {
3355    if (prefix == null)
3356      return ref;
3357    else
3358      return prefix+ref;
3359  }
3360
3361  private boolean genInclude(ResourceContext rcontext, XhtmlNode ul, ConceptSetComponent inc, String type, List<String> langs) throws FHIRException {
3362    boolean hasExtensions = false;
3363    XhtmlNode li;
3364    li = ul.li();
3365    CodeSystem e = context.fetchCodeSystem(inc.getSystem());
3366
3367    if (inc.hasSystem()) {
3368      if (inc.getConcept().size() == 0 && inc.getFilter().size() == 0) {
3369        li.addText(type+" all codes defined in ");
3370        addCsRef(inc, li, e);
3371      } else {
3372        if (inc.getConcept().size() > 0) {
3373          li.addText(type+" these codes as defined in ");
3374          addCsRef(inc, li, e);
3375
3376          XhtmlNode t = li.table("none");
3377          boolean hasComments = false;
3378          boolean hasDefinition = false;
3379          for (ConceptReferenceComponent c : inc.getConcept()) {
3380            hasComments = hasComments || ExtensionHelper.hasExtension(c, ToolingExtensions.EXT_VS_COMMENT);
3381            hasDefinition = hasDefinition || ExtensionHelper.hasExtension(c, ToolingExtensions.EXT_DEFINITION);
3382          }
3383          if (hasComments || hasDefinition)
3384            hasExtensions = true;
3385          addTableHeaderRowStandard(t, false, true, hasDefinition, hasComments, false);
3386          for (ConceptReferenceComponent c : inc.getConcept()) {
3387            XhtmlNode tr = t.tr();
3388            XhtmlNode td = tr.td();
3389            ConceptDefinitionComponent cc = getConceptForCode(e, c.getCode(), inc);
3390            addCodeToTable(false, inc.getSystem(), c.getCode(), c.hasDisplay()? c.getDisplay() : cc != null ? cc.getDisplay() : "", td);
3391
3392            td = tr.td();
3393            if (!Utilities.noString(c.getDisplay()))
3394              td.addText(c.getDisplay());
3395            else if (cc != null && !Utilities.noString(cc.getDisplay()))
3396              td.addText(cc.getDisplay());
3397
3398            td = tr.td();
3399            if (ExtensionHelper.hasExtension(c, ToolingExtensions.EXT_DEFINITION))
3400              smartAddText(td, ToolingExtensions.readStringExtension(c, ToolingExtensions.EXT_DEFINITION));
3401            else if (cc != null && !Utilities.noString(cc.getDefinition()))
3402              smartAddText(td, cc.getDefinition());
3403
3404            if (ExtensionHelper.hasExtension(c, ToolingExtensions.EXT_VS_COMMENT)) {
3405              smartAddText(tr.td(), "Note: "+ToolingExtensions.readStringExtension(c, ToolingExtensions.EXT_VS_COMMENT));
3406            }
3407            for (ConceptReferenceDesignationComponent cd : c.getDesignation()) {
3408              if (cd.hasLanguage() && !langs.contains(cd.getLanguage()))
3409                langs.add(cd.getLanguage());
3410            }
3411          }
3412        }
3413        boolean first = true;
3414        for (ConceptSetFilterComponent f : inc.getFilter()) {
3415          if (first) {
3416            li.addText(type+" codes from ");
3417            first = false;
3418          } else
3419            li.tx(" and ");
3420          addCsRef(inc, li, e);
3421          li.tx(" where "+f.getProperty()+" "+describe(f.getOp())+" ");
3422          if (e != null && codeExistsInValueSet(e, f.getValue())) {
3423            String href = prefix+getCsRef(e);
3424            if (href.contains("#"))
3425              href = href + "-"+Utilities.nmtokenize(f.getValue());
3426            else
3427              href = href + "#"+e.getId()+"-"+Utilities.nmtokenize(f.getValue());
3428            li.ah(href).addText(f.getValue());
3429          } else if ("concept".equals(f.getProperty()) && inc.hasSystem()) {
3430            li.addText(f.getValue());
3431            ValidationResult vr = context.validateCode(inc.getSystem(), f.getValue(), null);
3432            if (vr.isOk()) {
3433              li.tx(" ("+vr.getDisplay()+")");
3434            }
3435          }
3436          else
3437            li.addText(f.getValue());
3438          String disp = ToolingExtensions.getDisplayHint(f);
3439          if (disp != null)
3440            li.tx(" ("+disp+")");
3441        }
3442      }
3443      if (inc.hasValueSet()) {
3444        li.tx(", where the codes are contained in ");
3445        boolean first = true;
3446        for (UriType vs : inc.getValueSet()) {
3447          if (first)
3448            first = false;
3449          else
3450            li.tx(", ");
3451          AddVsRef(rcontext, vs.asStringValue(), li);
3452        }
3453      }
3454    } else {
3455      li = ul.li();
3456      li.tx("Import all the codes that are contained in ");
3457      boolean first = true;
3458      for (UriType vs : inc.getValueSet()) {
3459        if (first)
3460          first = false;
3461        else
3462          li.tx(", ");
3463        AddVsRef(rcontext, vs.asStringValue(), li);
3464      }
3465    }
3466    return hasExtensions;
3467  }
3468
3469  private String describe(FilterOperator op) {
3470    switch (op) {
3471    case EQUAL: return " = ";
3472    case ISA: return " is-a ";
3473    case ISNOTA: return " is-not-a ";
3474    case REGEX: return " matches (by regex) ";
3475                case NULL: return " ?? ";
3476                case IN: return " in ";
3477                case NOTIN: return " not in ";
3478    case DESCENDENTOF: return " descends from ";
3479    case EXISTS: return " exists ";
3480    case GENERALIZES: return " generalizes ";
3481    }
3482    return null;
3483  }
3484
3485  private ConceptDefinitionComponent getConceptForCode(CodeSystem e, String code, ConceptSetComponent inc) {
3486    // first, look in the code systems
3487    if (e == null)
3488    e = context.fetchCodeSystem(inc.getSystem());
3489    if (e != null) {
3490      ConceptDefinitionComponent v = getConceptForCode(e.getConcept(), code);
3491      if (v != null)
3492        return v;
3493    }
3494
3495    if (!context.hasCache()) {
3496      ValueSetExpansionComponent vse;
3497      try {
3498        vse = context.expandVS(inc, false);
3499      } catch (TerminologyServiceException e1) {
3500        return null;
3501      }
3502      if (vse != null) {
3503        ConceptDefinitionComponent v = getConceptForCodeFromExpansion(vse.getContains(), code);
3504      if (v != null)
3505        return v;
3506    }
3507    }
3508
3509    return context.validateCode(inc.getSystem(), code, null).asConceptDefinition();
3510  }
3511
3512
3513
3514  private ConceptDefinitionComponent getConceptForCode(List<ConceptDefinitionComponent> list, String code) {
3515    for (ConceptDefinitionComponent c : list) {
3516    if (code.equals(c.getCode()))
3517      return c;
3518      ConceptDefinitionComponent v = getConceptForCode(c.getConcept(), code);
3519      if (v != null)
3520        return v;
3521    }
3522    return null;
3523  }
3524
3525  private ConceptDefinitionComponent getConceptForCodeFromExpansion(List<ValueSetExpansionContainsComponent> list, String code) {
3526    for (ValueSetExpansionContainsComponent c : list) {
3527      if (code.equals(c.getCode())) {
3528        ConceptDefinitionComponent res = new ConceptDefinitionComponent();
3529        res.setCode(c.getCode());
3530        res.setDisplay(c.getDisplay());
3531        return res;
3532      }
3533      ConceptDefinitionComponent v = getConceptForCodeFromExpansion(c.getContains(), code);
3534      if (v != null)
3535        return v;
3536    }
3537    return null;
3538  }
3539
3540  private  <T extends Resource> void addCsRef(ConceptSetComponent inc, XhtmlNode li, T cs) {
3541    String ref = null;
3542    if (cs != null) {
3543      ref = (String) cs.getUserData("filename");
3544      if (Utilities.noString(ref))
3545        ref = (String) cs.getUserData("path");
3546    }
3547    String spec = getSpecialReference(inc.getSystem());
3548    if (spec != null) {
3549      XhtmlNode a = li.ah(spec);
3550      a.code(inc.getSystem());
3551    } else if (cs != null && ref != null) {
3552      if (!Utilities.noString(prefix) && ref.startsWith("http://hl7.org/fhir/"))
3553        ref = ref.substring(20)+"/index.html";
3554      else if (!ref.contains(".html"))
3555        ref = ref + ".html";
3556      XhtmlNode a = li.ah(prefix+ref.replace("\\", "/"));
3557      a.code(inc.getSystem());
3558    } else {
3559      li.code(inc.getSystem());
3560    }
3561  }
3562
3563  private String getSpecialReference(String system) {
3564    if ("http://snomed.info/sct".equals(system))
3565      return "http://www.snomed.org/";
3566    if (Utilities.existsInList(system, "http://loinc.org", "http://unitsofmeasure.org", "http://www.nlm.nih.gov/research/umls/rxnorm", "http://ncimeta.nci.nih.gov", "http://fdasis.nlm.nih.gov", 
3567         "http://www.radlex.org", "http://www.whocc.no/atc", "http://dicom.nema.org/resources/ontology/DCM", "http://www.genenames.org", "http://www.ensembl.org", "http://www.ncbi.nlm.nih.gov/nuccore", 
3568         "http://www.ncbi.nlm.nih.gov/clinvar", "http://sequenceontology.org", "http://www.hgvs.org/mutnomen", "http://www.ncbi.nlm.nih.gov/projects/SNP", "http://cancer.sanger.ac.uk/cancergenome/projects/cosmic", 
3569         "http://www.lrg-sequence.org", "http://www.omim.org", "http://www.ncbi.nlm.nih.gov/pubmed", "http://www.pharmgkb.org", "http://clinicaltrials.gov", "http://www.ebi.ac.uk/ipd/imgt/hla/")) 
3570      return system;
3571      
3572    return null;
3573  }
3574
3575  private String getCsRef(String system) {
3576    CodeSystem cs = context.fetchCodeSystem(system);
3577    return getCsRef(cs);
3578  }
3579
3580  private  <T extends Resource> String getCsRef(T cs) {
3581    String ref = (String) cs.getUserData("filename");
3582    if (ref == null)
3583      ref = (String) cs.getUserData("path");
3584    if (ref == null)
3585      return "??.html";
3586    if (!ref.contains(".html"))
3587      ref = ref + ".html";
3588    return ref.replace("\\", "/");
3589  }
3590
3591  private boolean codeExistsInValueSet(CodeSystem cs, String code) {
3592    for (ConceptDefinitionComponent c : cs.getConcept()) {
3593      if (inConcept(code, c))
3594        return true;
3595    }
3596    return false;
3597  }
3598
3599  private boolean inConcept(String code, ConceptDefinitionComponent c) {
3600    if (c.hasCodeElement() && c.getCode().equals(code))
3601      return true;
3602    for (ConceptDefinitionComponent g : c.getConcept()) {
3603      if (inConcept(code, g))
3604        return true;
3605    }
3606    return false;
3607  }
3608
3609  /**
3610   * This generate is optimised for the build tool in that it tracks the source extension.
3611   * But it can be used for any other use.
3612   *
3613   * @param vs
3614   * @param codeSystems
3615   * @throws DefinitionException
3616   * @throws Exception
3617   */
3618  public boolean generate(ResourceContext rcontext, OperationOutcome op) throws DefinitionException {
3619    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
3620    boolean hasSource = false;
3621    boolean success = true;
3622    for (OperationOutcomeIssueComponent i : op.getIssue()) {
3623        success = success && i.getSeverity() == IssueSeverity.INFORMATION;
3624        hasSource = hasSource || ExtensionHelper.hasExtension(i, ToolingExtensions.EXT_ISSUE_SOURCE);
3625    }
3626    if (success)
3627        x.para().tx("All OK");
3628    if (op.getIssue().size() > 0) {
3629                XhtmlNode tbl = x.table("grid"); // on the basis that we'll most likely be rendered using the standard fhir css, but it doesn't really matter
3630                XhtmlNode tr = tbl.tr();
3631                tr.td().b().tx("Severity");
3632                tr.td().b().tx("Location");
3633        tr.td().b().tx("Code");
3634        tr.td().b().tx("Details");
3635        tr.td().b().tx("Diagnostics");
3636                if (hasSource)
3637                        tr.td().b().tx("Source");
3638                for (OperationOutcomeIssueComponent i : op.getIssue()) {
3639                        tr = tbl.tr();
3640                        tr.td().addText(i.getSeverity().toString());
3641                        XhtmlNode td = tr.td();
3642                        boolean d = false;
3643                        for (StringType s : i.getLocation()) {
3644                                if (d)
3645                                        td.tx(", ");
3646                                else
3647                                        d = true;
3648                                td.addText(s.getValue());
3649                        }
3650          tr.td().addText(i.getCode().getDisplay());
3651          tr.td().addText(gen(i.getDetails()));
3652          smartAddText(tr.td(), i.getDiagnostics());
3653                        if (hasSource) {
3654                                Extension ext = ExtensionHelper.getExtension(i, ToolingExtensions.EXT_ISSUE_SOURCE);
3655            tr.td().addText(ext == null ? "" : gen(ext));
3656                        }
3657                }
3658        }
3659    inject(op, x, hasSource ? NarrativeStatus.EXTENSIONS :  NarrativeStatus.GENERATED);
3660    return true;
3661  }
3662
3663
3664  public String genType(Type type) throws DefinitionException {
3665    if (type instanceof Coding)
3666      return gen((Coding) type);
3667    if (type instanceof CodeableConcept)
3668      return displayCodeableConcept((CodeableConcept) type);
3669    if (type instanceof Quantity)
3670      return displayQuantity((Quantity) type);
3671    if (type instanceof Range)
3672      return displayRange((Range) type);
3673    return null;
3674  }
3675        private String gen(Extension extension) throws DefinitionException {
3676                if (extension.getValue() instanceof CodeType)
3677                        return ((CodeType) extension.getValue()).getValue();
3678                if (extension.getValue() instanceof Coding)
3679                        return gen((Coding) extension.getValue());
3680
3681          throw new DefinitionException("Unhandled type "+extension.getValue().getClass().getName());
3682  }
3683
3684        public String gen(CodeableConcept code) {
3685                if (code == null)
3686                return null;
3687                if (code.hasText())
3688                        return code.getText();
3689                if (code.hasCoding())
3690                        return gen(code.getCoding().get(0));
3691                return null;
3692        }
3693
3694        public String gen(Coding code) {
3695          if (code == null)
3696                return null;
3697          if (code.hasDisplayElement())
3698                return code.getDisplay();
3699          if (code.hasCodeElement())
3700                return code.getCode();
3701          return null;
3702  }
3703
3704  public boolean generate(ResourceContext rcontext, StructureDefinition sd, java.util.Set<String> outputTracker) throws EOperationOutcome, FHIRException, IOException {
3705    ProfileUtilities pu = new ProfileUtilities(context, null, pkp);
3706    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
3707    x.addChildNode(pu.generateTable(definitionsTarget, sd, true, destDir, false, sd.getId(), false, corePath, "", false, false, outputTracker));
3708    inject(sd, x, NarrativeStatus.GENERATED);
3709    return true;
3710  }
3711  
3712  public boolean generate(ResourceContext rcontext, ImplementationGuide ig) throws EOperationOutcome, FHIRException, IOException {
3713    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
3714    x.h2().addText(ig.getName());
3715    x.para().tx("The official URL for this implementation guide is: ");
3716    x.pre().tx(ig.getUrl());
3717    addMarkdown(x, ig.getDescription());
3718    inject(ig, x, NarrativeStatus.GENERATED);
3719    return true;
3720  }
3721        public boolean generate(ResourceContext rcontext, OperationDefinition opd) throws EOperationOutcome, FHIRException, IOException {
3722    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
3723    x.h2().addText(opd.getName());
3724    x.para().addText(Utilities.capitalize(opd.getKind().toString())+": "+opd.getName());
3725    x.para().tx("The official URL for this operation definition is: ");
3726    x.pre().tx(opd.getUrl());
3727    addMarkdown(x, opd.getDescription());
3728
3729    if (opd.getSystem())
3730      x.para().tx("URL: [base]/$"+opd.getCode());
3731    for (CodeType c : opd.getResource()) {
3732      if (opd.getType())
3733        x.para().tx("URL: [base]/"+c.getValue()+"/$"+opd.getCode());
3734      if (opd.getInstance())
3735        x.para().tx("URL: [base]/"+c.getValue()+"/[id]/$"+opd.getCode());
3736    }
3737
3738    x.para().tx("Parameters");
3739    XhtmlNode tbl = x.table( "grid");
3740    XhtmlNode tr = tbl.tr();
3741    tr.td().b().tx("Use");
3742    tr.td().b().tx("Name");
3743    tr.td().b().tx("Cardinality");
3744    tr.td().b().tx("Type");
3745    tr.td().b().tx("Binding");
3746    tr.td().b().tx("Documentation");
3747    for (OperationDefinitionParameterComponent p : opd.getParameter()) {
3748      genOpParam(rcontext, tbl, "", p);
3749    }
3750    addMarkdown(x, opd.getComment());
3751    inject(opd, x, NarrativeStatus.GENERATED);
3752    return true;
3753        }
3754
3755        private void genOpParam(ResourceContext rcontext, XhtmlNode tbl, String path, OperationDefinitionParameterComponent p) throws EOperationOutcome, FHIRException, IOException {
3756                XhtmlNode tr;
3757      tr = tbl.tr();
3758      tr.td().addText(p.getUse().toString());
3759                tr.td().addText(path+p.getName());
3760      tr.td().addText(Integer.toString(p.getMin())+".."+p.getMax());
3761      tr.td().addText(p.hasType() ? p.getType() : "");
3762      XhtmlNode td = tr.td();
3763      if (p.hasBinding() && p.getBinding().hasValueSet()) {
3764        if (p.getBinding().getValueSet() instanceof Reference)
3765          AddVsRef(rcontext, p.getBinding().getValueSetReference().getReference(), td);
3766        else
3767          td.ah(p.getBinding().getValueSetUriType().getValue()).tx("External Reference");
3768        td.tx(" ("+p.getBinding().getStrength().getDisplay()+")");
3769      }
3770      addMarkdown(tr.td(), p.getDocumentation());
3771      if (!p.hasType()) {
3772                        for (OperationDefinitionParameterComponent pp : p.getPart()) {
3773                                genOpParam(rcontext, tbl, path+p.getName()+".", pp);
3774        }
3775      }
3776    }
3777
3778        private void addMarkdown(XhtmlNode x, String text) throws FHIRFormatError, IOException, DefinitionException {
3779          if (text != null) {
3780            // 1. custom FHIR extensions
3781            while (text.contains("[[[")) {
3782              String left = text.substring(0, text.indexOf("[[["));
3783              String link = text.substring(text.indexOf("[[[")+3, text.indexOf("]]]"));
3784              String right = text.substring(text.indexOf("]]]")+3);
3785              String url = link;
3786              String[] parts = link.split("\\#");
3787              StructureDefinition p = context.fetchResource(StructureDefinition.class, parts[0]);
3788              if (p == null)
3789                p = context.fetchTypeDefinition(parts[0]);
3790              if (p == null)
3791                p = context.fetchResource(StructureDefinition.class, link);
3792              if (p != null) {
3793                url = p.getUserString("path");
3794                if (url == null)
3795                  url = p.getUserString("filename");
3796              } else
3797                throw new DefinitionException("Unable to resolve markdown link "+link);
3798
3799              text = left+"["+link+"]("+url+")"+right;
3800            }
3801
3802            // 2. markdown
3803            String s =  new MarkDownProcessor(Dialect.DARING_FIREBALL).process(Utilities.escapeXml(text), "NarrativeGenerator");
3804            XhtmlParser p = new XhtmlParser();
3805            XhtmlNode m;
3806                try {
3807                        m = p.parse("<div>"+s+"</div>", "div");
3808                } catch (org.hl7.fhir.exceptions.FHIRFormatError e) {
3809                        throw new FHIRFormatError(e.getMessage(), e);
3810                }
3811            x.addChildNodes(m.getChildNodes());
3812          }
3813  }
3814
3815  public boolean generate(ResourceContext rcontext, CompartmentDefinition cpd) {
3816    StringBuilder in = new StringBuilder();
3817    StringBuilder out = new StringBuilder();
3818    for (CompartmentDefinitionResourceComponent cc: cpd.getResource()) {
3819      CommaSeparatedStringBuilder rules = new CommaSeparatedStringBuilder();
3820      if (!cc.hasParam()) {
3821        out.append(" <li><a href=\"").append(cc.getCode().toLowerCase()).append(".html\">").append(cc.getCode()).append("</a></li>\r\n");
3822      } else if (!rules.equals("{def}")) {
3823        for (StringType p : cc.getParam())
3824          rules.append(p.asStringValue());
3825        in.append(" <tr><td><a href=\"").append(cc.getCode().toLowerCase()).append(".html\">").append(cc.getCode()).append("</a></td><td>").append(rules.toString()).append("</td></tr>\r\n");
3826      }
3827    }
3828    XhtmlNode x;
3829    try {
3830      x = new XhtmlParser().parseFragment("<div><p>\r\nThe following resources may be in this compartment:\r\n</p>\r\n" +
3831          "<table class=\"grid\">\r\n"+
3832          " <tr><td><b>Resource</b></td><td><b>Inclusion Criteria</b></td></tr>\r\n"+
3833          in.toString()+
3834          "</table>\r\n"+
3835          "<p>\r\nA resource is in this compartment if the nominated search parameter (or chain) refers to the patient resource that defines the compartment.\r\n</p>\r\n" +
3836          "<p>\r\n\r\n</p>\r\n" +
3837          "<p>\r\nThe following resources are never in this compartment:\r\n</p>\r\n" +
3838          "<ul>\r\n"+
3839          out.toString()+
3840          "</ul></div>\r\n");
3841      inject(cpd, x, NarrativeStatus.GENERATED);
3842      return true;
3843    } catch (Exception e) {
3844      e.printStackTrace();
3845      return false;
3846    }
3847  }
3848
3849  public boolean generate(ResourceContext rcontext, CapabilityStatement conf) throws FHIRFormatError, DefinitionException, IOException {
3850    XhtmlNode x = new XhtmlNode(NodeType.Element, "div");
3851    x.h2().addText(conf.getName());
3852    addMarkdown(x, conf.getDescription());
3853    if (conf.getRest().size() > 0) {
3854      CapabilityStatementRestComponent rest = conf.getRest().get(0);
3855      XhtmlNode t = x.table(null);
3856      addTableRow(t, "Mode", rest.getMode().toString());
3857      addTableRow(t, "Description", rest.getDocumentation());
3858
3859      addTableRow(t, "Transaction", showOp(rest, SystemRestfulInteraction.TRANSACTION));
3860      addTableRow(t, "System History", showOp(rest, SystemRestfulInteraction.HISTORYSYSTEM));
3861      addTableRow(t, "System Search", showOp(rest, SystemRestfulInteraction.SEARCHSYSTEM));
3862
3863      t = x.table(null);
3864      XhtmlNode tr = t.tr();
3865      tr.th().b().tx("Resource Type");
3866      tr.th().b().tx("Profile");
3867      tr.th().b().tx("Read");
3868      tr.th().b().tx("V-Read");
3869      tr.th().b().tx("Search");
3870      tr.th().b().tx("Update");
3871      tr.th().b().tx("Updates");
3872      tr.th().b().tx("Create");
3873      tr.th().b().tx("Delete");
3874      tr.th().b().tx("History");
3875
3876      for (CapabilityStatementRestResourceComponent r : rest.getResource()) {
3877        tr = t.tr();
3878        tr.td().addText(r.getType());
3879        if (r.hasProfile()) {
3880          tr.td().ah(prefix+r.getProfile().getReference()).addText(r.getProfile().getReference());
3881        }
3882        tr.td().addText(showOp(r, TypeRestfulInteraction.READ));
3883        tr.td().addText(showOp(r, TypeRestfulInteraction.VREAD));
3884        tr.td().addText(showOp(r, TypeRestfulInteraction.SEARCHTYPE));
3885        tr.td().addText(showOp(r, TypeRestfulInteraction.UPDATE));
3886        tr.td().addText(showOp(r, TypeRestfulInteraction.HISTORYINSTANCE));
3887        tr.td().addText(showOp(r, TypeRestfulInteraction.CREATE));
3888        tr.td().addText(showOp(r, TypeRestfulInteraction.DELETE));
3889        tr.td().addText(showOp(r, TypeRestfulInteraction.HISTORYTYPE));
3890      }
3891    }
3892
3893    inject(conf, x, NarrativeStatus.GENERATED);
3894    return true;
3895  }
3896
3897  private String showOp(CapabilityStatementRestResourceComponent r, TypeRestfulInteraction on) {
3898    for (ResourceInteractionComponent op : r.getInteraction()) {
3899      if (op.getCode() == on)
3900        return "y";
3901    }
3902    return "";
3903  }
3904
3905  private String showOp(CapabilityStatementRestComponent r, SystemRestfulInteraction on) {
3906    for (SystemInteractionComponent op : r.getInteraction()) {
3907      if (op.getCode() == on)
3908        return "y";
3909    }
3910    return "";
3911  }
3912
3913  private void addTableRow(XhtmlNode t, String name, String value) {
3914    XhtmlNode tr = t.tr();
3915    tr.td().addText(name);
3916    tr.td().addText(value);
3917  }
3918
3919  public XhtmlNode generateDocumentNarrative(Bundle feed) {
3920    /*
3921     When the document is presented for human consumption, applications must present the collated narrative portions of the following resources in order:
3922     * The Composition resource
3923     * The Subject resource
3924     * Resources referenced in the section.content
3925     */
3926    XhtmlNode root = new XhtmlNode(NodeType.Element, "div");
3927    Composition comp = (Composition) feed.getEntry().get(0).getResource();
3928    root.addChildNode(comp.getText().getDiv());
3929    Resource subject = ResourceUtilities.getById(feed, null, comp.getSubject().getReference());
3930    if (subject != null && subject instanceof DomainResource) {
3931      root.hr();
3932      root.addChildNode(((DomainResource)subject).getText().getDiv());
3933    }
3934    List<SectionComponent> sections = comp.getSection();
3935    renderSections(feed, root, sections, 1);
3936    return root;
3937  }
3938
3939  private void renderSections(Bundle feed, XhtmlNode node, List<SectionComponent> sections, int level) {
3940    for (SectionComponent section : sections) {
3941      node.hr();
3942      if (section.hasTitleElement())
3943        node.addTag("h"+Integer.toString(level)).addText(section.getTitle());
3944//      else if (section.hasCode())
3945//        node.addTag("h"+Integer.toString(level)).addText(displayCodeableConcept(section.getCode()));
3946
3947//      if (section.hasText()) {
3948//        node.addChildNode(section.getText().getDiv());
3949//      }
3950//
3951//      if (!section.getSection().isEmpty()) {
3952//        renderSections(feed, node.addTag("blockquote"), section.getSection(), level+1);
3953//      }
3954    }
3955  }
3956
3957
3958  public class ObservationNode {
3959    private String ref;
3960    private ResourceWrapper obs;
3961    private List<ObservationNode> contained = new ArrayList<NarrativeGenerator.ObservationNode>();
3962  }
3963
3964  public XhtmlNode generateDiagnosticReport(ResourceWrapper dr) {
3965    XhtmlNode root = new XhtmlNode(NodeType.Element, "div");
3966    XhtmlNode h2 = root.h2();
3967    displayCodeableConcept(h2, getProperty(dr, "code").value());
3968    h2.tx(" ");
3969    PropertyWrapper pw = getProperty(dr, "category");
3970    if (valued(pw)) {
3971      h2.tx("(");
3972      displayCodeableConcept(h2, pw.value());
3973      h2.tx(") ");
3974    }
3975    displayDate(h2, getProperty(dr, "issued").value());
3976
3977    XhtmlNode tbl = root.table( "grid");
3978    XhtmlNode tr = tbl.tr();
3979    XhtmlNode tdl = tr.td();
3980    XhtmlNode tdr = tr.td();
3981    populateSubjectSummary(tdl, getProperty(dr, "subject").value());
3982    tdr.b().tx("Report Details");
3983    tdr.br();
3984    pw = getProperty(dr, "perfomer");
3985    if (valued(pw)) {
3986      tdr.addText(pluralise("Performer", pw.getValues().size())+":");
3987      for (BaseWrapper v : pw.getValues()) {
3988        tdr.tx(" ");
3989        displayReference(tdr, v);
3990      }
3991      tdr.br();
3992    }
3993    pw = getProperty(dr, "identifier");
3994    if (valued(pw)) {
3995      tdr.addText(pluralise("Identifier", pw.getValues().size())+":");
3996      for (BaseWrapper v : pw.getValues()) {
3997        tdr.tx(" ");
3998        displayIdentifier(tdr, v);
3999      }
4000      tdr.br();
4001    }
4002    pw = getProperty(dr, "request");
4003    if (valued(pw)) {
4004      tdr.addText(pluralise("Request", pw.getValues().size())+":");
4005      for (BaseWrapper v : pw.getValues()) {
4006        tdr.tx(" ");
4007        displayReferenceId(tdr, v);
4008      }
4009      tdr.br();
4010    }
4011
4012    pw = getProperty(dr, "result");
4013    if (valued(pw)) {
4014      List<ObservationNode> observations = fetchObservations(pw.getValues());
4015      buildObservationsTable(root, observations);
4016    }
4017
4018    pw = getProperty(dr, "conclusion");
4019    if (valued(pw))
4020      displayText(root.para(), pw.value());
4021
4022    pw = getProperty(dr, "result");
4023    if (valued(pw)) {
4024      XhtmlNode p = root.para();
4025      p.b().tx("Coded Diagnoses :");
4026      for (BaseWrapper v : pw.getValues()) {
4027        tdr.tx(" ");
4028        displayCodeableConcept(tdr, v);
4029      }
4030    }
4031    return root;
4032  }
4033
4034  private void buildObservationsTable(XhtmlNode root, List<ObservationNode> observations) {
4035    XhtmlNode tbl = root.table( "none");
4036    for (ObservationNode o : observations) {
4037      addObservationToTable(tbl, o, 0);
4038    }
4039  }
4040
4041  private void addObservationToTable(XhtmlNode tbl, ObservationNode o, int i) {
4042    XhtmlNode tr = tbl.tr();
4043    if (o.obs == null) {
4044      XhtmlNode td = tr.td().colspan("6");
4045      td.i().tx("This Observation could not be resolved");
4046    } else {
4047      addObservationToTable(tr, o.obs, i);
4048      // todo: contained observations
4049    }
4050    for (ObservationNode c : o.contained) {
4051      addObservationToTable(tbl, c, i+1);
4052    }
4053  }
4054
4055  private void addObservationToTable(XhtmlNode tr, ResourceWrapper obs, int i) {
4056    // TODO Auto-generated method stub
4057
4058    // code (+bodysite)
4059    XhtmlNode td = tr.td();
4060    PropertyWrapper pw = getProperty(obs, "result");
4061    if (valued(pw)) {
4062      displayCodeableConcept(td, pw.value());
4063    }
4064    pw = getProperty(obs, "bodySite");
4065    if (valued(pw)) {
4066      td.tx(" (");
4067      displayCodeableConcept(td, pw.value());
4068      td.tx(")");
4069    }
4070
4071    // value / dataAbsentReason (in red)
4072    td = tr.td();
4073    pw = getProperty(obs, "value[x]");
4074    if (valued(pw)) {
4075      if (pw.getTypeCode().equals("CodeableConcept"))
4076        displayCodeableConcept(td, pw.value());
4077      else if (pw.getTypeCode().equals("string"))
4078        displayText(td, pw.value());
4079      else
4080        td.addText(pw.getTypeCode()+" not rendered yet");
4081    }
4082
4083    // units
4084    td = tr.td();
4085    td.tx("to do");
4086
4087    // reference range
4088    td = tr.td();
4089    td.tx("to do");
4090
4091    // flags (status other than F, interpretation, )
4092    td = tr.td();
4093    td.tx("to do");
4094
4095    // issued if different to DR
4096    td = tr.td();
4097    td.tx("to do");
4098  }
4099
4100  private boolean valued(PropertyWrapper pw) {
4101    return pw != null && pw.hasValues();
4102  }
4103
4104  private void displayText(XhtmlNode c, BaseWrapper v) {
4105    c.addText(v.toString());
4106  }
4107
4108  private String pluralise(String name, int size) {
4109    return size == 1 ? name : name+"s";
4110  }
4111
4112  private void displayIdentifier(XhtmlNode c, BaseWrapper v) {
4113    String hint = "";
4114    PropertyWrapper pw = v.getChildByName("type");
4115    if (valued(pw)) {
4116      hint = genCC(pw.value());
4117    } else {
4118      pw = v.getChildByName("system");
4119      if (valued(pw)) {
4120        hint = pw.value().toString();
4121      }
4122    }
4123    displayText(c.span(null, hint), v.getChildByName("value").value());
4124  }
4125
4126  private String genCoding(BaseWrapper value) {
4127    PropertyWrapper pw = value.getChildByName("display");
4128    if (valued(pw))
4129      return pw.value().toString();
4130    pw = value.getChildByName("code");
4131    if (valued(pw))
4132      return pw.value().toString();
4133    return "";
4134  }
4135
4136  private String genCC(BaseWrapper value) {
4137    PropertyWrapper pw = value.getChildByName("text");
4138    if (valued(pw))
4139      return pw.value().toString();
4140    pw = value.getChildByName("coding");
4141    if (valued(pw))
4142      return genCoding(pw.getValues().get(0));
4143    return "";
4144  }
4145
4146  private void displayReference(XhtmlNode c, BaseWrapper v) {
4147    c.tx("to do");
4148  }
4149
4150
4151  private void displayDate(XhtmlNode c, BaseWrapper baseWrapper) {
4152    c.tx("to do");
4153  }
4154
4155  private void displayCodeableConcept(XhtmlNode c, BaseWrapper property) {
4156    c.tx("to do");
4157  }
4158
4159  private void displayReferenceId(XhtmlNode c, BaseWrapper v) {
4160    c.tx("to do");
4161  }
4162
4163  private PropertyWrapper getProperty(ResourceWrapper res, String name) {
4164    for (PropertyWrapper t : res.children()) {
4165      if (t.getName().equals(name))
4166        return t;
4167    }
4168    return null;
4169  }
4170
4171  private void populateSubjectSummary(XhtmlNode container, BaseWrapper subject) {
4172    ResourceWrapper r = fetchResource(subject);
4173    if (r == null)
4174      container.tx("Unable to get Patient Details");
4175    else if (r.getName().equals("Patient"))
4176      generatePatientSummary(container, r);
4177    else
4178      container.tx("Not done yet");
4179  }
4180
4181  private void generatePatientSummary(XhtmlNode c, ResourceWrapper r) {
4182    c.tx("to do");
4183  }
4184
4185  private ResourceWrapper fetchResource(BaseWrapper subject) {
4186    if (resolver == null)
4187      return null;
4188    String url = subject.getChildByName("reference").value().toString();
4189    ResourceWithReference rr = resolver.resolve(url);
4190    return rr == null ? null : rr.resource;
4191  }
4192
4193  private List<ObservationNode> fetchObservations(List<BaseWrapper> list) {
4194    return new ArrayList<NarrativeGenerator.ObservationNode>();
4195  }
4196
4197  public XhtmlNode renderBundle(Bundle b) throws FHIRException {
4198    if (b.getType() == BundleType.DOCUMENT) {
4199      if (!b.hasEntry() || !(b.getEntryFirstRep().hasResource() && b.getEntryFirstRep().getResource() instanceof Composition))
4200        throw new FHIRException("Invalid document - first entry is not a Composition");
4201      Composition dr = (Composition) b.getEntryFirstRep().getResource();
4202      return dr.getText().getDiv();
4203    } else  {
4204      XhtmlNode root = new XhtmlNode(NodeType.Element, "div");
4205      root.para().addText("Bundle "+b.getId()+" of type "+b.getType().toCode());
4206      int i = 0;
4207      for (BundleEntryComponent be : b.getEntry()) {
4208        root.hr();
4209        root.para().addText("Entry "+Integer.toString(i)+(be.hasFullUrl() ? "Full URL = " + be.getFullUrl() : ""));
4210        if (be.hasRequest())
4211          renderRequest(root, be.getRequest());
4212        if (be.hasSearch())
4213          renderSearch(root, be.getSearch());
4214        if (be.hasResponse())
4215          renderResponse(root, be.getResponse());
4216        if (be.hasResource()) {
4217          root.para().addText("Resource "+be.getResource().fhirType()+":");
4218          if (be.hasResource() && be.getResource() instanceof DomainResource) {
4219            DomainResource dr = (DomainResource) be.getResource();
4220            if ( dr.getText().hasDiv())
4221              root.blockquote().addChildNodes(dr.getText().getDiv().getChildNodes());
4222          }
4223        }
4224      }
4225      return root;
4226    }
4227  }
4228
4229  private void renderSearch(XhtmlNode root, BundleEntrySearchComponent search) {
4230    StringBuilder b = new StringBuilder();
4231    b.append("Search: ");
4232    if (search.hasMode())
4233      b.append("mode = "+search.getMode().toCode());
4234    if (search.hasScore()) {
4235      if (search.hasMode())
4236        b.append(",");
4237      b.append("score = "+search.getScore());
4238    }
4239    root.para().addText(b.toString());    
4240  }
4241
4242  private void renderResponse(XhtmlNode root, BundleEntryResponseComponent response) {
4243    root.para().addText("Request:");
4244    StringBuilder b = new StringBuilder();
4245    b.append(response.getStatus()+"\r\n");
4246    if (response.hasLocation())
4247      b.append("Location: "+response.getLocation()+"\r\n");
4248    if (response.hasEtag())
4249      b.append("E-Tag: "+response.getEtag()+"\r\n");
4250    if (response.hasLastModified())
4251      b.append("LastModified: "+response.getEtag()+"\r\n");
4252    root.pre().addText(b.toString());    
4253  }
4254
4255  private void renderRequest(XhtmlNode root, BundleEntryRequestComponent request) {
4256    root.para().addText("Response:");
4257    StringBuilder b = new StringBuilder();
4258    b.append(request.getMethod()+" "+request.getUrl()+"\r\n");
4259    if (request.hasIfNoneMatch())
4260      b.append("If-None-Match: "+request.getIfNoneMatch()+"\r\n");
4261    if (request.hasIfModifiedSince())
4262      b.append("If-Modified-Since: "+request.getIfModifiedSince()+"\r\n");
4263    if (request.hasIfMatch())
4264      b.append("If-Match: "+request.getIfMatch()+"\r\n");
4265    if (request.hasIfNoneExist())
4266      b.append("If-None-Exist: "+request.getIfNoneExist()+"\r\n");
4267    root.pre().addText(b.toString());    
4268  }
4269
4270  public XhtmlNode renderBundle(org.hl7.fhir.dstu3.elementmodel.Element element) throws FHIRException {
4271    XhtmlNode root = new XhtmlNode(NodeType.Element, "div");
4272    for (Base b : element.listChildrenByName("entry")) {
4273      XhtmlNode c = getHtmlForResource(((org.hl7.fhir.dstu3.elementmodel.Element) b).getNamedChild("resource"));
4274      if (c != null)
4275        root.addChildNodes(c.getChildNodes());
4276      root.hr();
4277    }
4278    return root;
4279  }
4280
4281  private XhtmlNode getHtmlForResource(org.hl7.fhir.dstu3.elementmodel.Element element) {
4282    org.hl7.fhir.dstu3.elementmodel.Element text = element.getNamedChild("text");
4283    if (text == null)
4284      return null;
4285    org.hl7.fhir.dstu3.elementmodel.Element div = text.getNamedChild("div");
4286    if (div == null)
4287      return null;
4288    else
4289      return div.getXhtml();
4290  }
4291
4292  public String getDefinitionsTarget() {
4293    return definitionsTarget;
4294  }
4295
4296  public void setDefinitionsTarget(String definitionsTarget) {
4297    this.definitionsTarget = definitionsTarget;
4298  }
4299
4300  public String getCorePath() {
4301    return corePath;
4302  }
4303
4304  public void setCorePath(String corePath) {
4305    this.corePath = corePath;
4306  }
4307
4308  public String getDestDir() {
4309    return destDir;
4310  }
4311
4312  public void setDestDir(String destDir) {
4313    this.destDir = destDir;
4314  }
4315
4316  public ProfileKnowledgeProvider getPkp() {
4317    return pkp;
4318  }
4319
4320  public void setPkp(ProfileKnowledgeProvider pkp) {
4321    this.pkp = pkp;
4322  }
4323
4324  
4325}