/* * Copyright 2010 Red Hat, Inc. and/or its affiliates. * * 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.optaplanner.examples.travelingtournament.solver.move; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.Iterator; import java.util.List; import org.apache.commons.lang3.builder.EqualsBuilder; import org.apache.commons.lang3.builder.HashCodeBuilder; import org.optaplanner.core.impl.heuristic.move.AbstractMove; import org.optaplanner.core.impl.score.director.ScoreDirector; import org.optaplanner.examples.travelingtournament.domain.Day; import org.optaplanner.examples.travelingtournament.domain.Match; import org.optaplanner.examples.travelingtournament.domain.TravelingTournament; public class MatchChainRotationsMove extends AbstractMove<TravelingTournament> { private List<Match> firstMatchList; private List<Match> secondMatchList; public MatchChainRotationsMove(List<Match> firstMatchList, List<Match> secondMatchList) { this.firstMatchList = firstMatchList; this.secondMatchList = secondMatchList; } @Override public boolean isMoveDoable(ScoreDirector<TravelingTournament> scoreDirector) { return true; } @Override public MatchChainRotationsMove createUndoMove(ScoreDirector<TravelingTournament> scoreDirector) { List<Match> inverseFirstMatchList = new ArrayList<>(firstMatchList); Collections.reverse(inverseFirstMatchList); List<Match> inverseSecondMatchList = new ArrayList<>(secondMatchList); Collections.reverse(inverseSecondMatchList); return new MatchChainRotationsMove(inverseFirstMatchList, inverseSecondMatchList); } @Override protected void doMoveOnGenuineVariables(ScoreDirector<TravelingTournament> scoreDirector) { rotateList(scoreDirector, firstMatchList); if (!secondMatchList.isEmpty()) { // TODO create SingleMatchListRotateMove rotateList(scoreDirector, secondMatchList); } } private void rotateList(ScoreDirector<TravelingTournament> scoreDirector, List<Match> matchList) { Iterator<Match> it = matchList.iterator(); Match previousMatch = it.next(); Match match = null; Day firstDay = previousMatch.getDay(); while (it.hasNext()) { match = it.next(); TravelingTournamentMoveHelper.moveDay(scoreDirector, previousMatch, match.getDay()); previousMatch = match; } TravelingTournamentMoveHelper.moveDay(scoreDirector, match, firstDay); } @Override public Collection<? extends Object> getPlanningEntities() { List<Match> entities = new ArrayList<>(firstMatchList.size() + secondMatchList.size()); entities.addAll(firstMatchList); entities.addAll(secondMatchList); return entities; } @Override public Collection<? extends Object> getPlanningValues() { List<Day> values = new ArrayList<>(firstMatchList.size() + secondMatchList.size()); for (Match match : firstMatchList) { values.add(match.getDay()); } for (Match match : secondMatchList) { values.add(match.getDay()); } return values; } @Override public boolean equals(Object o) { if (this == o) { return true; } else if (o instanceof MatchChainRotationsMove) { MatchChainRotationsMove other = (MatchChainRotationsMove) o; return new EqualsBuilder() .append(firstMatchList, other.firstMatchList) .append(secondMatchList, other.secondMatchList) .isEquals(); } else { return false; } } @Override public int hashCode() { return new HashCodeBuilder() .append(firstMatchList) .append(secondMatchList) .toHashCode(); } @Override public String toString() { return "Rotation " + firstMatchList + " & Rotation " + secondMatchList; } }