All Downloads are FREE. Search and download functionalities are using the official Maven repository.

me.vertretungsplan.objects.diff.SubstitutionScheduleDayDiff Maven / Gradle / Ivy

/*
 * substitution-schedule-parser - Java library for parsing schools' substitution schedules
 * Copyright (c) 2016 Johan v. Forstner
 *
 * This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
 * If a copy of the MPL was not distributed with this file, You can obtain one at https://mozilla.org/MPL/2.0/.
 */

package me.vertretungsplan.objects.diff;

import me.vertretungsplan.objects.Substitution;
import me.vertretungsplan.objects.SubstitutionSchedule;
import me.vertretungsplan.objects.SubstitutionScheduleDay;
import org.joda.time.LocalDate;

import java.util.*;

/**
 * Represents the difference between two {@link SubstitutionScheduleDay}s.
 */
public class SubstitutionScheduleDayDiff implements Cloneable {
    private LocalDate date;
    private String dateString;
    private Set newSubstitutions;
    private Set editedSubstitutions;
    private Set removedSubstitutions;
    private List newMessages;
    private List removedMessages;

    public static SubstitutionScheduleDayDiff compare(SubstitutionScheduleDay a,
                                                      SubstitutionScheduleDay b) {
        if (!a.equalsByDate(b)) {
            throw new IllegalArgumentException("Days must have the same date");
        }
        SubstitutionScheduleDayDiff diff = new SubstitutionScheduleDayDiff();
        diff.date = a.getDate();
        diff.dateString = a.getDateString();
        diff.newMessages = new ArrayList<>();
        diff.removedMessages = new ArrayList<>();
        diff.newSubstitutions = new HashSet<>();
        diff.editedSubstitutions = new HashSet<>();
        diff.removedSubstitutions = new HashSet<>();

        for (String message : b.getMessages()) {
            if (!a.getMessages().contains(message)) {
                diff.newMessages.add(message);
            }
        }

        for (String message : a.getMessages()) {
            if (!b.getMessages().contains(message)) {
                diff.removedMessages.add(message);
            }
        }

        // save all old substitutions that were already handled here to speed up the second run
        Set handledOldSubstitutions = new HashSet<>();

        // first run: go through all new substitutions and search for matching old substitutions
        for (Substitution newSubstitution : b.getSubstitutions()) {
            if (a.getSubstitutions().contains(newSubstitution)) {
                // this substitution has not changed in any way
                handledOldSubstitutions.add(newSubstitution);
                continue;
            }

            Substitution oldSubstitution = findEqualSubtitutionExcludingClasses(newSubstitution, a.getSubstitutions(),
                    handledOldSubstitutions);
            if (oldSubstitution != null) {
                // The same substitution was already there, but the set of classes has changed
                Set newClasses = new HashSet<>();
                for (String currentClass : newSubstitution.getClasses()) {
                    if (!oldSubstitution.getClasses().contains(currentClass)) {
                        newClasses.add(currentClass);
                    }
                }
                if (!newClasses.isEmpty()) {
                    diff.newSubstitutions.add(new Substitution(newSubstitution, newClasses));
                }

                Set removedClasses = new HashSet<>();
                for (String currentClass : oldSubstitution.getClasses()) {
                    if (!newSubstitution.getClasses().contains(currentClass)) {
                        removedClasses.add(currentClass);
                    }
                }
                if (!removedClasses.isEmpty()) {
                    diff.removedSubstitutions.add(new Substitution(newSubstitution,
                            removedClasses));
                }
                handledOldSubstitutions.add(oldSubstitution);
                continue;
            }

            oldSubstitution = findSimilarSubstitution(newSubstitution, a.getSubstitutions(), handledOldSubstitutions);
            if (oldSubstitution != null) {
                // there is a similar substitution, create a SubstitutionDiff
                SubstitutionDiff substitutionDiff = SubstitutionDiff.compare(oldSubstitution, newSubstitution);
                // Only use the diff if its complexity is low enough. Otherwise, regard this as a new substitution.
                if (substitutionDiff.getComplexity() <= SubstitutionDiff.MAX_COMPLEXITY) {
                    diff.editedSubstitutions.add(substitutionDiff);
                    handledOldSubstitutions.add(oldSubstitution);
                    continue;
                }
            }

            // There seems to be no matching substitution, so this is new
            diff.newSubstitutions.add(newSubstitution);
        }

        // second run: go through all unhandled old substitutions -> they must have been removed
        for (Substitution oldSubstitution : a.getSubstitutions()) {
            if (!handledOldSubstitutions.contains(oldSubstitution)) {
                diff.removedSubstitutions.add(oldSubstitution);
            }
        }

        return diff;
    }

    private static int calculateSimilarityScore(Substitution a, Substitution b) {
        int score = 0;
        if (Objects.equals(a.getLesson(), b.getLesson())) score++;
        if (Objects.equals(a.getType(), b.getType())) score++;
        if (Objects.equals(a.getSubject(), b.getSubject())) score++;
        if (Objects.equals(a.getPreviousSubject(), b.getPreviousSubject())) score++;
        if (Objects.equals(a.getTeachers(), b.getTeachers())) score++;
        if (Objects.equals(a.getPreviousTeachers(), b.getPreviousTeachers())) score++;
        if (Objects.equals(a.getRoom(), b.getRoom())) score++;
        if (Objects.equals(a.getPreviousRoom(), b.getPreviousRoom())) score++;
        if (Objects.equals(a.getDesc(), b.getDesc())) score++;
        return score;
    }

    private static Substitution findSimilarSubstitution(Substitution subst, Set substs,
                                                        Set handledSubsts) {
        int maxScore = 0;
        Substitution maxScoreSubstitution = null;
        for (Substitution currentSubst : substs) {
            if (currentSubst.getClasses().equals(subst.getClasses())
                    && !handledSubsts.contains(currentSubst)) {
                int score = calculateSimilarityScore(currentSubst, subst);
                if (score > maxScore) {
                    maxScore = score;
                    maxScoreSubstitution = currentSubst;
                }
            }
        }
        if (maxScore > 0) {
            return maxScoreSubstitution;
        } else {
            return null;
        }
    }

    private static Substitution findEqualSubtitutionExcludingClasses(Substitution subst,
                                                                     Set substs,
                                                                     Set handledSubsts) {
        for (Substitution currentSubst : substs) {
            if (currentSubst.equalsExcludingClasses(subst) && !handledSubsts.contains(currentSubst)) {
                return currentSubst;
            }
        }
        return null;
    }

    public LocalDate getDate() {
        return date;
    }

    public void setDate(LocalDate date) {
        this.date = date;
    }

    public String getDateString() {
        return dateString;
    }

    public void setDateString(String dateString) {
        this.dateString = dateString;
    }

    /**
     * @return the set of substitutions that were added to the schedule for this day
     */
    public Set getNewSubstitutions() {
        return newSubstitutions;
    }

    public void setNewSubstitutions(Set newSubstitutions) {
        this.newSubstitutions = newSubstitutions;
    }

    /**
     * @return the set of substitutions that were edited on the schedule for this day
     */
    public Set getEditedSubstitutions() {
        return editedSubstitutions;
    }

    public void setEditedSubstitutions(Set editedSubstitutions) {
        this.editedSubstitutions = editedSubstitutions;
    }

    /**
     * @return the set of substitutions that were removed from the schedule for this day
     */
    public Set getRemovedSubstitutions() {
        return removedSubstitutions;
    }

    public void setRemovedSubstitutions(Set removedSubstitutions) {
        this.removedSubstitutions = removedSubstitutions;
    }

    /**
     * @return the list of messages that were added to the schedule for this day
     */
    public List getNewMessages() {
        return newMessages;
    }

    public void setNewMessages(List newMessages) {
        this.newMessages = newMessages;
    }

    /**
     * @return the list of messages that were removed from the schedule for this day
     */
    public List getRemovedMessages() {
        return removedMessages;
    }

    public void setRemovedMessages(List removedMessages) {
        this.removedMessages = removedMessages;
    }

    public boolean isNotEmpty() {
        return !newMessages.isEmpty() || !removedMessages.isEmpty() || !newSubstitutions.isEmpty()
                || !removedSubstitutions.isEmpty() || !editedSubstitutions.isEmpty();
    }

    public Set getNewSubstitutionsByClassAndExcludedSubject(String theClass, Set excludedSubjects) {
        return SubstitutionSchedule.filterBySubject(excludedSubjects, SubstitutionSchedule
                .filterByClass(theClass, newSubstitutions));
    }

    public Set getRemovedSubstitutionsByClassAndExcludedSubject(String theClass,
                                                                              Set excludedSubjects) {
        return SubstitutionSchedule.filterBySubject(excludedSubjects, SubstitutionSchedule
                .filterByClass(theClass, removedSubstitutions));
    }

    public Set getEditedSubstitutionsByClassAndExcludedSubject(String theClass,
                                                                                 Set excludedSubjects) {
        return SubstitutionScheduleDiff.filterBySubject(excludedSubjects, SubstitutionScheduleDiff
                .filterByClass(theClass, editedSubstitutions));
    }

    public Set getNewSubstitutionsByTeacherAndExcludedSubject(String teacher,
                                                                            Set excludedSubjects) {
        return SubstitutionSchedule.filterBySubject(excludedSubjects, SubstitutionSchedule
                .filterByTeacher(teacher, newSubstitutions));
    }

    public Set getRemovedSubstitutionsByTeacherAndExcludedSubject(String teacher,
                                                                                Set excludedSubjects) {
        return SubstitutionSchedule.filterBySubject(excludedSubjects, SubstitutionSchedule
                .filterByTeacher(teacher, removedSubstitutions));
    }

    public Set getEditedSubstitutionsByTeacherAndExcludedSubject(String teacher,
                                                                                   Set excludedSubjects) {
        return SubstitutionScheduleDiff.filterBySubject(excludedSubjects, SubstitutionScheduleDiff
                .filterByTeacher(teacher, editedSubstitutions));
    }

    public SubstitutionScheduleDayDiff clone() {
        try {
            return (SubstitutionScheduleDayDiff) super.clone();
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException(e);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy