设为首页收藏本站|繁體中文

Project1

 找回密码
 注册会员
搜索
查看: 8053|回复: 6
打印 上一主题 下一主题

[通用发布] 保存Bitmap为PNG文件(快速+无DLL)

[复制链接]

Lv1.梦旅人

梦石
0
星屑
60
在线时间
156 小时
注册时间
2011-12-13
帖子
133
跳转到指定楼层
1
发表于 2012-3-4 13:06:46 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

加入我们,或者,欢迎回来。

您需要 登录 才可以下载或查看,没有帐号?注册会员

x
本帖最后由 dant 于 2012-3-4 13:40 编辑

虽然没有用DLL,但是BGRA->RGBA的处理还是用了一种很龌龊的方法
RUBY 代码复制
  1. #encoding:utf-8
  2. #==============================================================================
  3. # ■ Bitmap
  4. #------------------------------------------------------------------------------
  5. #  位图的类。所谓位图即表示图像其本身。
  6. #==============================================================================
  7.  
  8. class Bitmap
  9.   #--------------------------------------------------------------------------
  10.   # ● 传送到内存的API函数
  11.   #--------------------------------------------------------------------------
  12.   RtlMoveMemory_pi = Win32API.new('kernel32', 'RtlMoveMemory', 'pii', 'i')
  13.   #--------------------------------------------------------------------------
  14.   # ● Bitmap地址
  15.   #--------------------------------------------------------------------------
  16.   # [[[bitmap.object_id * 2 + 16] + 8] + 16] == 数据的开头
  17.   def address
  18.     buffer, ad = "rgba", object_id * 2 + 16
  19.     RtlMoveMemory_pi.call(buffer, ad, 4)
  20.     ad = buffer.unpack("L")[0] + 8
  21.     RtlMoveMemory_pi.call(buffer, ad, 4)
  22.     ad = buffer.unpack("L")[0] + 16
  23.     RtlMoveMemory_pi.call(buffer, ad, 4)
  24.     return buffer.unpack("L")[0]
  25.   end
  26.  
  27.   module Bitmap2PNG
  28.     module_function
  29.     Malloc = Win32API.new('msvcrt.dll','malloc','i','i')
  30.     Memcpy_pi = Win32API.new('kernel32.dll','RtlMoveMemory','pii','v')
  31.     Memcpy_ii = Win32API.new('kernel32.dll','RtlMoveMemory','iii','v')
  32.     Memcpy_ip = Win32API.new('kernel32.dll','RtlMoveMemory','ipi','v')
  33.     Free = Win32API.new('msvcrt.dll','free','i','v')
  34.     Callsub = Win32API.new('user32.dll','CallWindowProcW','iiiii','i')
  35.     #--------------------------------------------------------------------------
  36.     # ● 主处理
  37.     #--------------------------------------------------------------------------
  38.     def make_png(bitmap_Fx)
  39.       @bitmap_Fx = bitmap_Fx
  40.       return make_header + make_ihdr + make_idat + make_iend
  41.     end
  42.     #--------------------------------------------------------------------------
  43.     # ● PNG文件头数据块
  44.     #--------------------------------------------------------------------------
  45.     def make_header
  46.       return [0x89,0x50,0x4e,0x47,0x0d,0x0a,0x1a,0x0a].pack("C*")
  47.     end
  48.     #--------------------------------------------------------------------------
  49.     # ● PNG文件情报头数据块(IHDR)
  50.     #--------------------------------------------------------------------------
  51.     def make_ihdr
  52.       ih_size = [13].pack("N")
  53.       ih_sign = "IHDR"
  54.       ih_width = [@bitmap_Fx.width].pack("N")
  55.       ih_height = [@bitmap_Fx.height].pack("N")
  56.       ih_bit_depth = [8].pack("C")
  57.       ih_color_type = [6].pack("C")
  58.       ih_compression_method = [0].pack("C")
  59.       ih_filter_method = [0].pack("C")
  60.       ih_interlace_method = [0].pack("C")
  61.       string = ih_sign + ih_width + ih_height + ih_bit_depth + ih_color_type +
  62.       ih_compression_method + ih_filter_method + ih_interlace_method
  63.       ih_crc = [Zlib.crc32(string)].pack("N")
  64.       return ih_size + string + ih_crc
  65.     end
  66.     #--------------------------------------------------------------------------
  67.     # ● 生成图像数据(IDAT)
  68.     #--------------------------------------------------------------------------
  69.     def make_idat
  70.       header = "IDAT"
  71.       data = make_bitmap_data_dant
  72.       data = Zlib::Deflate.deflate(data)
  73.       crc = [Zlib.crc32(header + data)].pack("N")
  74.       size = [data.length].pack("N")
  75.       return size + header + data + crc
  76.     end
  77.     #--------------------------------------------------------------------------
  78.     # ● 从Bitmap对象的原始数据生成PNG图像数据
  79.     #--------------------------------------------------------------------------
  80.     def make_bitmap_data_dant
  81.       len = @bitmap_Fx.width * @bitmap_Fx.height * 4
  82.       pBuf = Malloc.call(len)
  83.       bitmap=@bitmap_Fx#.reverse
  84.       addr=bitmap.address
  85.       Memcpy_ii.call(pBuf,addr,len)
  86.       Callsub.call(@pCode,pBuf,bitmap.width*bitmap.height,0,0)
  87.       len2=len+bitmap.height
  88.       len3=bitmap.width*4
  89.       pBuf2 = Malloc.call(len2+512)
  90.       pos=pBuf2
  91.       pos2=pBuf+len-len3
  92.       Memcpy_ip.call(pos,0.chr,1)
  93.       pos+=1
  94.       for i in 0...bitmap.height
  95.         Memcpy_ii.call(pos,pos2,len3)
  96.         pos+=len3
  97.         pos2-=len3
  98.         Memcpy_ip.call(pos,0.chr,1)
  99.         pos+=1
  100.       end
  101.       buf="\0"*len2
  102.       Memcpy_pi.call(buf,pBuf2,len2)
  103.       Free.call(pBuf)
  104.       Free.call(pBuf2)
  105. #~       bitmap.dispose
  106.       return buf
  107.     end
  108.     #--------------------------------------------------------------------------
  109.     # ● PNG文件尾数据块(IEND)
  110.     #--------------------------------------------------------------------------
  111.     def make_iend
  112.       return [0x00,0x00,0x00,0x00,0x49,0x45,0x4E,0x44,0xAE,0x42,0x60,0x82].pack("C*")
  113.       #ie_size = [0].pack("N")
  114.       #ie_sign = "IEND"
  115.       #ie_crc = [Zlib.crc32(ie_sign)].pack("N")
  116.       #return ie_size + ie_sign + ie_crc
  117.     end
  118.  
  119.     def init
  120.       @pCode=Malloc.call(512)
  121. =begin
  122.       qwe proc src,pixels,un,used
  123.       mov eax,src
  124.       xor ecx,ecx
  125.       .while ecx < pixels
  126.       mov edx,[eax] ;AARRGGBB|BB GG RR AA
  127.       bswap edx ;BBGGRRAA|AA RR GG BB
  128.       ror edx,8 ;AABBGGRR|RR GG BB AA
  129.       mov [eax],edx
  130.       inc ecx
  131.       add eax,4
  132.       .endw
  133.       ret
  134.       qwe endp
  135. =end
  136.       code = [0x55,0x8B,0xEC,0x8B,0x45,0x8,0x33,0xC9,0xEB,0xD,0x8B,0x10,0xF,0xCA,0xC1,0xCA,0x8,0x89,0x10,0x41,0x83,0xC0,0x4,0x3B,0x4D,0xC,0x72,0xEE,0xC9,0xC2,0x10,0x0].pack('C*')
  137.       Memcpy_ip.call(@pCode,code,code.length)
  138.  
  139.     end
  140.  
  141.     def close
  142.       Free.call(@pCode)
  143.     end
  144.  
  145.   end #Bitmap2PNG
  146.  
  147.   def save_png(filename)
  148.     File.open(filename,"wb"){|f|f.write(Bitmap2PNG.make_png(self))}
  149.   end
  150.  
  151. end #Bitmap
  152.  
  153.  
  154. Bitmap::Bitmap2PNG.init
  155. END {Bitmap::Bitmap2PNG.close}

范例工程(VX)
savePNG.zip(250 KB, 下载次数: undefined, 售价: 4294967295 经验, 备用地址1, 备用地址2)

评分

参与人数 3星屑 +1632 收起 理由
taroxd + 240 果然龌龊~
fux2 + 1332 触手
iisnow + 60 LZ厉害,并且非常感谢

查看全部评分

Lv3.寻梦者

宛若

梦石
0
星屑
1568
在线时间
526 小时
注册时间
2007-8-19
帖子
1493

极短24参与开拓者

2
发表于 2012-3-4 13:26:09 | 只看该作者
竟然用Dropbox……害得咱还要开VPN……
而且这奇葩的伪装……

点评

我这里用https://dl-web..............可以的 奇葩的天朝局域网  发表于 2012-3-4 13:48
[url=http://rpg.blue/thread-219730-1-1.html]http://unhero.sinaapp.com/wi.php[/url]
[color=Red]如你所见这是个死坑,没错这就是打我的脸用的[/color]
回复 支持 反对

使用道具 举报

Lv2.观梦者

梦石
0
星屑
508
在线时间
1478 小时
注册时间
2011-9-17
帖子
1316

开拓者贵宾

3
发表于 2012-3-4 13:29:20 | 只看该作者
本帖最后由 iisnow 于 2012-3-4 13:30 编辑

过来学习~~
顺便吐槽下,为什么附件下载不了;

然后LZ能不能推荐一下PNG相关的API的一些资料啊

点评

关于PNG的资料 http://www.w3.org/TR/PNG/ ms这里也有一些 http://www.libpng.org/pub/png/libpng.html  发表于 2012-3-4 13:44
加了两个地址  发表于 2012-3-4 13:41
我帖子中要有是不HX的空白,请Ctrl + A
回复 支持 反对

使用道具 举报

Lv6.析梦学徒

Fuzzy Ginkgo
Taciturn Knight

梦石
0
星屑
60819
在线时间
1934 小时
注册时间
2010-6-26
帖子
1605

烫烫烫开拓者

4
发表于 2012-3-4 13:46:13 | 只看该作者
惊险触手 这奇葩的附件伪装= =b
下次用Dropbox建议你建立一个专门的文件夹来放这些东西……
不要图方便免得以后不能整理。
我的言论只代表我个人的观点,不代表雇主及/或任何第三方的立场。
Opinions expressed are solely my own and do not express the views or opinions of my employer and/or any third parties.
捐赠 | GitHub
回复 支持 反对

使用道具 举报

头像被屏蔽

Lv2.观梦者 (禁止发言)

梦石
0
星屑
653
在线时间
3774 小时
注册时间
2011-2-26
帖子
1839

开拓者

5
发表于 2012-3-6 21:56:32 | 只看该作者
提示: 作者被禁止或删除 内容自动屏蔽
签名被屏蔽
回复 支持 反对

使用道具 举报

Lv1.梦旅人

梦石
0
星屑
60
在线时间
156 小时
注册时间
2011-12-13
帖子
133
6
 楼主| 发表于 2012-3-7 13:41:50 | 只看该作者
晴兰 发表于 2012-3-6 21:56
惊现触手
其实可以直接取code的地址 不需要RtlMoveMemory再复制一遍的 而且malloc用Win32API是有问题的  ...

直接执行String里的代码有时会出错,原因不明
malloc改成VirtualAlloc应该就没问题了

点评

???  发表于 2012-3-8 18:23
其实新人俺还有一个问题  发表于 2012-3-7 14:27
回复 支持 反对

使用道具 举报

Lv1.梦旅人

梦石
0
星屑
50
在线时间
66 小时
注册时间
2013-9-4
帖子
94
7
发表于 2013-12-17 18:49:51 | 只看该作者
怎么个龌龊法?(纯属新手,问下怎么做到的)
本人以后只点评!不回复!更不发主题!
(每次发资源都被扣分,还扣那么多!)
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 注册会员

本版积分规则

拿上你的纸笔,建造一个属于你的梦想世界,加入吧。
 注册会员
找回密码

站长信箱:[email protected]|手机版|小黑屋|无图版|Project1游戏制作

GMT+8, 2024-12-21 21:52

Powered by Discuz! X3.1

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表