package auction.persistence; import org.hibernate.*; import org.hibernate.usertype.CompositeUserType; import org.hibernate.engine.SessionImplementor; import org.hibernate.type.Type; import auction.model.MonetaryAmount; import java.io.Serializable; import java.math.BigDecimal; import java.sql.*; import java.util.Currency; /** * This is a simple Hibernate custom mapping type for MonetaryAmount value types. *

* Basically the same as the simple MonetaryAmountSimpleUserType, but * implementing the Hibernate CompositeUserType interface. This interface * has some additional methods that allow Hibernate to analyze the value type you * are mapping. This is mostly useful for HQL queries: with this custom mapping * type, you can use the "amount" and "currency" sub-components in HQL queries. * * @see MonetaryAmountSimpleUserType * @author Christian Bauer */ public class MonetaryAmountCompositeUserType implements CompositeUserType { public Class returnedClass() { return MonetaryAmount.class; } public boolean isMutable() { return false; } public Object deepCopy(Object value) { return value; } public Serializable disassemble(Object value, SessionImplementor session) { return (Serializable) value; } public Object assemble(Serializable cached, SessionImplementor session, Object owner) { return cached; } public Object replace(Object original, Object target, SessionImplementor session, Object owner) { return original; } public boolean equals(Object x, Object y) { if (x == y) return true; if (x == null || y == null) return false; return x.equals(y); } public int hashCode(Object x) { return x.hashCode(); } public Object nullSafeGet(ResultSet resultSet, String[] names, SessionImplementor session, Object owner) throws SQLException { BigDecimal value = resultSet.getBigDecimal( names[0] ); if (resultSet.wasNull()) return null; Currency currency = Currency.getInstance(resultSet.getString( names[1] ) ); return new MonetaryAmount(value, currency); } public void nullSafeSet(PreparedStatement statement, Object value, int index, SessionImplementor session) throws SQLException { if (value==null) { statement.setNull(index, Hibernate.BIG_DECIMAL.sqlType()); statement.setNull(index+1, Hibernate.CURRENCY.sqlType()); } else { MonetaryAmount amount = (MonetaryAmount) value; String currencyCode = amount.getCurrency().getCurrencyCode(); statement.setBigDecimal( index, amount.getValue() ); statement.setString( index+1, currencyCode ); } } public String[] getPropertyNames() { return new String[] { "value", "currency" }; } public Type[] getPropertyTypes() { return new Type[] { Hibernate.BIG_DECIMAL, Hibernate.CURRENCY }; } public Object getPropertyValue(Object component, int property) { MonetaryAmount monetaryAmount = (MonetaryAmount) component; if (property == 0) return monetaryAmount.getValue(); else return monetaryAmount.getCurrency(); } public void setPropertyValue(Object component, int property, Object value) { throw new UnsupportedOperationException("MonetaryAmount is immutable"); } }