/* * Copyright (c) 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017 David Berkman * * This file is part of the SmallMind Code Project. * * The SmallMind Code Project is free software, you can redistribute * it and/or modify it under either, at your discretion... * * 1) The terms of GNU Affero General Public License as published by the * Free Software Foundation, either version 3 of the License, or (at * your option) any later version. * * ...or... * * 2) The terms of the Apache License, Version 2.0. * * The SmallMind Code Project is distributed in the hope that it will * be useful, but WITHOUT ANY WARRANTY; without even the implied warranty * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License or Apache License for more details. * * You should have received a copy of the GNU Affero General Public License * and the Apache License along with the SmallMind Code Project. If not, see * <http://www.gnu.org/licenses/> or <http://www.apache.org/licenses/LICENSE-2.0>. * * Additional permission under the GNU Affero GPL version 3 section 7 * ------------------------------------------------------------------ * If you modify this Program, or any covered work, by linking or * combining it with other code, such other code is not for that reason * alone subject to any of the requirements of the GNU Affero GPL * version 3. */ package org.smallmind.nutsnbolts.reflection; import java.lang.reflect.Field; import java.util.List; import org.smallmind.nutsnbolts.lang.TypeMismatchException; import org.smallmind.nutsnbolts.reflection.type.GenericUtility; import org.smallmind.nutsnbolts.reflection.type.UnexpectedGenericDeclaration; public abstract class Overlay<O extends Overlay<O>> implements Differentiable<O> { private final transient Class<O> overlayClass; public Overlay () { List<Class<?>> typeArguments = GenericUtility.getTypeArguments(Overlay.class, this.getClass()); if (typeArguments.size() != 1) { throw new UnexpectedGenericDeclaration("Expecting a single generic type"); } else if (!Overlay.class.isAssignableFrom(overlayClass = (Class<O>)typeArguments.get(0))) { throw new UnexpectedGenericDeclaration("Expecting a single generic type extending %s", Overlay.class.getSimpleName()); } } public Class<O> getOverlayClass () { return overlayClass; } public O overlay (Object... overlays) throws IllegalAccessException { return overlay(overlays, null); } public O overlay (Object[] overlays, Field[] exclusions) throws IllegalAccessException { if ((overlays != null) && (overlays.length > 0)) { for (Object overlay : overlays) { if (overlay != null) { if (!overlay.getClass().isAssignableFrom(this.getClass())) { throw new TypeMismatchException("Overlays must be assignable from type(%s)", this.getClass()); } else { boolean excluded; for (Field field : FieldUtility.getFields(this.getClass())) { excluded = false; if ((exclusions != null) && (exclusions.length > 0)) { for (Field exclusion : exclusions) { if (!exclusion.getDeclaringClass().isAssignableFrom(this.getClass())) { throw new TypeMismatchException("The type(%s) does not contain the excluded field(%s)", this.getClass().getName(), exclusion.getName()); } else if (exclusion.equals(field)) { excluded = true; break; } } } if (!excluded) { Object value; if ((value = field.get(overlay)) != null) { field.set(this, value); } } } } } } } return overlayClass.cast(this); } }