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

org.d2ab.collection.BiMappedList Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2016 Daniel Skogquist Åborg
 *
 * 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.d2ab.collection;

import java.util.*;
import java.util.function.Function;

/**
 * A {@link List} that presents a mapped view of another {@link List}.
 */
public interface BiMappedList {
	static  List from(List list, Function mapper,
	                                  Function backMapper) {
		if (list instanceof RandomAccess)
			return new RandomAccessList<>(list, mapper, backMapper);
		else
			return new SequentialList<>(list, mapper, backMapper);
	}

	class RandomAccessList extends AbstractList implements RandomAccess, SizedIterable {
		private final List list;
		private final SizeType sizeType;
		private final Function mapper;
		private final Function backMapper;

		public RandomAccessList(List list, Function mapper,
		                        Function backMapper) {
			this.list = list;
			this.sizeType = Iterables.sizeType(list);
			this.mapper = mapper;
			this.backMapper = backMapper;
		}

		@Override
		public U get(int index) {
			return mapper.apply(list.get(index));
		}

		@Override
		public U remove(int index) {
			return mapper.apply(list.remove(index));
		}

		@Override
		public U set(int index, U element) {
			return mapper.apply(list.set(index, backMapper.apply(element)));
		}

		@Override
		public void add(int index, U element) {
			list.add(index, backMapper.apply(element));
		}

		@Override
		public int size() {
			return list.size();
		}

		@Override
		public SizeType sizeType() {
			return sizeType;
		}

		@Override
		public boolean add(U element) {
			return list.add(backMapper.apply(element));
		}
	}

	class SequentialList extends AbstractSequentialList implements SizedIterable {
		private final List list;
		private final SizeType sizeType;
		private final Function mapper;
		private final Function backMapper;

		public SequentialList(List list, Function mapper,
		                      Function backMapper) {
			this.list = list;
			this.sizeType = Iterables.sizeType(list);
			this.mapper = mapper;
			this.backMapper = backMapper;
		}

		@Override
		public ListIterator listIterator(int index) {
			ListIterator listIterator = list.listIterator(index);
			return new ListIterator() {
				@Override
				public boolean hasNext() {
					return listIterator.hasNext();
				}

				@Override
				public U next() {
					return mapper.apply(listIterator.next());
				}

				@Override
				public boolean hasPrevious() {
					return listIterator.hasPrevious();
				}

				@Override
				public U previous() {
					return mapper.apply(listIterator.previous());
				}

				@Override
				public int nextIndex() {
					return listIterator.nextIndex();
				}

				@Override
				public int previousIndex() {
					return listIterator.previousIndex();
				}

				@Override
				public void remove() {
					listIterator.remove();
				}

				@Override
				public void set(U u) {
					listIterator.set(backMapper.apply(u));
				}

				@Override
				public void add(U u) {
					listIterator.add(backMapper.apply(u));
				}
			};
		}

		@Override
		public int size() {
			return list.size();
		}

		@Override
		public SizeType sizeType() {
			return sizeType;
		}

		@Override
		public boolean add(U element) {
			return list.add(backMapper.apply(element));
		}
	}
}