001    /**
002     * Copyright 2005-2013 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.kuali.rice.location.api.state;
017    
018    import org.apache.commons.lang.StringUtils;
019    import org.kuali.rice.core.api.CoreConstants;
020    import org.kuali.rice.core.api.mo.AbstractDataTransferObject;
021    import org.kuali.rice.core.api.mo.ModelBuilder;
022    import org.kuali.rice.location.api.LocationConstants;
023    import org.w3c.dom.Element;
024    
025    import javax.xml.bind.annotation.XmlAccessType;
026    import javax.xml.bind.annotation.XmlAccessorType;
027    import javax.xml.bind.annotation.XmlAnyElement;
028    import javax.xml.bind.annotation.XmlElement;
029    import javax.xml.bind.annotation.XmlRootElement;
030    import javax.xml.bind.annotation.XmlType;
031    import java.io.Serializable;
032    import java.util.Collection;
033    
034    /**
035     * POJO implementation of StateContract that is immutable. Instances of State can be (un)marshalled to and from XML.
036     *
037     * @see StateContract
038     * @author Kuali Rice Team (rice.collab@kuali.org)
039     */
040    @XmlRootElement(name = State.Constants.ROOT_ELEMENT_NAME)
041    @XmlAccessorType(XmlAccessType.NONE)
042    @XmlType(name = State.Constants.TYPE_NAME, propOrder = {
043            State.Elements.CODE,
044            State.Elements.NAME,
045            State.Elements.COUNTRY_CODE,
046            State.Elements.ACTIVE,
047            CoreConstants.CommonElements.VERSION_NUMBER,
048            CoreConstants.CommonElements.FUTURE_ELEMENTS
049    })
050    public final class State extends AbstractDataTransferObject implements StateContract {
051    
052        private static final long serialVersionUID = 6097498602725305353L;
053    
054        @XmlElement(name = Elements.CODE, required = true)
055        private final String code;
056    
057        @XmlElement(name = Elements.NAME, required = true)
058        private final String name;
059    
060        @XmlElement(name = Elements.COUNTRY_CODE, required = true)
061        private final String countryCode;
062    
063        @XmlElement(name = Elements.ACTIVE, required = true)
064        private final boolean active;
065    
066        @XmlElement(name = CoreConstants.CommonElements.VERSION_NUMBER, required = false)
067        private final Long versionNumber;
068    
069        @SuppressWarnings("unused")
070        @XmlAnyElement
071        private final Collection<Element> _futureElements = null;
072    
073        /**
074         * This constructor should never be called except during JAXB unmarshalling.
075         */
076        @SuppressWarnings("unused")
077        private State() {
078            this.code = null;
079            this.name = null;
080            this.countryCode = null;
081            this.active = false;
082            this.versionNumber = null;
083        }
084    
085        private State(Builder builder) {
086            code = builder.getCode();
087            name = builder.getName();
088            countryCode = builder.getCountryCode();
089            active = builder.isActive();
090            versionNumber = builder.getVersionNumber();
091        }
092    
093        /** {@inheritDoc} */
094        @Override
095        public String getCode() {
096            return code;
097        }
098    
099        /** {@inheritDoc} */
100        @Override
101        public String getName() {
102            return name;
103        }
104    
105        /** {@inheritDoc} */
106        @Override
107        public String getCountryCode() {
108            return countryCode;
109        }
110    
111        /** {@inheritDoc} */
112        @Override
113        public boolean isActive() {
114            return active;
115        }
116    
117        /** {@inheritDoc} */
118        @Override
119        public Long getVersionNumber() {
120            return versionNumber;
121        }
122    
123        /**
124         * This builder constructs a State enforcing the constraints of the {@link StateContract}.
125         */
126        public static class Builder implements StateContract, ModelBuilder, Serializable {
127    
128            private static final long serialVersionUID = 7077484401017765844L;
129    
130            private String code;
131    
132            private String name;
133    
134            private String countryCode;
135    
136            private boolean active;
137    
138            private Long versionNumber;
139    
140            private Builder(String code, String name, String countryCode) {
141                setCode(code);
142                setName(name);
143                setCountryCode(countryCode);
144            }
145    
146            /**
147             * creates a State with the required fields.
148             * @param code represents code for the State being built
149             * @param name represents full name for the State being built
150             * @param countryCode code for the Country this State is associated with
151             * @return a bootstrapped Builder defaulted with the passed in code, name, and countryCode.
152             */
153            public static Builder create(String code, String name, String countryCode) {
154                final Builder builder = new Builder(code, name, countryCode);
155                builder.setActive(true);
156                return builder;
157            }
158    
159            /**
160             * creates a Parameter from an existing {@link StateContract}.
161             */
162            public static Builder create(StateContract contract) {
163                final Builder builder = new Builder(contract.getCode(), contract.getName(), contract.getCountryCode());
164                builder.setActive(contract.isActive());
165                builder.setVersionNumber(contract.getVersionNumber());
166                return builder;
167            }
168    
169            @Override
170            public String getCode() {
171                return code;
172            }
173    
174            /**
175             * Sets the code to be used for the State created from this Builder.
176             * @param code String code for a State.
177             * @throws IllegalArgumentException if the passed in code is null or a blank String.
178             */
179            public void setCode(String code) {
180                if (StringUtils.isBlank(code)) {
181                    throw new IllegalArgumentException("code is blank");
182                }
183    
184                this.code = code;
185            }
186    
187            @Override
188            public String getName() {
189                return name;
190            }
191    
192            /**
193             * Sets the full name of the State created from this Builder.
194             * @param name String representing the full name for the State
195             * @throws IllegalArgumentException if the passed in name is null or a blank String.
196             */
197            public void setName(String name) {
198                if (StringUtils.isBlank(name)) {
199                    throw new IllegalArgumentException("name is blank");
200                }
201    
202                this.name = name;
203            }
204    
205            @Override
206            public String getCountryCode() {
207                return countryCode;
208            }
209    
210            /**
211             * Sets the Country code to be associated with the State created from this Builder.
212             * @param countryCode String representing the Country Code
213             * @throws IllegalArgumentException if the passed in countryCode is null or a blank String.
214             */
215            public void setCountryCode(String countryCode) {
216                if (StringUtils.isBlank(countryCode)) {
217                    throw new IllegalArgumentException("countryCode is blank");
218                }
219    
220                this.countryCode = countryCode;
221            }
222    
223            @Override
224            public boolean isActive() {
225                return active;
226            }
227    
228            public void setActive(boolean active) {
229                this.active = active;
230            }
231    
232            @Override
233            public Long getVersionNumber() {
234                return versionNumber;
235            }
236    
237            public void setVersionNumber(Long versionNumber) {
238                this.versionNumber = versionNumber;
239            }
240    
241            @Override
242            public State build() {
243                return new State(this);
244            }
245        }
246    
247        /**
248         * Defines some internal constants used on this class.
249         */
250        static class Constants {
251            final static String ROOT_ELEMENT_NAME = "state";
252            final static String TYPE_NAME = "StateType";
253        }
254    
255        /**
256         * A private class which exposes constants which define the XML element names to use
257         * when this object is marshalled to XML.
258         */
259        static class Elements {
260            final static String CODE = "code";
261            final static String NAME = "name";
262            final static String COUNTRY_CODE = "countryCode";
263            final static String ACTIVE = "active";
264        }
265    
266        public static class Cache {
267            public static final String NAME = LocationConstants.Namespaces.LOCATION_NAMESPACE_2_0 + "/" + State.Constants.TYPE_NAME;
268        }
269    }