001/*
002 * #%L
003 * HAPI FHIR - Core Library
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 *      http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.model.api;
021
022import ca.uhn.fhir.model.api.annotation.Child;
023import ca.uhn.fhir.model.api.annotation.Description;
024import org.apache.commons.lang3.Validate;
025import org.hl7.fhir.instance.model.api.IBaseDatatype;
026
027import java.util.ArrayList;
028import java.util.Collections;
029import java.util.HashMap;
030import java.util.List;
031import java.util.Map;
032
033public abstract class BaseElement implements /*IElement, */ ISupportsUndeclaredExtensions {
034
035        private static final long serialVersionUID = -3092659584634499332L;
036        private List<String> myFormatCommentsPost;
037        private List<String> myFormatCommentsPre;
038        private Map<String, Object> userData;
039
040        @Child(
041                        name = "extension",
042                        type = {ExtensionDt.class},
043                        order = 0,
044                        min = 0,
045                        max = Child.MAX_UNLIMITED,
046                        modifier = false,
047                        summary = false)
048        @Description(
049                        shortDefinition = "Additional Content defined by implementations",
050                        formalDefinition =
051                                        "May be used to represent additional information that is not part of the basic definition of the resource. In order to make the use of extensions safe and manageable, there is a strict set of governance  applied to the definition and use of extensions. Though any implementer is allowed to define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension.")
052        private List<ExtensionDt> myUndeclaredExtensions;
053
054        /**
055         * May be used to represent additional information that is not part of the basic definition of the resource, and that modifies the understanding of the element that contains it. Usually modifier elements provide negation or qualification. In order to make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer is allowed to define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension. Applications processing a resource are required to check for modifier extensions.
056         */
057        @Child(
058                        name = "modifierExtension",
059                        type = {ExtensionDt.class},
060                        order = 1,
061                        min = 0,
062                        max = Child.MAX_UNLIMITED,
063                        modifier = true,
064                        summary = false)
065        @Description(
066                        shortDefinition = "Extensions that cannot be ignored",
067                        formalDefinition =
068                                        "May be used to represent additional information that is not part of the basic definition of the resource, and that modifies the understanding of the element that contains it. Usually modifier elements provide negation or qualification. In order to make the use of extensions safe and manageable, there is a strict set of governance applied to the definition and use of extensions. Though any implementer is allowed to define an extension, there is a set of requirements that SHALL be met as part of the definition of the extension. Applications processing a resource are required to check for modifier extensions.")
069        private List<ExtensionDt> myUndeclaredModifierExtensions;
070
071        @Override
072        public ExtensionDt addUndeclaredExtension(boolean theIsModifier, String theUrl) {
073                Validate.notEmpty(theUrl, "URL must be populated");
074
075                ExtensionDt retVal = new ExtensionDt(theIsModifier, theUrl);
076                if (theIsModifier) {
077                        getUndeclaredModifierExtensions();
078                        myUndeclaredModifierExtensions.add(retVal);
079                } else {
080                        getUndeclaredExtensions();
081                        myUndeclaredExtensions.add(retVal);
082                }
083                return retVal;
084        }
085
086        @Override
087        public ExtensionDt addUndeclaredExtension(boolean theIsModifier, String theUrl, IBaseDatatype theValue) {
088                Validate.notEmpty(theUrl, "URL must be populated");
089                Validate.notNull(theValue, "Value must not be null");
090                ExtensionDt retVal = new ExtensionDt(theIsModifier, theUrl, theValue);
091                if (theIsModifier) {
092                        getUndeclaredModifierExtensions();
093                        myUndeclaredModifierExtensions.add(retVal);
094                } else {
095                        getUndeclaredExtensions();
096                        myUndeclaredExtensions.add(retVal);
097                }
098                return retVal;
099        }
100
101        @Override
102        public void addUndeclaredExtension(ExtensionDt theExtension) {
103                Validate.notNull(theExtension, "Extension can not be null");
104                if (theExtension.isModifier()) {
105                        getUndeclaredModifierExtensions();
106                        myUndeclaredModifierExtensions.add(theExtension);
107                } else {
108                        getUndeclaredExtensions();
109                        myUndeclaredExtensions.add(theExtension);
110                }
111        }
112
113        @Override
114        public List<ExtensionDt> getAllUndeclaredExtensions() {
115                ArrayList<ExtensionDt> retVal = new ArrayList<ExtensionDt>();
116                if (myUndeclaredExtensions != null) {
117                        retVal.addAll(myUndeclaredExtensions);
118                }
119                if (myUndeclaredModifierExtensions != null) {
120                        retVal.addAll(myUndeclaredModifierExtensions);
121                }
122                return Collections.unmodifiableList(retVal);
123        }
124
125        @Override
126        public List<String> getFormatCommentsPost() {
127                if (myFormatCommentsPost == null) myFormatCommentsPost = new ArrayList<String>();
128                return myFormatCommentsPost;
129        }
130
131        @Override
132        public List<String> getFormatCommentsPre() {
133                if (myFormatCommentsPre == null) myFormatCommentsPre = new ArrayList<String>();
134                return myFormatCommentsPre;
135        }
136
137        @Override
138        public List<ExtensionDt> getUndeclaredExtensions() {
139                if (myUndeclaredExtensions == null) {
140                        myUndeclaredExtensions = new ArrayList<ExtensionDt>();
141                }
142                return (myUndeclaredExtensions);
143        }
144
145        @Override
146        public List<ExtensionDt> getUndeclaredExtensionsByUrl(String theUrl) {
147                org.apache.commons.lang3.Validate.notNull(theUrl, "URL can not be null");
148                ArrayList<ExtensionDt> retVal = new ArrayList<ExtensionDt>();
149                for (ExtensionDt next : getAllUndeclaredExtensions()) {
150                        if (theUrl.equals(next.getUrlAsString())) {
151                                retVal.add(next);
152                        }
153                }
154                return Collections.unmodifiableList(retVal);
155        }
156
157        @Override
158        public List<ExtensionDt> getUndeclaredModifierExtensions() {
159                if (myUndeclaredModifierExtensions == null) {
160                        myUndeclaredModifierExtensions = new ArrayList<ExtensionDt>();
161                }
162                return (myUndeclaredModifierExtensions);
163        }
164
165        @Override
166        public boolean hasFormatComment() {
167                return (myFormatCommentsPre != null && !myFormatCommentsPre.isEmpty())
168                                || (myFormatCommentsPost != null && !myFormatCommentsPost.isEmpty());
169        }
170
171        @Override
172        public Object getUserData(String name) {
173                if (userData == null) return null;
174                return userData.get(name);
175        }
176
177        @Override
178        public void setUserData(String name, Object value) {
179                if (userData == null) {
180                        userData = new HashMap<>();
181                }
182                userData.put(name, value);
183        }
184
185        /**
186         * Intended to be called by extending classes {@link #isEmpty()} implementations, returns <code>true</code> if all
187         * content in this superclass instance is empty per the semantics of {@link #isEmpty()}.
188         */
189        protected boolean isBaseEmpty() {
190                if (myUndeclaredExtensions != null) {
191                        for (ExtensionDt next : myUndeclaredExtensions) {
192                                if (next == null) {
193                                        continue;
194                                }
195                                if (!next.isEmpty()) {
196                                        return false;
197                                }
198                        }
199                }
200                if (myUndeclaredModifierExtensions != null) {
201                        for (ExtensionDt next : myUndeclaredModifierExtensions) {
202                                if (next == null) {
203                                        continue;
204                                }
205                                if (!next.isEmpty()) {
206                                        return false;
207                                }
208                        }
209                }
210                return true;
211        }
212}