You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
206 lines
8.5 KiB
206 lines
8.5 KiB
6 years ago
|
import { cutSegmentWithSegments, IntegerSegment, IntegerSegmentsSet } from '../../src/utils/segments';
|
||
|
|
||
|
import 'jest';
|
||
|
|
||
|
|
||
|
function IS(from: number, to: number) {
|
||
|
return new IntegerSegment(from, to);
|
||
|
}
|
||
|
|
||
|
function ISS(xs: [number, number][]) {
|
||
|
return new IntegerSegmentsSet(xs.map(x => IS(x[0], x[1])));
|
||
|
}
|
||
|
|
||
|
function cutSpan(from: number, to: number, cuts: [number, number][]): [number, number][] {
|
||
|
return cutSegmentWithSegments(
|
||
|
new IntegerSegment(from, to),
|
||
|
cuts.map(([from, to]) => new IntegerSegment(from, to))
|
||
|
).map(({ from, to }) => [from, to] as [number, number]);
|
||
|
}
|
||
|
|
||
|
describe('IntegerSegment', function() {
|
||
|
it('should throw an error on float from or to', function() {
|
||
|
expect(() => IS(0.1, 0)).toThrow();
|
||
|
expect(() => IS(1, 5.04)).toThrow();
|
||
|
expect(() => IS(1, 5)).not.toThrow();
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('IntegerSegment.intersect', function() {
|
||
|
it('return undefined if segments don`t intersect', function() {
|
||
|
expect(IS(4, 5).insersect(IS(6, 10))).toEqual(undefined);
|
||
|
expect(IS(7, 10).insersect(IS(1, 3))).toEqual(undefined);
|
||
|
});
|
||
|
|
||
|
it('return a point when borders intersect', function() {
|
||
|
expect(IS(4, 5).insersect(IS(5, 6))).toEqual(IS(5, 5));
|
||
|
expect(IS(4, 5).insersect(IS(4, 4))).toEqual(IS(4, 4));
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('IntegerSegmentSet constructor', function() {
|
||
|
it('can construct from empty segments list', function() {
|
||
|
expect(() => ISS([])).not.toThrow();
|
||
|
});
|
||
|
|
||
|
it('should sort segments', function() {
|
||
|
expect(ISS([[10, 15], [5, 8]]).segments).toEqual([IS(5, 8), IS(10, 15)]);
|
||
|
expect(ISS([[10, 15], [-Infinity, 8]]).segments).toEqual([IS(-Infinity, 8), IS(10, 15)]);
|
||
|
expect(ISS([[10, Infinity], [-Infinity, 8]]).segments).toEqual([IS(-Infinity, 8), IS(10, Infinity)]);
|
||
|
});
|
||
|
|
||
|
it('should merge segments', function() {
|
||
|
expect(ISS([[5, 10], [7, 20]]).segments).toEqual([IS(5, 20)]); // it's because 7 <= 10
|
||
|
expect(ISS([[5, 10], [10, 20]]).segments).toEqual([IS(5, 20)]);
|
||
|
expect(ISS([[5, 10], [11, 20]]).segments).toEqual([IS(5, 20)]); // it's because [..., 10], [11, ...],
|
||
|
// there is nothing between 10 and 11
|
||
|
expect(ISS([[3, 11], [4, 10]]).segments).toEqual([IS(3, 11)]);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('IntegerSegmentSet.inversed', function() {
|
||
|
it('should return Infinite segment whes set is empty', function() {
|
||
|
let setA = ISS([]);
|
||
|
expect(setA.inversed()).toEqual(ISS([[-Infinity, Infinity]]));
|
||
|
});
|
||
|
|
||
|
it('should return empty segment whes set is infinite', function() {
|
||
|
let setA = ISS([[-Infinity, Infinity]]);
|
||
|
expect(setA.inversed()).toEqual(ISS([]));
|
||
|
});
|
||
|
|
||
|
it('should inverse a point', function() {
|
||
|
expect(ISS([[4, 4]]).inversed()).toEqual(ISS([[-Infinity, 3], [5, Infinity]]));
|
||
|
});
|
||
|
|
||
|
it('should inverse basic cases', function() {
|
||
|
expect(ISS([[3, 10]]).inversed()).toEqual(ISS([[-Infinity, 2], [11, Infinity]]));
|
||
|
expect(ISS([[3, 10], [15, 20]]).inversed()).toEqual(ISS([[-Infinity, 2], [11, 14] , [21, Infinity]]));
|
||
|
});
|
||
|
|
||
|
it('should inverse infinites', function() {
|
||
|
expect(ISS([[3, Infinity]]).inversed()).toEqual(ISS([[-Infinity, 2]]));
|
||
|
expect(ISS([[-Infinity, 3]]).inversed()).toEqual(ISS([[4, Infinity]]));
|
||
|
});
|
||
|
|
||
|
});
|
||
|
|
||
|
describe('IntegerSegmentSet.intersected', function() {
|
||
|
it('should return empty set if one of intersection is empty', function() {
|
||
|
let setA = ISS([]);
|
||
|
let setB = ISS([[1, 5]]);
|
||
|
expect(setA.intersect(setB).segments).toEqual([]);
|
||
|
expect(setB.intersect(setA).segments).toEqual([]);
|
||
|
});
|
||
|
|
||
|
it('should intersect two segments', function() {
|
||
|
let setA = ISS([[2, 5]]);
|
||
|
let setB = ISS([[1, 4]]);
|
||
|
expect(setA.intersect(setB)).toEqual(ISS([[2, 4]]));
|
||
|
});
|
||
|
|
||
|
it('should intersect basic cases', function() {
|
||
|
let setA = ISS([[2, 5], [6, 10]]);
|
||
|
let setB = ISS([[1, 9]]);
|
||
|
let setC = ISS([[2, 5], [6, 10]]);
|
||
|
let setD = ISS([[4, 4], [10, 10]]);
|
||
|
let setE = ISS([[4, 4], [10, 10], [12, 15]]);
|
||
|
expect(setA.intersect(setB)).toEqual(ISS([[2, 5], [6, 9]]));
|
||
|
expect(setA.intersect(setC)).toEqual(ISS([[2, 5], [6, 10]]));
|
||
|
expect(setA.intersect(setD)).toEqual(ISS([[4, 4], [10, 10]]));
|
||
|
expect(setA.intersect(setE)).toEqual(ISS([[4, 4], [10, 10]]));
|
||
|
expect(setE.intersect(setA)).toEqual(ISS([[4, 4], [10, 10]]));
|
||
|
});
|
||
|
|
||
|
});
|
||
|
|
||
|
describe('cutSpanWithSpans', function() {
|
||
|
|
||
|
it('should handle empty input spans list case', function() {
|
||
|
expect(cutSpan(4, 10, [])).toEqual([[4, 10]]);
|
||
|
});
|
||
|
|
||
|
it('should handle works fine one point results', function() {
|
||
|
expect(cutSpan(1, 10, [[2, 10]])).toEqual([[1, 1]]);
|
||
|
expect(cutSpan(1, 10, [[2, 11]])).toEqual([[1, 1]]);
|
||
|
expect(cutSpan(1, 10, [[1, 9]])).toEqual([[10, 10]]);
|
||
|
expect(cutSpan(1, 10, [[0, 9]])).toEqual([[10, 10]]);
|
||
|
expect(cutSpan(1, 10, [[1, 4], [6, 10]])).toEqual([[5, 5]]);
|
||
|
expect(cutSpan(1, 10, [[2, 9]])).toEqual([[1, 1], [10, 10]]);
|
||
|
});
|
||
|
|
||
|
it('should throw error is cut contains float border', function() {
|
||
|
expect(() => cutSpan(0, 10, [[0.1, 5]])).toThrow()
|
||
|
expect(() => cutSpan(1, 10, [[0.9, 0.0]])).toThrow();
|
||
|
expect(() => cutSpan(0.5, 10, [[1, 5]])).toThrow();
|
||
|
});
|
||
|
|
||
|
it('should handle one-point cuts', function() {
|
||
|
expect(cutSpan(1, 10, [[5, 5]])).toEqual([[1, 4], [6, 10]]);
|
||
|
expect(cutSpan(1, 10, [[1, 1]])).toEqual([[2, 10]]);
|
||
|
expect(cutSpan(1, 10, [[10, 10]])).toEqual([[1, 9]]);
|
||
|
expect(cutSpan(1, 10, [[11, 11]])).toEqual([[1, 10]]);
|
||
|
expect(cutSpan(1, 15, [[11, 11], [12, 12]])).toEqual([[1, 10], [13, 15]]);
|
||
|
});
|
||
|
|
||
|
it('should handle basic cases', function() {
|
||
|
let cutSpans = [[3, 4], [6, 8], [11, 20]] as [number, number][];
|
||
|
|
||
|
expect(cutSpan(0, 11, cutSpans)).toEqual([[0, 2], [5, 5], [9, 10]]);
|
||
|
expect(cutSpan(5, 11, cutSpans)).toEqual([[5, 5], [9, 10]]);
|
||
|
expect(cutSpan(4, 10, cutSpans)).toEqual([[5, 5], [9, 10]]);
|
||
|
expect(cutSpan(5, 10, cutSpans)).toEqual([[5, 5], [9, 10]]);
|
||
|
expect(cutSpan(4, 20, cutSpans)).toEqual([[5, 5], [9, 10]]);
|
||
|
expect(cutSpan(4, 21, cutSpans)).toEqual([[5, 5], [9, 10], [21, 21]]);
|
||
|
expect(cutSpan(2, 20, cutSpans)).toEqual([[2, 2], [5, 5], [9, 10]]);
|
||
|
expect(cutSpan(2, 21, cutSpans)).toEqual([[2, 2], [5, 5], [9, 10], [21, 21]]);
|
||
|
expect(cutSpan(3, 11, cutSpans)).toEqual([[5, 5], [9, 10]]);
|
||
|
expect(cutSpan(3, 20, cutSpans)).toEqual([[5, 5], [9, 10]]);
|
||
|
expect(cutSpan(4, 7, [[3, 5], [6, 8]])).toEqual([]);
|
||
|
});
|
||
|
|
||
|
it('should handle infitie span and infinite cuts', function() {
|
||
|
expect(cutSpan(0, Infinity, [[5, 10]])).toEqual([[0, 4], [11, Infinity]]);
|
||
|
expect(cutSpan(0, 6, [[0, Infinity]])).toEqual([]);
|
||
|
expect(cutSpan(0, 6, [[2, Infinity]])).toEqual([[0, 1]]);
|
||
|
expect(cutSpan(-Infinity, Infinity, [[-Infinity, Infinity]])).toEqual([]);
|
||
|
});
|
||
|
|
||
|
it('should handle case when from and to are inside of one big span', function() {
|
||
|
expect(cutSpan(4, 10, [[1, 20]])).toEqual([]);
|
||
|
expect(cutSpan(4, 10, [[1, 10]])).toEqual([]);
|
||
|
expect(cutSpan(4, 10, [[4, 20]])).toEqual([]);
|
||
|
expect(cutSpan(4, 10, [[4, 10]])).toEqual([]);
|
||
|
});
|
||
|
|
||
|
it('should be ready to get not-sorted cuts', function() {
|
||
|
expect(cutSpan(0, 20, [[3, 5], [1, 2]])).toEqual([[0, 0], [6, 20]]);
|
||
|
expect(cutSpan(0, 20, [[3, 5], [1, 2], [0, 0]])).toEqual([[6, 20]]);
|
||
|
});
|
||
|
|
||
|
it('should be ready to get overlayed cuts', function() {
|
||
|
expect(cutSpan(0, 20, [[3, 5], [4, 10]])).toEqual([[0, 2], [11, 20]]);
|
||
|
expect(cutSpan(0, 20, [[3, 9], [4, 9]])).toEqual([[0, 2], [10, 20]]);
|
||
|
expect(cutSpan(0, 20, [[3, 11], [4, 10]])).toEqual([[0, 2], [12, 20]]);
|
||
|
expect(cutSpan(0, 20, [[3, 11], [3, 12]])).toEqual([[0, 2], [13, 20]]);
|
||
|
expect(cutSpan(0, 20, [[3, 11], [3, 12], [3, 10], [3, 15], [3, 14]])).toEqual([[0, 2], [16, 20]]);
|
||
|
expect(cutSpan(0, 20, [[2, 11], [3, 12]])).toEqual([[0, 1], [13, 20]]);
|
||
|
expect(cutSpan(0, 20, [[2, 15], [3, 12]])).toEqual([[0, 1], [16, 20]]);
|
||
|
expect(cutSpan(0, 20, [[2, 15], [3, 12], [1, 18]])).toEqual([[0, 0], [19, 20]]);
|
||
|
expect(cutSpan(0, 20, [[2, 15], [3, Infinity], [1, 18]])).toEqual([[0, 0]]);
|
||
|
expect(cutSpan(0, 20, [[3, 3], [3, Infinity]])).toEqual([[0, 2]]);
|
||
|
expect(cutSpan(0, 20, [[3, 3], [3, Infinity], [3, 3]])).toEqual([[0, 2]]);
|
||
|
expect(cutSpan(0, 20, [[3, 3], [3, Infinity], [3, 3], [4, 4]])).toEqual([[0, 2]]);
|
||
|
expect(cutSpan(0, 20, [[3, 3], [3, Infinity], [3, 3], [4, 4], [3, 5]])).toEqual([[0, 2]]);
|
||
|
expect(cutSpan(-Infinity, Infinity, [[3, 3], [3, Infinity], [3, 3], [4, 4], [3, 5]])).toEqual([[-Infinity, 2]]);
|
||
|
});
|
||
|
|
||
|
it('should handle cuts from point span', function() {
|
||
|
expect(cutSpan(1, 1, [[1, 1]])).toEqual([]);
|
||
|
expect(cutSpan(1, 1, [[0, 2]])).toEqual([]);
|
||
|
expect(cutSpan(1, 1, [[0, 1]])).toEqual([]);
|
||
|
expect(cutSpan(1, 1, [[1, 2]])).toEqual([]);
|
||
|
});
|
||
|
|
||
|
});
|