+/*
+ * NoseConeHandler.java
+ */
+package net.sf.openrocket.file.rocksim;
+
+import net.sf.openrocket.aerodynamics.WarningSet;
+import net.sf.openrocket.file.simplesax.ElementHandler;
+import net.sf.openrocket.file.simplesax.PlainTextHandler;
+import net.sf.openrocket.material.Material;
+import net.sf.openrocket.rocketcomponent.NoseCone;
+import net.sf.openrocket.rocketcomponent.RocketComponent;
+import net.sf.openrocket.rocketcomponent.Transition;
+import org.xml.sax.SAXException;
+
+import java.util.HashMap;
+
+/**
+ * The SAX nose cone handler for Rocksim NoseCones.
+ */
+class NoseConeHandler extends BaseHandler<NoseCone> {
+
+ /**
+ * The OpenRocket NoseCone.
+ */
+ private final NoseCone noseCone = new NoseCone();
+
+ /**
+ * The wall thickness. Used for hollow nose cones.
+ */
+ private double thickness = 0d;
+
+ /**
+ * Constructor.
+ *
+ * @param c the parent component to the nosecone
+ * @throws IllegalArgumentException thrown if <code>c</code> is null
+ */
+ public NoseConeHandler(RocketComponent c) throws IllegalArgumentException {
+ if (c == null) {
+ throw new IllegalArgumentException("The parent component of a nose cone may not be null.");
+ }
+ c.addChild(noseCone);
+ noseCone.setAftRadiusAutomatic(false);
+ }
+
+ @Override
+ public ElementHandler openElement(String element, HashMap<String, String> attributes, WarningSet warnings) {
+ //Nose cones in Rocksim may have attached parts - namely Mass Objects - as children.
+ if ("AttachedParts".equals(element)) {
+ return new AttachedPartsHandler(noseCone);
+ }
+ return PlainTextHandler.INSTANCE;
+ }
+
+ @Override
+ public void closeElement(String element, HashMap<String, String> attributes,
+ String content, WarningSet warnings) throws SAXException {
+ super.closeElement(element, attributes, content, warnings);
+
+ try {
+ if ("ShapeCode".equals(element)) {
+ noseCone.setType(RocksimNoseConeCode.fromCode(Integer.parseInt(content)).asOpenRocket());
+ }
+ if ("Len".equals(element)) {
+ noseCone.setLength(Math.max(0, Double.parseDouble(content) / RocksimHandler.ROCKSIM_TO_OPENROCKET_LENGTH));
+ }
+ if ("BaseDia".equals(element)) {
+ noseCone.setAftRadius(Math.max(0, Double.parseDouble(content) / RocksimHandler.ROCKSIM_TO_OPENROCKET_RADIUS));
+ }
+ if ("WallThickness".equals(element)) {
+ thickness = Math.max(0, Double.parseDouble(content) / RocksimHandler.ROCKSIM_TO_OPENROCKET_LENGTH);
+ }
+ if ("ShoulderOD".equals(element)) {
+ noseCone.setAftShoulderRadius(Math.max(0, Double.parseDouble(
+ content) / RocksimHandler.ROCKSIM_TO_OPENROCKET_RADIUS));
+ }
+ if ("ShoulderLen".equals(element)) {
+ noseCone.setAftShoulderLength(Math.max(0, Double.parseDouble(
+ content) / RocksimHandler.ROCKSIM_TO_OPENROCKET_LENGTH));
+ }
+ if ("ShapeParameter".equals(element)) {
+ //The Rocksim ShapeParameter only applies to certain shapes, although it is included
+ //in the design file for all nose cones. Applying it when it should not be causes oddities so
+ //a check is made for the allowable shapes.
+ if (Transition.Shape.POWER.equals(noseCone.getType()) ||
+ Transition.Shape.HAACK.equals(noseCone.getType()) ||
+ Transition.Shape.PARABOLIC.equals(noseCone.getType())) {
+ noseCone.setShapeParameter(Double.parseDouble(content));
+ }
+ }
+ if ("ConstructionType".equals(element)) {
+ int typeCode = Integer.parseInt(content);
+ if (typeCode == 0) {
+ //SOLID
+ noseCone.setFilled(true);
+ }
+ else if (typeCode == 1) {
+ //HOLLOW
+ noseCone.setFilled(false);
+ }
+ }
+ if ("FinishCode".equals(element)) {
+ noseCone.setFinish(RocksimFinishCode.fromCode(Integer.parseInt(content)).asOpenRocket());
+ }
+ if ("Material".equals(element)) {
+ setMaterialName(content);
+ }
+ }
+ catch (NumberFormatException nfe) {
+ warnings.add("Could not convert " + element + " value of " + content + ". It is expected to be a number.");
+ }
+ }
+
+ @Override
+ public void endHandler(String element, HashMap<String, String> attributes, String content, WarningSet warnings)
+ throws SAXException {
+ super.endHandler(element, attributes, content, warnings);
+
+ if (noseCone.isFilled()) {
+ noseCone.setAftShoulderThickness(noseCone.getAftShoulderRadius());
+ }
+ else {
+ noseCone.setThickness(thickness);
+ noseCone.setAftShoulderThickness(thickness);
+ }
+ }
+
+ /**
+ * Get the nose cone component this handler is working upon.
+ *
+ * @return a nose cone component
+ */
+ @Override
+ public NoseCone getComponent() {
+ return noseCone;
+ }
+
+ /**
+ * Get the required type of material for this component.
+ *
+ * @return BULK
+ */
+ public Material.Type getMaterialType() {
+ return Material.Type.BULK;
+ }
+
+}