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

de.regnis.q.sequence.QSequenceDifferenceBlockShifter Maven / Gradle / Ivy

There is a newer version: 1.1.0.beta
Show newest version
/*
 * ====================================================================
 * Copyright (c) 2004 Marc Strapetz, [email protected]. 
 * All rights reserved.
 *
 * This software is licensed as described in the file COPYING, which
 * you should have received as part of this distribution. Use is
 * subject to license terms.
 * ====================================================================
 */

package de.regnis.q.sequence;

import java.util.*;

import de.regnis.q.sequence.core.*;
import de.regnis.q.sequence.media.*;

/**
 * @author Marc Strapetz
 */
public class QSequenceDifferenceBlockShifter {

	// Static =================================================================

	public static void joinBlocks(List blocks) {
		QSequenceDifferenceBlock lastBlock = null;
		for (int index = 0; index < blocks.size();) {
			final QSequenceDifferenceBlock block = (QSequenceDifferenceBlock)blocks.get(index);
			if (lastBlock == null) {
				index++;
				lastBlock = block;
				continue;
			}

			QSequenceAssert.assertTrue(lastBlock.getLeftTo() < block.getLeftFrom());
			QSequenceAssert.assertTrue(lastBlock.getRightTo() < block.getRightFrom());

			if (lastBlock.getLeftTo() + 1 != block.getLeftFrom()) {
				QSequenceAssert.assertTrue(lastBlock.getRightTo() != block.getRightFrom() + 1);
				lastBlock = block;
				index++;
				continue;
			}

			if (lastBlock.getRightTo() + 1 != block.getRightFrom()) {
				QSequenceAssert.assertTrue(lastBlock.getLeftTo() != block.getLeftFrom() + 1);
				lastBlock = block;
				index++;
				continue;
			}

			lastBlock.setLeftTo(block.getLeftTo());
			lastBlock.setRightTo(block.getRightTo());
			blocks.remove(index);
			continue;
		}
	}

	// Fields =================================================================

	private final QSequenceMedia media;
	private final QSequenceMediaComparer comparer;

	// Setup ==================================================================

	public QSequenceDifferenceBlockShifter(QSequenceMedia media, QSequenceMediaComparer comparer) {
		QSequenceAssert.assertNotNull(media);
		QSequenceAssert.assertNotNull(comparer);

		this.media = media;
		this.comparer = comparer;
	}

	// Accessing ==============================================================

	public void shiftBlocks(List blocks) throws QSequenceException {
		if (blocks.isEmpty()) {
			return;
		}

		joinBlocks(blocks);

		for (int index = 0; index < blocks.size();) {
			if (tryShiftUp(blocks, index, true)) {
				continue;
			}

			index++;
		}

		for (int index = 0; index < blocks.size();) {
			if (tryShiftDown(blocks, index)) {
				continue;
			}

			index++;
		}
	}

	public boolean tryShiftUp(List blocks, int blockIndex, boolean requireMerge) throws QSequenceException {
		if (blockIndex == 0) {
			return false;
		}

		final QSequenceDifferenceBlock prevBlock = (QSequenceDifferenceBlock)blocks.get(blockIndex - 1);
		final QSequenceDifferenceBlock block = (QSequenceDifferenceBlock)blocks.get(blockIndex);

		final int prevLeftTo = prevBlock.getLeftTo();
		final int prevRightTo = prevBlock.getRightTo();

		int leftFrom = block.getLeftFrom();
		int leftTo = block.getLeftTo();
		int rightFrom = block.getRightFrom();
		int rightTo = block.getRightTo();

		QSequenceAssert.assertTrue(leftFrom > prevLeftTo);
		QSequenceAssert.assertTrue(rightFrom > prevRightTo);
		QSequenceAssert.assertTrue(leftFrom <= leftTo || rightFrom <= rightTo);

		if (leftFrom - prevLeftTo != rightFrom - prevRightTo) {
			return false;
		}

		while (leftFrom > prevLeftTo + 1) {
			if (leftFrom <= leftTo && !comparer.equalsLeft(leftFrom - 1, leftTo)) {
				break;
			}

			if (rightFrom <= rightTo && !comparer.equalsRight(rightFrom - 1, rightTo)) {
				break;
			}

			leftFrom--;
			leftTo--;
			rightFrom--;
			rightTo--;
		}

		if (leftFrom > prevLeftTo + 1) {
			if (requireMerge) {
				return false;
			}

			block.setLeftFrom(leftFrom);
			block.setLeftTo(leftTo);
			block.setRightFrom(rightFrom);
			block.setRightTo(rightTo);
		}
		else {
			prevBlock.setLeftTo(prevBlock.getLeftTo() + (leftTo - leftFrom + 1));
			prevBlock.setRightTo(prevBlock.getRightTo() + (rightTo - rightFrom + 1));
			blocks.remove(blockIndex);
		}

		return true;
	}

	public boolean tryShiftDown(List blocks, int blockIndex) throws QSequenceException {
		final QSequenceDifferenceBlock nextBlock = blockIndex < blocks.size() - 1
		    ? (QSequenceDifferenceBlock)blocks.get(blockIndex + 1) : null;
		final QSequenceDifferenceBlock block = (QSequenceDifferenceBlock)blocks.get(blockIndex);

		final int nextLeftFrom = nextBlock != null ? nextBlock.getLeftFrom() : media.getLeftLength();
		final int nextRightFrom = nextBlock != null ? nextBlock.getRightFrom() : media.getRightLength();

		int leftFrom = block.getLeftFrom();
		int leftTo = block.getLeftTo();
		int rightFrom = block.getRightFrom();
		int rightTo = block.getRightTo();

		QSequenceAssert.assertTrue(leftTo < nextLeftFrom);
		QSequenceAssert.assertTrue(rightTo < nextRightFrom);
		QSequenceAssert.assertTrue(leftFrom <= leftTo || rightFrom <= rightTo);

		while (leftTo < nextLeftFrom - 1 && rightTo < nextRightFrom - 1) {
			if (leftFrom <= leftTo && !comparer.equalsLeft(leftFrom, leftTo + 1)) {
				break;
			}

			if (rightFrom <= rightTo && !comparer.equalsRight(rightFrom, rightTo + 1)) {
				break;
			}

			leftFrom++;
			leftTo++;
			rightFrom++;
			rightTo++;
		}

		if (nextBlock != null && leftTo == nextLeftFrom - 1 && rightTo == nextRightFrom - 1) {
			nextBlock.setLeftFrom(nextBlock.getLeftFrom() - (leftTo - leftFrom + 1));
			nextBlock.setRightFrom(nextBlock.getRightFrom() - (rightTo - rightFrom + 1));
			blocks.remove(blockIndex);
			return true;
		}

		block.setLeftFrom(leftFrom);
		block.setLeftTo(leftTo);
		block.setRightFrom(rightFrom);
		block.setRightTo(rightTo);
		return false;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy