/* * Copyright 2004-2012 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.springframework.webflow.engine.model; import java.util.Collections; import java.util.LinkedList; import org.springframework.util.StringUtils; /** * Contains basic merge functions that can be utilized by other models. * * @author Scott Andrews */ public abstract class AbstractModel implements Model { /** * Merge two objects. If the child is null, the parent will be returned. Else the child will be returned. * @param child the child object to merge * @param parent the parent object to merge * @return the merged string */ protected Object merge(Object child, Object parent) { if (child == null) { return parent; } else { return child; } } /** * Merge two strings. If the child is null, the parent will be returned. Else the child will be returned. * @param child the child string to merge * @param parent the parent string to merge * @return the merged string */ protected String merge(String child, String parent) { return (String) merge((Object) child, (Object) parent); } /** * Merge two model elements. If the child is null, the parent will be returned. Else the parent element will be * merged into the child element with the result returned * @param child the child model element to merge * @param parent the parent model element to merge * @return the merged element model */ protected Model merge(Model child, Model parent) { if (child == null) { if (parent == null) { return null; } else { return parent.createCopy(); } } else if (parent == null) { return child; } else { child.merge(parent); return child; } } /** * Merge two lists. All child element will be in the merged list. All parent elements not in the child list will be * added. Mergeable elements in both lists will be merged according to that element merge rules. New items are added * to the end of the list * @param child the child list to merge * @param parent the parent list to merge * @return the merged list */ protected <T extends Model> LinkedList<T> merge(LinkedList<T> child, LinkedList<T> parent) { return merge(child, parent, true); } /** * Merge two lists. All child element will be in the merged list. All parent elements not in the child list will be * added. Mergeable elements in both lists will be merged according to that element merge rules. * @param child the child list to merge * @param parent the parent list to merge * @param addAtEnd if true new items will be added at the end of the list, otherwise the beginning * @return the merged list */ protected <T extends Model> LinkedList<T> merge(LinkedList<T> child, LinkedList<T> parent, boolean addAtEnd) { if (child == null) { return copyList(parent); } if (parent == null) { return child; } if (!addAtEnd) { parent = new LinkedList<T>(parent); Collections.reverse(parent); } for (T parentModel : parent) { addOrMerge(child, parentModel, addAtEnd); } return child; } private <T extends Model> void addOrMerge(LinkedList<T> list, T modelToMerge, boolean addAtEnd) { for (T model : list) { if (model.isMergeableWith(modelToMerge)) { model.merge(modelToMerge); return; } } @SuppressWarnings("unchecked") T copy = (T) modelToMerge.createCopy(); if (addAtEnd) { list.addLast(copy); } else { list.addFirst(copy); } } protected Model copy(Model model) { if (model == null) { return null; } return model.createCopy(); } @SuppressWarnings("unchecked") protected <T extends Model> LinkedList<T> copyList(LinkedList<T> list) { if (list == null) { return null; } LinkedList<T> copy = new LinkedList<T>(); for (T model : list) { copy.add((T) model.createCopy()); } return copy; } }