ani2spritesheet.py 3.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  1. from PIL import Image
  2. import io,os,sys
  3. import logging
  4. logging.basicConfig(format='%(asctime)s - %(pathname)s[line:%(lineno)d] - %(levelname)s: %(message)s',level=logging.INFO)
  5. def analyzeANIFile(filePath):
  6. with open(filePath,'rb') as f:
  7. if f.read(4) != b'RIFF':
  8. return {"code":-1,"msg":"File is not a ANI File!"}
  9. logging.debug('文件头检查完成!')
  10. fileSize = int.from_bytes(f.read(4), byteorder='little', signed=False)
  11. # if os.path.getsize(filePath) != fileSize:
  12. # return {"code":-2,"msg":"File is damaged!"}
  13. logging.debug('文件长度检查完成!')
  14. if f.read(4) != b'ACON':
  15. return {"code":-1,"msg":"File is not a ANI File!"}
  16. logging.debug('魔数检查完成!')
  17. frameRate = (1/60)*1000
  18. while(True):
  19. chunkName = f.read(4)
  20. if chunkName == b'LIST':
  21. break
  22. chunkSize = int.from_bytes(f.read(4), byteorder='little', signed=False)
  23. if chunkName.lower() == b'rate':
  24. logging.debug('发现自定义速率!')
  25. frameRate = frameRate * int.from_bytes(f.read(4), byteorder='little', signed=False)
  26. logging.warning('发现自定义速率!由于GIF限制,将取第一帧与第二帧的速率作为整体速率!')
  27. f.read(chunkSize - 4)
  28. else:
  29. logging.debug('发现自定义Chunk!')
  30. f.read(chunkSize)
  31. listChunkSize = int.from_bytes(f.read(4), byteorder='little', signed=False)
  32. if f.read(4) != b'fram':
  33. return {"code":-3,"msg":"File not a ANI File!(No Frames)"}
  34. logging.debug('frame头检查完成!')
  35. frameList = []
  36. nowSize = 4
  37. while(nowSize < listChunkSize):
  38. if f.read(4) != b'icon':
  39. return {"code":-4,"msg":"File not a ANI File!(Other Kind Frames)"}
  40. nowSize += 4
  41. subChunkSize = int.from_bytes(f.read(4), byteorder='little', signed=False)
  42. nowSize += 4
  43. frameList.append(f.read(subChunkSize))
  44. nowSize += subChunkSize
  45. return {"code":0,"msg":frameList,"frameRate":frameRate}
  46. def CURPaletteFix(image):
  47. # type: (Image.ImageFile.ImageFile) -> None
  48. if (image.mode == 'P'):
  49. palette = list(image.palette.getdata()[1])
  50. for i in range(4, len(palette), 4):
  51. if sum(palette[i:i + 3]) == 0:
  52. break
  53. palette[i + 3] = 255
  54. image.putpalette(palette, 'BGRA')
  55. image = image.convert('RGBA')
  56. if __name__ == '__main__':
  57. OUTPUT_SIZE = (48,48)
  58. if len(sys.argv) < 2:
  59. logging.fatal("Usage:python ani2spritesheet.py <inputFile> <outputFile,Option>")
  60. else:
  61. res = analyzeANIFile(sys.argv[1])
  62. GIFframes = []
  63. if res["code"] == 0:
  64. logging.info('ANI文件分析完成,帧提取完成!')
  65. output = Image.new("RGBA", (OUTPUT_SIZE[0], OUTPUT_SIZE[1] * len(res["msg"])))
  66. for frameIndex in range(len(res["msg"])):
  67. frameImage = Image.open(io.BytesIO(res["msg"][frameIndex]),formats=['cur'])
  68. CURPaletteFix(frameImage)
  69. extracted_frame = frameImage.resize(OUTPUT_SIZE)
  70. position = (0, OUTPUT_SIZE[0] * frameIndex)
  71. output.paste(extracted_frame, position)
  72. if(len(sys.argv) >= 3):
  73. output.save(sys.argv[2],format="PNG")
  74. else:
  75. output.save(f"{sys.argv[1].strip('.ani')}.png",format="PNG")
  76. logging.info('SpriteSheet生成完成!')
  77. else:
  78. logging.fatal(res["msg"])