blob: f899dbae6b3ab2a773289189a00aa7f61042e8fd [file] [log] [blame]
/*=============================================================================#
# Copyright (c) 2009, 2020 Stephan Wahlbrink and others.
#
# This program and the accompanying materials are made available under the
# terms of the Eclipse Public License 2.0 which is available at
# https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
# which is available at https://www.apache.org/licenses/LICENSE-2.0.
#
# SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
#
# Contributors:
# Stephan Wahlbrink <sw@wahlbrink.eu> - initial API and implementation
#=============================================================================*/
package org.eclipse.statet.rj.data.impl;
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.Arrays;
import org.eclipse.statet.jcommons.lang.NonNullByDefault;
import org.eclipse.statet.jcommons.lang.Nullable;
import org.eclipse.statet.rj.data.RJIO;
/**
* This implementation is limited to length of 2<sup>31</sup>-1.
*/
@NonNullByDefault
public class RCharacter32Store extends AbstractCharacterStore
implements RDataResizeExtension<String>, ExternalizableRStore, Externalizable {
private int length;
protected @Nullable String [] charValues;
public RCharacter32Store() {
this.length= 0;
this.charValues= EMPTY_STRING_ARRAY;
}
public RCharacter32Store(final int length) {
this.length= length;
this.charValues= new @Nullable String [length];
Arrays.fill(this.charValues, "");
}
public RCharacter32Store(final @Nullable String [] values) {
this.length= values.length;
this.charValues= values;
}
public RCharacter32Store(final @Nullable String [] values, final int length) {
this.length= length;
this.charValues= values;
}
RCharacter32Store(final RCharacter32Store source, final boolean reuse) {
@Nullable String [] values;
if (reuse) {
values= source.charValues;
}
else {
values= ensureCapacity(this.charValues, source.length);
System.arraycopy(source, 0, values, 0, source.length);
}
this.charValues= values;
this.length= source.length;
}
public RCharacter32Store(final RJIO io, final int length) throws IOException {
this.length= length;
this.charValues= new @Nullable String [length];
io.readStringData(this.charValues, length);
}
@Override
public void writeExternal(final RJIO io) throws IOException {
io.writeStringData(this.charValues, this.length);
}
@Override
public void readExternal(final ObjectInput in) throws IOException {
this.length= in.readInt();
this.charValues= new @Nullable String [this.length];
for (int i= 0; i < this.length; i++) {
final int l= in.readInt();
if (l >= 0) {
final char[] c= new char[l];
for (int j= 0; j < l; j++) {
c[j]= in.readChar();
}
this.charValues[i]= new String(c);
}
// else {
// this.charValues[i]= null;
// }
}
}
@Override
public void writeExternal(final ObjectOutput out) throws IOException {
out.writeInt(this.length);
for (int i= 0; i < this.length; i++) {
if (this.charValues[i] != null) {
out.writeInt(this.charValues[i].length());
out.writeChars(this.charValues[i]);
}
else {
out.writeInt(-1);
}
}
}
@Override
protected final boolean isStructOnly() {
return false;
}
public final int length() {
return this.length;
}
@Override
public final long getLength() {
return this.length;
}
@Override
public boolean isNA(final int idx) {
return (this.charValues[idx] == null);
}
@Override
public boolean isNA(final long idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return (this.charValues[(int) idx] == null);
}
@Override
public void setNA(final int idx) {
// if (this.charValues[idx] == null) {
// return;
// }
this.charValues[idx]= null;
}
@Override
public void setNA(final long idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
// if (this.charValues[(int) idx] == null) {
// return;
// }
this.charValues[(int) idx]= null;
}
@Override
public boolean isMissing(final int idx) {
return (this.charValues[idx] == null);
}
@Override
public boolean isMissing(final long idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return (this.charValues[(int) idx] == null);
}
@Override
@SuppressWarnings("null")
public String getChar(final int idx) {
return this.charValues[idx];
}
@Override
@SuppressWarnings("null")
public String getChar(final long idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return this.charValues[(int) idx];
}
@Override
public void setChar(final int idx, final String value) {
// assert (value != null);
this.charValues[idx]= value;
}
@Override
public void setChar(final long idx, final String value) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
// assert (value != null);
this.charValues[(int) idx]= value;
}
private void prepareInsert(final int[] idxs) {
this.charValues= prepareInsert(this.charValues, this.length, idxs);
this.length+= idxs.length;
}
public void insertChar(final int idx, final String value) {
// assert (value != null);
prepareInsert(new int[] { idx });
this.charValues[idx]= value;
}
@Override
public void insertNA(final int idx) {
prepareInsert(new int[] { idx });
this.charValues[idx]= null;
}
@Override
public void insertNA(final int[] idxs) {
if (idxs.length == 0) {
return;
}
prepareInsert(idxs);
for (int i= 0; i < idxs.length; i++) {
this.charValues[idxs[i]]= null;
}
}
@Override
public void remove(final int idx) {
this.charValues= remove(this.charValues, this.length, new int[] { idx });
this.length--;
}
@Override
public void remove(final int[] idxs) {
this.charValues= remove(this.charValues, this.length, idxs);
this.length-= idxs.length;
}
@Override
public @Nullable String get(final int idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return this.charValues[idx];
}
@Override
public @Nullable String get(final long idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return this.charValues[(int) idx];
}
@Override
public @Nullable String [] toArray() {
final var array= new @Nullable String [this.length];
System.arraycopy(this.charValues, 0, array, 0, array.length);
return array;
}
@Override
public long indexOfNA(long fromIdx) {
if (fromIdx >= Integer.MAX_VALUE) {
return -1;
}
if (fromIdx < 0) {
fromIdx= 0;
}
final int l= this.length;
final String[] chars= this.charValues;
for (int i= (int) fromIdx; i < l; i++) {
if (chars[i] == null) {
return i;
}
}
return -1;
}
public int indexOfNA(int fromIdx) {
if (fromIdx < 0) {
fromIdx= 0;
}
final int l= this.length;
final String[] chars= this.charValues;
for (int i= fromIdx; i < l; i++) {
if (chars[i] == null) {
return i;
}
}
return -1;
}
@Override
public long indexOf(final String character, long fromIdx) {
if (character == null
|| fromIdx >= Integer.MAX_VALUE) {
return -1;
}
if (fromIdx < 0) {
fromIdx= 0;
}
final int l= this.length;
final String[] chars= this.charValues;
for (int i= (int) fromIdx; i < l; i++) {
if (chars[i] != null && chars[i].equals(character)) {
return i;
}
}
return -1;
}
public int indexOf(final String character, int fromIdx) {
if (character == null) {
throw new NullPointerException();
}
if (fromIdx < 0) {
fromIdx= 0;
}
final int l= this.length;
final String[] chars= this.charValues;
while (fromIdx < l) {
if (chars[fromIdx] != null && chars[fromIdx].equals(character)) {
return fromIdx;
}
fromIdx++;
}
return -1;
}
}