commonMain.ag.granular.tiff.io.ByteReader.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of tiffany-jvm Show documentation
Show all versions of tiffany-jvm Show documentation
A library for parsing Tagged Image File Format (Tiff) files
package ag.granular.tiff.io
import ag.granular.io.ByteBuffer
import ag.granular.io.ByteOrder
/**
* Read through a byte array
*
* @param bytes
* bytes
* @param byteReaderOrder
* byte order
*/
class ByteReader(
private val bytes: ByteArray,
var byteReaderOrder: ByteOrder = ByteOrder.LITTLE_ENDIAN
) {
/**
* Next byte index to read
*/
var nextByte = 0
/**
* Check if there is at least one more byte left to read
*
* @return true more bytes left to read
*/
fun hasByte(): Boolean = hasBytes(1)
/**
* Check if there is at least one more byte left to read
*
* @param offset
* byte offset
* @return true more bytes left to read
*/
fun hasByte(offset: Int): Boolean = hasBytes(offset, 1)
/**
* Check if there are the provided number of bytes left to read
*
* @param count
* number of bytes
* @return true if has at least the number of bytes left
*/
fun hasBytes(count: Int): Boolean = hasBytes(nextByte, count)
/**
* Check if there are the provided number of bytes left to read
*
* @param offset
* byte offset
* @param count
* number of bytes
* @return true if has at least the number of bytes left
*/
fun hasBytes(offset: Int, count: Int): Boolean = offset + count <= bytes.size
/**
* Read a String from the provided number of bytes
*
* @param num
* number of bytes
* @return String
* @throws UnsupportedEncodingException
*/
// @Throws(UnsupportedEncodingException::class)
fun readString(num: Int): String? {
val value = readString(nextByte, num)
nextByte += num
return value
}
/**
* Read a String from the provided number of bytes
*
* @param offset
* byte offset
* @param num
* number of bytes
* @return String
* @throws UnsupportedEncodingException
*/
// @Throws(UnsupportedEncodingException::class)
@UseExperimental(ExperimentalStdlibApi::class)
fun readString(offset: Int, num: Int): String? = verifyRemainingBytes(offset, num) {
if (num != 1 || bytes[offset].toInt() != 0) {
bytes.decodeToString(offset, offset + num)
// String(bytes, offset, num, Charset.forName("ASCII"))
} else {
null
}
}
/**
* Read a byte
*
* @return byte
*/
fun readByte(): Byte {
val value = readByte(nextByte)
nextByte++
return value
}
/**
* Read a byte
*
* @param offset
* byte offset
* @return byte
*/
fun readByte(offset: Int): Byte = verifyRemainingBytes(offset, 1) {
bytes[offset]
}
/**
* Read an unsigned byte
*
* @return unsigned byte as short
*/
fun readUnsignedByte(): Short {
val value = readUnsignedByte(nextByte)
nextByte++
return value
}
/**
* Read an unsigned byte
*
* @param offset
* byte offset
* @return unsigned byte as short
*/
fun readUnsignedByte(offset: Int): Short = (readByte(offset).toInt() and 0xff).toShort()
/**
* Read a number of bytes
*
* @param num
* number of bytes
* @return bytes
*/
fun readBytes(num: Int): ByteArray {
val readBytes = readBytes(nextByte, num)
nextByte += num
return readBytes
}
/**
* Read a number of bytes
*
* @param offset
* byte offset
* @param num
* number of bytes
* @return bytes
*/
fun readBytes(offset: Int, num: Int): ByteArray = verifyRemainingBytes(offset, num) {
bytes.copyOfRange(offset, offset + num)
}
/**
* Read a short
*
* @return short
*/
fun readShort(): Short {
val value = readShort(nextByte)
nextByte += 2
return value
}
/**
* Read a short
*
* @param offset
* byte offset
* @return short
*/
fun readShort(offset: Int): Short = verifyRemainingBytes(offset, 2) {
ByteBuffer.wrap(bytes, offset, 2).order(byteReaderOrder).getShort()
// byteReadPacket(bytes, offset, 2) {
// readShort(byteReaderOrder)
// }
}
/**
* Read an unsigned short
*
* @return unsigned short as int
*/
fun readUnsignedShort(): Int {
val value = readUnsignedShort(nextByte)
nextByte += 2
return value
}
/**
* Read an unsigned short
*
* @param offset
* byte offset
* @return unsigned short as int
*/
fun readUnsignedShort(offset: Int): Int = readShort(offset).toInt() and 0xffff
/**
* Read an integer
*
* @return integer
*/
fun readInt(): Int {
val value = readInt(nextByte)
nextByte += 4
return value
}
/**
* Read an integer
*
* @param offset
* byte offset
* @return integer
*/
fun readInt(offset: Int): Int = verifyRemainingBytes(offset, 4) {
ByteBuffer.wrap(bytes, offset, 4).order(byteReaderOrder).getInt()
// byteReadPacket(bytes, offset, 4) {
// readInt(byteReaderOrder)
// }
}
/**
* Read an unsigned int
*
* @return unsigned int as long
*/
fun readUnsignedInt(): Long {
val value = readUnsignedInt(nextByte)
nextByte += 4
return value
}
/**
* Read an unsigned int
*
* @param offset
* byte offset
* @return unsigned int as long
*/
fun readUnsignedInt(offset: Int): Long = readInt(offset).toLong() and 0xffffffffL
/**
* Read a float
*
* @return float
*/
fun readFloat(): Float {
val value = readFloat(nextByte)
nextByte += 4
return value
}
/**
* Read a float
*
* @param offset
* byte offset
* @return float
*/
fun readFloat(offset: Int): Float = verifyRemainingBytes(offset, 4) {
ByteBuffer.wrap(bytes, offset, 4).order(byteReaderOrder).getFloat()
// byteReadPacket(bytes, offset, 4) {
// readFloat(byteReaderOrder)
// }
}
/**
* Read a double
*
* @return double
*/
fun readDouble(): Double {
val value = readDouble(nextByte)
nextByte += 8
return value
}
/**
* Read a double
*
* @param offset
* byte offset
* @return double
*/
fun readDouble(offset: Int): Double = verifyRemainingBytes(offset, 8) {
ByteBuffer.wrap(bytes, offset, 8).order(byteReaderOrder).getDouble()
// byteReadPacket(bytes, offset, 8) {
// readDouble(byteReaderOrder)
// }
}
/**
* Get the byte length
*
* @return byte length
*/
fun byteLength(): Int = bytes.size
/**
* Verify with the remaining bytes that there are enough remaining to read
* the provided amount
*
* @param offset
* byte offset
* @param bytesToRead
* number of bytes to read
*/
private fun verifyRemainingBytes(offset: Int, bytesToRead: Int, block: () -> T): T {
if (offset + bytesToRead > bytes.size) {
throw IllegalStateException(
"No more remaining bytes to read. Total Bytes: " +
bytes.size + ", Byte offset: " + offset +
", Attempted to read: " + bytesToRead
)
}
return block()
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy