commit
f64d97c463
5 changed files with 346 additions and 139 deletions
|
@ -49,7 +49,7 @@ proc decodeImage*(data: string): Image {.raises: [PixieError].} =
|
||||||
(data.readStr(0, 5) == xmlSignature or data.readStr(0, 4) == svgSignature):
|
(data.readStr(0, 5) == xmlSignature or data.readStr(0, 4) == svgSignature):
|
||||||
newImage(parseSvg(data))
|
newImage(parseSvg(data))
|
||||||
elif data.len > 6 and data.readStr(0, 6) in gifSignatures:
|
elif data.len > 6 and data.readStr(0, 6) in gifSignatures:
|
||||||
decodeGif(data)
|
newImage(decodeGif(data))
|
||||||
elif data.len > (14+8) and data.readStr(0, 4) == qoiSignature:
|
elif data.len > (14+8) and data.readStr(0, 4) == qoiSignature:
|
||||||
decodeQoi(data).convertToImage()
|
decodeQoi(data).convertToImage()
|
||||||
elif data.len > 9 and data.readStr(0, 2) in ppmSignatures:
|
elif data.len > 9 and data.readStr(0, 2) in ppmSignatures:
|
||||||
|
|
|
@ -1,8 +1,20 @@
|
||||||
import chroma, flatty/binny, math, pixie/common, pixie/images, zippy/bitstreams
|
import chroma, flatty/binny, pixie/common, pixie/images, std/math, std/strutils,
|
||||||
|
vmath, zippy/bitstreams
|
||||||
|
|
||||||
|
# See: https://www.w3.org/Graphics/GIF/spec-gif89a.txt
|
||||||
|
|
||||||
const gifSignatures* = @["GIF87a", "GIF89a"]
|
const gifSignatures* = @["GIF87a", "GIF89a"]
|
||||||
|
|
||||||
# See: https://en.wikipedia.org/wiki/GIF
|
type
|
||||||
|
Gif* = ref object
|
||||||
|
frames*: seq[Image]
|
||||||
|
intervals*: seq[float32] # Floating point seconds
|
||||||
|
duration*: float32
|
||||||
|
|
||||||
|
ControlExtension = object
|
||||||
|
fields: uint8
|
||||||
|
delayTime: uint16
|
||||||
|
transparentColorIndex: uint8
|
||||||
|
|
||||||
template failInvalid() =
|
template failInvalid() =
|
||||||
raise newException(PixieError, "Invalid GIF buffer, unable to load")
|
raise newException(PixieError, "Invalid GIF buffer, unable to load")
|
||||||
|
@ -10,173 +22,356 @@ template failInvalid() =
|
||||||
when defined(release):
|
when defined(release):
|
||||||
{.push checks: off.}
|
{.push checks: off.}
|
||||||
|
|
||||||
proc decodeGif*(data: string): Image {.raises: [PixieError].} =
|
proc decodeGif*(data: string): Gif {.raises: [PixieError].} =
|
||||||
## Decodes GIF data into an Image.
|
## Decodes GIF data.
|
||||||
|
if data.len < 13:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
if data.len <= 13: failInvalid()
|
if data[0 .. 5] notin gifSignatures:
|
||||||
let version = data[0 .. 5]
|
|
||||||
if version notin gifSignatures:
|
|
||||||
raise newException(PixieError, "Invalid GIF file signature")
|
raise newException(PixieError, "Invalid GIF file signature")
|
||||||
|
|
||||||
|
result = Gif()
|
||||||
|
|
||||||
let
|
let
|
||||||
# Read information about the image.
|
screenWidth = data.readInt16(6).int
|
||||||
width = data.readInt16(6).int
|
screenHeight = data.readInt16(8).int
|
||||||
height = data.readInt16(8).int
|
globalFlags = data.readUint8(10).int
|
||||||
flags = data.readUint8(10).int
|
hasGlobalColorTable = (globalFlags and 0b10000000) != 0
|
||||||
hasColorTable = (flags and 0x80) != 0
|
globalColorTableSize = 2 ^ ((globalFlags and 0b00000111) + 1)
|
||||||
originalDepth = ((flags and 0x70) shr 4) + 1
|
bgColorIndex = data.readUint8(11).int
|
||||||
colorTableSorted = (flags and 0x8) != 0
|
pixelAspectRatio = data.readUint8(12)
|
||||||
colorTableSize = 2 ^ ((flags and 0x7) + 1)
|
|
||||||
bgColorIndex = data.readUint8(11)
|
|
||||||
pixelAspectRatio = data.readUint8(11)
|
|
||||||
|
|
||||||
result = newImage(width, height)
|
if bgColorIndex > globalColorTableSize:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
if pixelAspectRatio != 0:
|
||||||
|
raise newException(PixieError, "Unsupported GIF, pixel aspect ratio")
|
||||||
|
|
||||||
|
var pos = 13
|
||||||
|
|
||||||
|
if pos + globalColorTableSize * 3 > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
# Read the main color table.
|
|
||||||
var
|
var
|
||||||
colors: seq[ColorRGBA]
|
globalColorTable: seq[ColorRGBX]
|
||||||
i = 13
|
bgColor: ColorRGBX
|
||||||
if hasColorTable:
|
if hasGlobalColorTable:
|
||||||
if i + colorTableSize * 3 >= data.len: failInvalid()
|
globalColorTable.setLen(globalColorTableSize)
|
||||||
for c in 0 ..< colorTableSize:
|
for i in 0 ..< globalColorTable.len:
|
||||||
let
|
globalColorTable[i] = rgbx(
|
||||||
r = data.readUint8(i + 0)
|
data.readUint8(pos + 0),
|
||||||
g = data.readUint8(i + 1)
|
data.readUint8(pos + 1),
|
||||||
b = data.readUint8(i + 2)
|
data.readUint8(pos + 2),
|
||||||
colors.add(rgba(r, g, b, 255))
|
255
|
||||||
i += 3
|
)
|
||||||
|
pos += 3
|
||||||
|
bgColor = globalColorTable[bgColorIndex]
|
||||||
|
|
||||||
# Read the image blocks.
|
proc skipSubBlocks() =
|
||||||
|
while true: # Skip data sub-blocks
|
||||||
|
if pos + 1 > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
let subBlockSize = data.readUint8(pos).int
|
||||||
|
inc pos
|
||||||
|
|
||||||
|
if subBlockSize == 0:
|
||||||
|
break
|
||||||
|
|
||||||
|
pos += subBlockSize
|
||||||
|
|
||||||
|
var controlExtension: ControlExtension
|
||||||
while true:
|
while true:
|
||||||
let blockType = data.readUint8(i)
|
if pos + 1 > data.len:
|
||||||
i += 1
|
failInvalid()
|
||||||
|
|
||||||
|
let blockType = data.readUint8(pos)
|
||||||
|
inc pos
|
||||||
|
|
||||||
case blockType:
|
case blockType:
|
||||||
of 0x2c: # Read IMAGE block.
|
of 0x2c: # Image
|
||||||
if i + 9 >= data.len: failInvalid()
|
if pos + 9 > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
let
|
let
|
||||||
left = data.readUint16(i + 0)
|
imageLeftPos = data.readUint16(pos + 0).int
|
||||||
top = data.readUint16(i + 2)
|
imageTopPos = data.readUint16(pos + 2).int
|
||||||
w = data.readUint16(i + 4).int
|
imageWidth = data.readUint16(pos + 4).int
|
||||||
h = data.readUint16(i + 6).int
|
imageHeight = data.readUint16(pos + 6).int
|
||||||
flags = data.readUint8(i + 8)
|
imageFlags = data.readUint16(pos + 8)
|
||||||
|
hasLocalColorTable = (imageFlags and 0b10000000) != 0
|
||||||
|
interlaced = (imageFlags and 0b01000000) != 0
|
||||||
|
localColorTableSize = 2 ^ ((imageFlags and 0b00000111) + 1)
|
||||||
|
|
||||||
hasColorTable = (flags and 0x80) != 0
|
pos += 9
|
||||||
interlace = (flags and 0x40) != 0
|
|
||||||
colorTableSorted = (flags and 0x8) != 0
|
|
||||||
colorTableSize = 2 ^ ((flags and 0x7) + 1)
|
|
||||||
|
|
||||||
i += 9
|
if pos + localColorTableSize * 3 > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
# Make sure we support the GIF features.
|
var localColorTable: seq[ColorRGBX]
|
||||||
if left != 0 or top != 0 or w != result.width or h != result.height:
|
if hasLocalColorTable:
|
||||||
raise newException(PixieError, "GIF block offsets not supported")
|
localColorTable.setLen(localColorTableSize)
|
||||||
|
for i in 0 ..< localColorTable.len:
|
||||||
|
localColorTable[i] = rgbx(
|
||||||
|
data.readUint8(pos + 0),
|
||||||
|
data.readUint8(pos + 1),
|
||||||
|
data.readUint8(pos + 2),
|
||||||
|
255
|
||||||
|
)
|
||||||
|
pos += 3
|
||||||
|
|
||||||
if hasColorTable:
|
if pos + 1 > data.len:
|
||||||
raise newException(
|
failInvalid()
|
||||||
PixieError, "GIF color table per block not yet supported"
|
|
||||||
)
|
|
||||||
|
|
||||||
if interlace:
|
let minCodeSize = data.readUint8(pos).int
|
||||||
raise newException(PixieError, "Interlaced GIF not yet supported")
|
inc pos
|
||||||
|
|
||||||
# Read the lzw data chunks.
|
if minCodeSize > 11:
|
||||||
if i >= data.len: failInvalid()
|
failInvalid()
|
||||||
let lzwMinBitSize = data.readUint8(i)
|
|
||||||
i += 1
|
# The image data is contained in a sequence of sub-blocks
|
||||||
var lzwData = ""
|
var lzwDataBlocks: seq[(int, int)] # (offset, len)
|
||||||
while true:
|
while true:
|
||||||
if i >= data.len: failInvalid()
|
if pos + 1 > data.len:
|
||||||
let lzwEncodedLen = data.readUint8(i)
|
failInvalid()
|
||||||
i += 1
|
|
||||||
if lzwEncodedLen == 0:
|
let subBlockSize = data.readUint8(pos).int
|
||||||
# Stop reading when chunk len is 0.
|
inc pos
|
||||||
|
|
||||||
|
if subBlockSize == 0:
|
||||||
break
|
break
|
||||||
if i + lzwEncodedLen.int > data.len: failInvalid()
|
|
||||||
lzwData.add data[i ..< i + lzwEncodedLen.int]
|
if pos + subBlockSize > data.len:
|
||||||
i += lzwEncodedLen.int
|
failInvalid()
|
||||||
|
|
||||||
|
lzwDataBlocks.add((pos, subBlockSize))
|
||||||
|
|
||||||
|
pos += subBlockSize
|
||||||
|
|
||||||
|
var lzwDataLen: int
|
||||||
|
for (_, len) in lzwDataBlocks:
|
||||||
|
lzwDataLen += len
|
||||||
|
|
||||||
|
var
|
||||||
|
lzwData = newString(lzwDataLen)
|
||||||
|
i: int
|
||||||
|
for (offset, len) in lzwDataBlocks:
|
||||||
|
copyMem(lzwData[i].addr, data[offset].unsafeAddr, len)
|
||||||
|
i += len
|
||||||
|
|
||||||
let
|
let
|
||||||
clearCode = 1 shl lzwMinBitSize
|
clearCode = 1 shl minCodeSize
|
||||||
endCode = clearCode + 1
|
endCode = clearCode + 1
|
||||||
|
|
||||||
# Turn full lzw data into bit stream.
|
|
||||||
var
|
var
|
||||||
bs = BitStreamReader(
|
b = BitStreamReader(
|
||||||
src: cast[ptr UncheckedArray[uint8]](lzwData[0].addr),
|
src: cast[ptr UncheckedArray[uint8]](lzwData.cstring),
|
||||||
len: lzwData.len
|
len: lzwData.len
|
||||||
)
|
)
|
||||||
bitSize = lzwMinBitSize + 1
|
|
||||||
currentCodeTableMax = (1 shl (bitSize)) - 1
|
|
||||||
codeLast: int = -1
|
|
||||||
codeTable: seq[seq[int]]
|
|
||||||
colorIndexes: seq[int]
|
colorIndexes: seq[int]
|
||||||
|
codeSize = minCodeSize + 1
|
||||||
|
table = newSeq[(int, int)](endCode + 1)
|
||||||
|
prev: (int, int)
|
||||||
|
|
||||||
# Main decode loop.
|
while true:
|
||||||
while codeLast != endCode:
|
let code = b.readBits(codeSize).int
|
||||||
if bs.pos + bitSize.int > bs.len * 8: failInvalid()
|
if b.bitsBuffered < 0:
|
||||||
var
|
failInvalid()
|
||||||
# Read variable bits out of the table.
|
if code == endCode:
|
||||||
codeId = bs.readBits(bitSize.int).int
|
|
||||||
# Some time we need to carry over table information.
|
|
||||||
carryOver: seq[int]
|
|
||||||
|
|
||||||
if codeId == clearCode:
|
|
||||||
# Clear and re-init the tables.
|
|
||||||
bitSize = lzwMinBitSize + 1
|
|
||||||
currentCodeTableMax = (1 shl (bitSize)) - 1
|
|
||||||
codeLast = -1
|
|
||||||
codeTable.setLen(0)
|
|
||||||
for x in 0 ..< endCode + 1:
|
|
||||||
codeTable.add(@[x])
|
|
||||||
|
|
||||||
elif codeId == endCode:
|
|
||||||
# Exit we are done.
|
|
||||||
break
|
break
|
||||||
|
|
||||||
elif codeId < codeTable.len and codeTable[codeId].len > 0:
|
if code == clearCode:
|
||||||
# Its in the current table, use it.
|
codeSize = minCodeSize + 1
|
||||||
let current = codeTable[codeId]
|
table.setLen(endCode + 1)
|
||||||
colorIndexes.add(current)
|
prev = (0, 0)
|
||||||
carryOver = @[current[0]]
|
continue
|
||||||
|
|
||||||
elif codeLast notin [-1, clearCode, endCode]:
|
# Increase the code size if needed
|
||||||
# Its in the current table use it.
|
if table.len == (1 shl codeSize) - 1 and codeSize < 12:
|
||||||
if codeLast >= codeTable.len: failInvalid()
|
inc codeSize
|
||||||
var previous = codeTable[codeLast]
|
|
||||||
carryOver = @[previous[0]]
|
|
||||||
colorIndexes.add(previous & carryOver)
|
|
||||||
|
|
||||||
if codeTable.len == currentCodeTableMax and bitSize < 12:
|
let start = colorIndexes.len
|
||||||
# We need to expand the codeTable max and the bit size.
|
if code < table.len: # If we have seen the code before
|
||||||
inc bitSize
|
if code < clearCode:
|
||||||
currentCodeTableMax = (1 shl (bitSize)) - 1
|
colorIndexes.add(code)
|
||||||
|
if prev[1] > 0:
|
||||||
|
table.add((prev[0], prev[1] + 1))
|
||||||
|
prev = (start, 1)
|
||||||
|
else:
|
||||||
|
let (offset, len) = table[code]
|
||||||
|
for i in 0 ..< len:
|
||||||
|
colorIndexes.add(colorIndexes[offset + i])
|
||||||
|
table.add((prev[0], prev[1] + 1))
|
||||||
|
prev = (start, len)
|
||||||
|
else:
|
||||||
|
if prev[1] == 0:
|
||||||
|
failInvalid()
|
||||||
|
for i in 0 ..< prev[1]:
|
||||||
|
colorIndexes.add(colorIndexes[prev[0] + i])
|
||||||
|
colorIndexes.add(colorIndexes[prev[0]])
|
||||||
|
table.add((start, prev[1] + 1))
|
||||||
|
prev = (start, prev[1] + 1)
|
||||||
|
|
||||||
if codeLast notin [-1, clearCode, endCode]:
|
if colorIndexes.len != imageWidth * imageHeight:
|
||||||
# We had some left over and need to expand table.
|
failInvalid()
|
||||||
if codeLast >= codeTable.len: failInvalid()
|
|
||||||
codeTable.add(codeTable[codeLast] & carryOver)
|
|
||||||
|
|
||||||
codeLast = codeId
|
let image = newImage(imageWidth, imageHeight)
|
||||||
|
|
||||||
# Convert color indexes into real colors.
|
var transparentColorIndex = -1
|
||||||
for j, idx in colorIndexes:
|
if (controlExtension.fields and 1) != 0: # Transparent index flag
|
||||||
if idx >= colors.len or j >= result.data.len: failInvalid()
|
transparentColorIndex = controlExtension.transparentColorIndex.int
|
||||||
result.data[j] = colors[idx].rgbx()
|
|
||||||
|
let disposalMethod = (controlExtension.fields and 0b00011100) shr 2
|
||||||
|
if disposalMethod == 2:
|
||||||
|
let frame = newImage(screenWidth, screenHeight)
|
||||||
|
frame.fill(bgColor)
|
||||||
|
result.frames.add(frame)
|
||||||
|
else:
|
||||||
|
if hasLocalColorTable:
|
||||||
|
for i, colorIndex in colorIndexes:
|
||||||
|
if colorIndex >= localColorTable.len:
|
||||||
|
# failInvalid()
|
||||||
|
continue
|
||||||
|
if colorIndex != transparentColorIndex:
|
||||||
|
image.data[i] = localColorTable[colorIndex]
|
||||||
|
else:
|
||||||
|
for i, colorIndex in colorIndexes:
|
||||||
|
if colorIndex >= globalColorTable.len:
|
||||||
|
# failInvalid()
|
||||||
|
continue
|
||||||
|
if colorIndex != transparentColorIndex:
|
||||||
|
image.data[i] = globalColorTable[colorIndex]
|
||||||
|
|
||||||
|
if interlaced:
|
||||||
|
let deinterlaced = newImage(image.width, image.height)
|
||||||
|
var
|
||||||
|
y: int
|
||||||
|
i: int
|
||||||
|
while i < image.height:
|
||||||
|
copyMem(
|
||||||
|
deinterlaced.data[deinterlaced.dataIndex(0, i)].addr,
|
||||||
|
image.data[image.dataIndex(0, y)].addr,
|
||||||
|
image.width * 4
|
||||||
|
)
|
||||||
|
i += 8
|
||||||
|
inc y
|
||||||
|
i = 4
|
||||||
|
while i < image.height:
|
||||||
|
copyMem(
|
||||||
|
deinterlaced.data[deinterlaced.dataIndex(0, i)].addr,
|
||||||
|
image.data[image.dataIndex(0, y)].addr,
|
||||||
|
image.width * 4
|
||||||
|
)
|
||||||
|
i += 8
|
||||||
|
inc y
|
||||||
|
i = 2
|
||||||
|
while i < image.height:
|
||||||
|
copyMem(
|
||||||
|
deinterlaced.data[deinterlaced.dataIndex(0, i)].addr,
|
||||||
|
image.data[image.dataIndex(0, y)].addr,
|
||||||
|
image.width * 4
|
||||||
|
)
|
||||||
|
i += 4
|
||||||
|
inc y
|
||||||
|
i = 1
|
||||||
|
while i < image.height:
|
||||||
|
copyMem(
|
||||||
|
deinterlaced.data[deinterlaced.dataIndex(0, i)].addr,
|
||||||
|
image.data[image.dataIndex(0, y)].addr,
|
||||||
|
image.width * 4
|
||||||
|
)
|
||||||
|
i += 2
|
||||||
|
inc y
|
||||||
|
|
||||||
|
image.data = move deinterlaced.data
|
||||||
|
|
||||||
|
if imageWidth != screenWidth or imageHeight != screenHeight or
|
||||||
|
imageTopPos != 0 or imageLeftPos != 0:
|
||||||
|
let frame = newImage(screenWidth, screenHeight)
|
||||||
|
frame.draw(
|
||||||
|
image,
|
||||||
|
translate(vec2(imageLeftPos.float32, imageTopPos.float32))
|
||||||
|
)
|
||||||
|
result.frames.add(frame)
|
||||||
|
else:
|
||||||
|
result.frames.add(image)
|
||||||
|
|
||||||
|
result.intervals.add(controlExtension.delayTime.float32 / 100)
|
||||||
|
|
||||||
|
# Reset the control extension since it only applies to one image
|
||||||
|
controlExtension = ControlExtension()
|
||||||
|
|
||||||
|
of 0x21: # Extension
|
||||||
|
if pos + 1 > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
let extensionType = data.readUint8(pos + 0)
|
||||||
|
inc pos
|
||||||
|
|
||||||
|
case extensionType:
|
||||||
|
of 0xf9:
|
||||||
|
# Graphic Control Extension
|
||||||
|
if pos + 1 > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
let blockSize = data.readUint8(pos).int
|
||||||
|
inc pos
|
||||||
|
|
||||||
|
if blockSize != 4:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
if pos + blockSize > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
controlExtension.fields = data.readUint8(pos + 0)
|
||||||
|
controlExtension.delayTime = data.readUint16(pos + 1)
|
||||||
|
controlExtension.transparentColorIndex = data.readUint8(pos + 3)
|
||||||
|
|
||||||
|
pos += blockSize
|
||||||
|
inc pos # Block terminator
|
||||||
|
|
||||||
|
of 0xfe:
|
||||||
|
# Comment
|
||||||
|
skipSubBlocks()
|
||||||
|
|
||||||
|
# of 0x01:
|
||||||
|
# # Plain Text
|
||||||
|
|
||||||
|
of 0xff:
|
||||||
|
# Application Specific
|
||||||
|
if pos + 1 > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
let blockSize = data.readUint8(pos).int
|
||||||
|
inc pos
|
||||||
|
|
||||||
|
if blockSize != 11:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
if pos + blockSize > data.len:
|
||||||
|
failInvalid()
|
||||||
|
|
||||||
|
pos += blockSize
|
||||||
|
|
||||||
|
skipSubBlocks()
|
||||||
|
|
||||||
|
else:
|
||||||
|
raise newException(
|
||||||
|
PixieError,
|
||||||
|
"Unexpected GIF extension type " & toHex(extensionType)
|
||||||
|
)
|
||||||
|
|
||||||
|
of 0x3b: # Trailer
|
||||||
|
break
|
||||||
|
|
||||||
of 0x21: # Read EXTENSION block.
|
|
||||||
# Skip over all extensions (mostly animation information).
|
|
||||||
let extentionType = data.readUint8(i)
|
|
||||||
inc i
|
|
||||||
let byteLen = data.readUint8(i)
|
|
||||||
inc i
|
|
||||||
i += byteLen.int
|
|
||||||
doAssert data.readUint8(i) == 0
|
|
||||||
inc i
|
|
||||||
of 0x3b: # Read TERMINAL block.
|
|
||||||
# Exit block byte - we are done.
|
|
||||||
return
|
|
||||||
else:
|
else:
|
||||||
raise newException(PixieError, "Invalid GIF block type")
|
raise newException(
|
||||||
|
PixieError,
|
||||||
|
"Unexpected GIF block type " & toHex(blockType)
|
||||||
|
)
|
||||||
|
|
||||||
|
for interval in result.intervals:
|
||||||
|
result.duration += interval
|
||||||
|
|
||||||
proc decodeGifDimensions*(
|
proc decodeGifDimensions*(
|
||||||
data: string
|
data: string
|
||||||
|
@ -191,5 +386,8 @@ proc decodeGifDimensions*(
|
||||||
result.width = data.readInt16(6).int
|
result.width = data.readInt16(6).int
|
||||||
result.height = data.readInt16(8).int
|
result.height = data.readInt16(8).int
|
||||||
|
|
||||||
|
proc newImage*(gif: Gif): Image {.raises: [PixieError].} =
|
||||||
|
gif.frames[0].copy()
|
||||||
|
|
||||||
when defined(release):
|
when defined(release):
|
||||||
{.pop.}
|
{.pop.}
|
||||||
|
|
BIN
tests/fileformats/gif/newtons_cradle.gif
Normal file
BIN
tests/fileformats/gif/newtons_cradle.gif
Normal file
Binary file not shown.
After ![]() (image error) Size: 301 KiB |
|
@ -14,14 +14,14 @@ for i in 0 ..< 10_000:
|
||||||
data[pos] = value
|
data[pos] = value
|
||||||
echo &"{i} {pos} {value}"
|
echo &"{i} {pos} {value}"
|
||||||
try:
|
try:
|
||||||
let img = decodeGif(data)
|
let img = newImage(decodeGif(data))
|
||||||
doAssert img.height > 0 and img.width > 0
|
doAssert img.height > 0 and img.width > 0
|
||||||
except PixieError:
|
except PixieError:
|
||||||
discard
|
discard
|
||||||
|
|
||||||
data = data[0 ..< pos]
|
data = data[0 ..< pos]
|
||||||
try:
|
try:
|
||||||
let img = decodeGif(data)
|
let img = newImage(decodeGif(data))
|
||||||
doAssert img.height > 0 and img.width > 0
|
doAssert img.height > 0 and img.width > 0
|
||||||
except PixieError:
|
except PixieError:
|
||||||
discard
|
discard
|
||||||
|
|
|
@ -3,7 +3,7 @@ import pixie, pixie/fileformats/gif
|
||||||
block:
|
block:
|
||||||
let
|
let
|
||||||
path = "tests/fileformats/gif/3x5.gif"
|
path = "tests/fileformats/gif/3x5.gif"
|
||||||
image = decodeGIF(readFile(path))
|
image = readImage(path)
|
||||||
dimensions = decodeGifDimensions(readFile(path))
|
dimensions = decodeGifDimensions(readFile(path))
|
||||||
image.writeFile("tests/fileformats/gif/3x5.png")
|
image.writeFile("tests/fileformats/gif/3x5.png")
|
||||||
doAssert image.width == dimensions.width
|
doAssert image.width == dimensions.width
|
||||||
|
@ -12,7 +12,7 @@ block:
|
||||||
block:
|
block:
|
||||||
let
|
let
|
||||||
path = "tests/fileformats/gif/audrey.gif"
|
path = "tests/fileformats/gif/audrey.gif"
|
||||||
image = decodeGIF(readFile(path))
|
image = readImage(path)
|
||||||
dimensions = decodeGifDimensions(readFile(path))
|
dimensions = decodeGifDimensions(readFile(path))
|
||||||
image.writeFile("tests/fileformats/gif/audrey.png")
|
image.writeFile("tests/fileformats/gif/audrey.png")
|
||||||
doAssert image.width == dimensions.width
|
doAssert image.width == dimensions.width
|
||||||
|
@ -21,7 +21,7 @@ block:
|
||||||
block:
|
block:
|
||||||
let
|
let
|
||||||
path = "tests/fileformats/gif/sunflower.gif"
|
path = "tests/fileformats/gif/sunflower.gif"
|
||||||
image = decodeGIF(readFile(path))
|
image = readImage(path)
|
||||||
dimensions = decodeGifDimensions(readFile(path))
|
dimensions = decodeGifDimensions(readFile(path))
|
||||||
image.writeFile("tests/fileformats/gif/sunflower.png")
|
image.writeFile("tests/fileformats/gif/sunflower.png")
|
||||||
doAssert image.width == dimensions.width
|
doAssert image.width == dimensions.width
|
||||||
|
@ -30,8 +30,17 @@ block:
|
||||||
block:
|
block:
|
||||||
let
|
let
|
||||||
path = "tests/fileformats/gif/sunflower.gif"
|
path = "tests/fileformats/gif/sunflower.gif"
|
||||||
image = decodeGIF(readFile(path))
|
image = readImage(path)
|
||||||
dimensions = decodeGifDimensions(readFile(path))
|
dimensions = decodeGifDimensions(readFile(path))
|
||||||
image.writeFile("tests/fileformats/gif/sunflower.png")
|
image.writeFile("tests/fileformats/gif/sunflower.png")
|
||||||
doAssert image.width == dimensions.width
|
doAssert image.width == dimensions.width
|
||||||
doAssert image.height == dimensions.height
|
doAssert image.height == dimensions.height
|
||||||
|
|
||||||
|
block:
|
||||||
|
let img4 = readImage("tests/fileformats/gif/newtons_cradle.gif")
|
||||||
|
img4.writeFile("tests/fileformats/gif/newtons_cradle.png")
|
||||||
|
|
||||||
|
let animatedGif =
|
||||||
|
decodeGif(readFile("tests/fileformats/gif/newtons_cradle.gif"))
|
||||||
|
doAssert animatedGif.frames.len == 36
|
||||||
|
doAssert animatedGif.intervals.len == animatedGif.frames.len
|
||||||
|
|
Loading…
Reference in a new issue