#image #file #image-processing
#изображение #файл #обработка изображений
Вопрос:
Как я могу преобразовать изображения .cpbitmap в .png или обычный тип изображений?
Спасибо 🙂
Комментарии:
1. Вот некоторый код из CodeProject: codeproject.com/KB/recipes/Apple_CPBitmap_Reader.aspx
2. @MarkRansom я видел это до того, как спросил здесь … это не для Mac OS X, это для Windows, а у меня нет Windows, у меня Mac…
3. Эта ссылка не просто предоставляет код, она предоставляет тривиально простое описание файла. Просто закодируйте его на любом доступном вам языке. Или, если у вас есть утилита, которая может конвертировать изображения фиксированного размера без сжатия и без заголовка, используйте это.
4. @MarkRansom Спасибо, я попробую это; Я думал, что смогу найти приложение для преобразования .cpbitmap в .png…
5. Рад, что моя статья каким-то образом помогла! 🙂
Ответ №1:
На самом деле, идея написать код на python превосходна, поскольку ее легче выполнить, чем запускать некоторые вещи xcode.
Как заявил предыдущий автор, он не тестировал код, но я это сделал. Я обнаружил, что он создает изображение, в котором КРАСНЫЕ и СИНИЕ компоненты неуместны.
Вот почему я решил опубликовать правильную версию этого кода здесь:
#!/usr/bin/python
from PIL import Image,ImageOps
import struct
import sys
if len(sys.argv) < 3:
print "Need two args: filename and result_filenamen";
sys.exit(0)
filename = sys.argv[1]
result_filename = sys.argv[2]
with open(filename) as f:
contents = f.read()
unk1, width, height, unk2, unk3, unk4 = struct.unpack('<6i', contents[-24:])
im = Image.fromstring('RGBA', (width,height), contents, 'raw', 'RGBA', 0, 1)
r,g,b,a = im.split()
im = Image.merge('RGBA', (b,g,r,a))
im.save(result_filename)
Поместите этот код в файл decode_cpbitmap, выполните
chmod 755 decode_cpbitmap
чтобы сделать его исполняемым, и теперь вы можете вызвать его следующим образом:
./decode_cpbitmap input_filename output_filename
где input_filename — это файл «* .cpbitmap», который у вас уже есть и который вы хотите декодировать, а output_filename — это smth.png (он будет создан с помощью этого кода).
Вы можете получить сообщение об ошибке
ImportError: No module named PIL
Затем вам необходимо установить модуль PIL python. Я не буду объяснять, как устанавливать модули python, потому что вы можете найти это в другом месте.
Ответ №2:
Вот быстрая программа на Python для этого. Я не смог его протестировать, потому что у меня нет никаких изображений .cpbitmap для использования.
from PIL import Image
import struct
with open(filename) as f:
contents = f.read()
unk1, width, height, unk2, unk3, unk4 = struct.unpack('<6i', contents[-24:])
im = Image.fromstring('RGBA', (width,height), contents, 'raw', 'RGBA', 0, 1)
im.save('converted.png')
Комментарии:
1. Я поместил приведенный выше код в файл и назвал его «convert_cpbitmap.py » и запустил его, и я получил эту ошибку «Traceback (последний последний вызов): File «/Users/ …/ Desktop/ без названия folder/convert_cpbitmap.py «, строка 1, в <module> из PIL import Image ImportError: нет модуляименованный файл »
2. @user1028111, PIL — это необязательный компонент, который необходимо загрузить и установить. pythonware.com/products/pil
3. Я весь день безуспешно пытался установить его на mac os x lion :
4. Я преобразовал его, но цвета неправильные, как это исправить?
5. @user1028111, если вы можете опубликовать свое исходное изображение где-нибудь, я могу его диагностировать. В противном случае попробуйте изменить один из ‘RGBA’ на ‘BGRA’.
Ответ №3:
Я пытался конвертировать изображения из iOS 11, и эти скрипты не работают. Сегодня размер каждой строки округляется до числа в 8 пикселей путем заполнения.
Я написал узел.JS-скрипт. Перед запуском скрипта установите модуль jimp ( npm install jimp
). Проверено на Node.JS v9.2.0 и jimp 0.2.28.
const fs = require('fs')
const util = require('util')
const Jimp = require('jimp')
const main = async () => {
if (process.argv.length != 4) {
console.log('Need two args: input filename and result filename')
console.log(`Example: ${process.argv[0]} ${process.argv[1]} HomeBackground.cpbitmap HomeBackground.png`)
return
}
const inpFileName = process.argv[2]
const outFileName = process.argv[3]
const readFile = util.promisify(fs.readFile)
const cpbmp = await readFile(inpFileName)
const width = cpbmp.readInt32LE(cpbmp.length - 4 * 5)
const height = cpbmp.readInt32LE(cpbmp.length - 4 * 4)
console.log(`Image height: ${height}, width: ${width}`)
const image = await new Jimp(width, height, 0x000000FF)
const calcOffsetInCpbmp = (x, y, width) => {
const lineSize = Math.ceil(width / 8) * 8
return x * 4 y * lineSize * 4
}
const calcOffsetInImage = (x, y, width) => {
return x * 4 y * width * 4
}
const swapRBColors = (c) => {
const r = c amp; 0xFF
const b = (c amp; 0xFF0000) >> 16
c amp;= 0xFF00FF00
c |= r << 16
c |= b
return c
}
for (let y = 0; y < height; y ) {
for (let x = 0; x < width; x ) {
const color = cpbmp.readInt32LE(calcOffsetInCpbmp(x, y, width))
image.bitmap.data.writeInt32LE(swapRBColors(color), calcOffsetInImage(x, y, width))
}
}
await image.write(outFileName)
console.log('Done')
}
main()
Комментарии:
1. Работает отлично. Спасибо!
2. Я только что попробовал это, и изображения отображаются некорректно. Должно быть что-то, чего я не понимаю. Изображения выглядят следующим образом: imgur.com/Oj4TQDV . Есть идеи?
3. Неважно, оказывается, смещение было увеличено до 16, поэтому я просто изменил 8 в константе lineSize на 16, и все работает отлично!
4. Спасибо @dr15, я опубликовал репозиторий GitHub и указал ваш ответ в качестве источника. — github.com/hthetiot/cpbitmap-to-png/blob/main/README.md У моего отца был старый iphone 5 на iOS 10, и он хотел получить изображение экрана блокировки и домашнего экрана, но его не было на его фотографии, поэтому я смог удалить его из резервной копии и преобразовать с помощью вашего решения. Спасибо!