2003-09-25 20:15:53 +00:00
|
|
|
#! /usr/bin/ruby -w
|
|
|
|
#
|
2004-02-24 17:21:53 +00:00
|
|
|
# Copyright (C) 2003 Free Software Foundation, Inc.
|
2003-09-25 20:15:53 +00:00
|
|
|
#
|
|
|
|
# This unifont2pff.rb is free software; the author
|
|
|
|
# gives unlimited permission to copy and/or distribute it,
|
|
|
|
# with or without modifications, as long as this notice is preserved.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY, to the extent permitted by law; without
|
|
|
|
# even the implied warranty of MERCHANTABILITY or FITNESS FOR A
|
|
|
|
# PARTICULAR PURPOSE.
|
|
|
|
|
|
|
|
# The magic number of the font file.
|
|
|
|
MAGIC = "PPF\x7f"
|
|
|
|
|
|
|
|
def usage(status = 0)
|
2006-03-31 13:32:52 +00:00
|
|
|
puts "Usage: ruby unifont2pff.rb [RANGE ...] FILE"
|
2003-09-25 20:15:53 +00:00
|
|
|
exit(status)
|
|
|
|
end
|
|
|
|
|
2006-03-31 13:32:52 +00:00
|
|
|
if ARGV.length == 0
|
|
|
|
usage(1)
|
|
|
|
end
|
|
|
|
|
2003-09-25 20:15:53 +00:00
|
|
|
file = ARGV.pop
|
|
|
|
|
|
|
|
ranges = []
|
|
|
|
ARGV.each do |range|
|
2006-03-31 13:32:52 +00:00
|
|
|
if /\A0x([0-9a-fA-F]+)[:-]0x([0-9a-fA-F]+)\z/ =~ range
|
2003-09-25 20:15:53 +00:00
|
|
|
ranges << [$1.hex, $2.hex]
|
2006-03-31 13:32:52 +00:00
|
|
|
elsif /\A0x([0-9a-fA-F]+)\z/ =~ range
|
2003-09-25 20:15:53 +00:00
|
|
|
ranges << [$1.hex, $1.hex]
|
2006-03-31 13:32:52 +00:00
|
|
|
elsif /\A([0-9]+)[:-]([0-9]+)\z/ =~ range
|
|
|
|
ranges << [$1.to_i, $2.to_i]
|
|
|
|
elsif /\A([0-9]+)\z/ =~ range
|
|
|
|
ranges << [$1.to_i, $1.to_i]
|
2003-09-25 20:15:53 +00:00
|
|
|
else
|
|
|
|
usage(1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def ranges.contain?(code)
|
|
|
|
if self.empty?
|
|
|
|
true
|
|
|
|
else
|
|
|
|
self.each do |r|
|
|
|
|
return true if r[0] <= code and r[1] >= code
|
|
|
|
end
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
fonts = []
|
|
|
|
IO.foreach(file) do |line|
|
|
|
|
if /^([0-9A-F]+):([0-9A-F]+)$/ =~ line
|
|
|
|
code = $1.hex
|
|
|
|
next unless ranges.contain?(code)
|
|
|
|
|
|
|
|
bitmap = $2
|
|
|
|
if bitmap.size != 32 and bitmap.size != 64
|
|
|
|
raise "invalid bitmap size: #{bitmap}"
|
|
|
|
end
|
|
|
|
|
2006-03-31 13:32:52 +00:00
|
|
|
# Fix byte ordering
|
|
|
|
w = (bitmap.size / 32)
|
|
|
|
temp = Array.new
|
|
|
|
for y in 0...16
|
|
|
|
for x in 0...w
|
|
|
|
temp[(y * w + x) * 2 + 0] = bitmap[(x * 16 + y) * 2 + 0].chr
|
|
|
|
temp[(y * w + x) * 2 + 1] = bitmap[(x * 16 + y) * 2 + 1].chr
|
|
|
|
end
|
|
|
|
end
|
|
|
|
bitmap = temp.to_s
|
|
|
|
|
2003-09-25 20:15:53 +00:00
|
|
|
fonts << [code, bitmap]
|
|
|
|
else
|
|
|
|
raise "invalid line format: #{line}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
fonts.sort! {|a,b| a[0] <=> b[0]}
|
|
|
|
|
|
|
|
# Output the result.
|
|
|
|
print MAGIC
|
|
|
|
print [fonts.size].pack('V')
|
|
|
|
|
|
|
|
offset = 8 + fonts.size * 8
|
|
|
|
fonts.each do |f|
|
|
|
|
print [f[0]].pack('V')
|
|
|
|
print [offset].pack('V')
|
|
|
|
offset += 4 + 16 * f[1].size / 32
|
|
|
|
end
|
|
|
|
|
|
|
|
fonts.each do |f|
|
|
|
|
print [f[1].size / 32].pack('V')
|
|
|
|
print f[1].scan(/../).collect {|a| a.hex}.pack('C*')
|
|
|
|
end
|