From 6c0e0261f95ccfa9d47d384261db319c1e6eaee4 Mon Sep 17 00:00:00 2001 From: John Bintz Date: Tue, 31 Mar 2009 21:00:19 -0400 Subject: [PATCH] initial commit --- README | 1 + minicomic-backend.rb | 610 +++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 611 insertions(+) create mode 100644 README create mode 100755 minicomic-backend.rb diff --git a/README b/README new file mode 100644 index 0000000..5eab12f --- /dev/null +++ b/README @@ -0,0 +1 @@ +minicomic-backend.rb builds Web- and print-ready comic images from SVG source files. diff --git a/minicomic-backend.rb b/minicomic-backend.rb new file mode 100755 index 0000000..1753af1 --- /dev/null +++ b/minicomic-backend.rb @@ -0,0 +1,610 @@ +#!/usr/bin/ruby + +require 'yaml' +require 'time' +require 'singleton' + +class Filter + include Singleton + @config = {} + @cleanup = [] + + attr_accessor :config, :cleanup + + def initialize + @config = {} + @cleanup = [] + end + + def cleanup + @cleanup.each do |f| + if File.exists? f; File.unlink(f); end + end + end + + def recalc_pixels + if @config['print'] + if @config['width_inches'] && @config['width_inches'] != 0 + @config['width'] = @config['width_inches'].to_i * @config['dpi'].to_i + end + if @config['height_inches'] && @config['height_inches'] != 0 + @config['height'] = @config['height_inches'].to_i * @config['dpi'].to_i + end + end + end + + # + # Set the config + # + def config=(c) + @config = c + + recalc_pixels + end + + # + # Build a temporary PNG from an SVG file + # + def inkscape(input, target) + params = [] + + width = @config['width'] + height = @config['height'] + inkscape_target = target + + if @config['rotate'] + case @config['rotate'] + when 90, -90 + t = width; width = height; height = t + inkscape_target = target + "-pre.png" + end + end + + if width; params << "-w #{width} "; end + if height; params << "-h #{height} "; end + + system("inkscape -e \"#{inkscape_target}\" -y 1.0 #{params.join(" ")} \"#{input}\"") + + if @config['rotate'] + command = [ + "\"#{inkscape_target}\"", + "-rotate #{@config['rotate']}", + "\"#{target}\"" + ] + + convert(command) + File.unlink(inkscape_target) + end + end + + def convert(command, verbose = false) + system("convert " + (verbose ? "-verbose " : "" ) + [ command ].flatten.join(" ")) + end + + # + # Get the dimensions of a file + # + def get_dimensions(input) + dimensions = nil + IO.popen("identify -format '%w,%h' \"#{input}\"") do |fh| + dimensions = fh.readlines.first.split(",").collect { |d| d.to_i } + end + dimensions + end +end + +class InputFilter < Filter + OutputFilename = "tmp.png" +end + +class OutputFilter < Filter + # + # get the output filename for this filter + # + def filename(info) + target = @config['target'] + info.each { |k,v| target = target.gsub("{#{k}}", v.to_s) } + target + end + + # + # get the output targets for this filter + # + def targets(info); filename(info); end +end + +module PrintHandling + # + # calculate the page size in PPI + # + def calculate_page_size + if @config['dpi'] + if @config['page_size'] + case @config['page_size'].downcase + when "letter", "letter_portrait" + page_width = 8.5; page_height = 11 + when "letter_landscape" + page_width = 11; page_height = 8.5 + when "half_letter_landscape" + page_width = 5.5; page_height = 8.5 + end + else + page_width = @config['page_width_inches'] + page_height = @config['page_height_inches'] + end + + page_width *= @config['dpi'] + page_height *= @config['dpi'] + else + page_width = @config['page_width'] + page_height = @config['page_height'] + end + + [ page_width, page_height ] + end + + # + # align the provided image on a white page + # + def build_for_print(input, output, side = "none") + page_width, page_height = calculate_page_size + command = [ + "-density #{config['dpi']}", + "-size #{page_width.to_i}x#{page_height.to_i}", + "xc:white" + ] + + case side + when "none" + command << "-gravity Center" + when "left" + command << "-gravity East" + when "right" + command << "-gravity West" + end + + command << "-draw 'image Over 0,0 0,0 \"#{input}\"'" + command << "\"#{output}\"" + + convert(command) + end +end + +class SVGToTempBitmap < InputFilter + include PrintHandling + + # + # select which build method to use based on the number of provided images. + # + def build(input) + (input.instance_of? Array) ? multiple(input) : single(input) + end + + # + # process a single input file, possibly splitting it into two output files if it's a spread + # + def single(input) + filename = Dir.pwd + '/' + OutputFilename + inkscape(input, filename) + @cleanup << filename + + if @config['spread'] + width, height = get_dimensions(filename) + targets = [] + + [ [ "left", 0 ], [ "right", width / 2 ] ].each do |side, offset| + target = filename + "-#{side}.png" + command = [ + "\"#{filename}\"", + "-gravity Northwest", + "-crop #{width/2}x#{height}+#{offset}+0", + "+repage", + "\"#{target}\"" + ] + + convert(command) + + targets << target + @cleanup << target + end + + return targets + end + + return filename + end + + # + # combine multiple input files onto a single page-sized canvas. + # images are added left-to-right and top-to-bottom starting from the top-left of the page. + # leave a grid square blank by passing the filename "/blank". + # + def multiple(files) + if @config['spread']; raise "Spreads and grids combined do not make sense"; end + + width, height = @config['grid'].split("x").collect { |f| f.to_f } + joined_files = [] + + page_width, page_height = calculate_page_size + + grid_width = page_width / width + grid_height = page_height / height + + 0.upto(files.length - 1) do |i| + x = i % width + y = (i / width).floor + + if files[i].split('/').last != "blank" + tmp_svg_output = OutputFilename + "-#{i}.png" + inkscape(Dir.pwd + '/' + files[i], tmp_svg_output) + + joined_files << [ tmp_svg_output, x, y ] + @cleanup << tmp_svg_output + end + end + + command = [ + "-size #{page_width}x#{page_height}", + "xc:white" + ] + + joined_files.each do |file, x, y| + image_width, image_height = get_dimensions(file) + + x_offset = (grid_width - image_width) / 2 + y_offset = (grid_height - image_height) / 2 + + command << "-draw 'image Over #{x * grid_width + x_offset},#{y * grid_height + y_offset} 0,0 \"#{file}\"'" + end + + command << OutputFilename + + convert(command) + + @cleanup << OutputFilename + OutputFilename + end +end + +# +# Process an input file for the Web +# +class TempBitmapToWeb < OutputFilter + def build(input, output) + quality = @config['quality'] ? @config['quality'] : 80 + convert("\"#{input}\" -quality #{quality} \"#{output}\"") + end + + def filename(info) + if !@config['start_date']; raise "Must define a start date!"; end + if !@config['period']; raise "Must define a period!"; end + + index = info['index'].to_i + if index == 0 && @config['announce_date'] + comic_date = @config['announce_date'] + else + day_of_week = 0 + weeks_from_start = 0 + + case @config['period'] + when "daily" + day_of_week = index % 5 + weeks_from_start = (index / 5).floor + when "weekly" + weeks_from_start = index + end + + if @config['delay_at_index'] && @config['delay_length_weeks'] + if index >= @config['delay_at_index'] + weeks_from_start += @config['delay_length_weeks'] + end + end + + comic_date = Time.parse(@config['start_date']) + ((((day_of_week + weeks_from_start * 7) * 24) + 6) * 60 * 60) + end + + info['date'] = comic_date.strftime("%Y-%m-%d") + super(info) + end +end + +# +# Code to help with pagination +# +module Pagination + def paginate(files) + if !files.instance_of? Array; raise "File list must be an array"; end + if files.length == 0; raise "File list cannot be empty"; end + if (files.length % 4) != 0; raise "File list must be divisible by 4"; end + + number_of_sheet_faces = (files.length / 4) * 2 + + sheet_faces = [] + + is_right = 1 + is_descending = 1 + sheet_face_index = 0 + + files.each do |file| + if !sheet_faces[sheet_face_index]; sheet_faces[sheet_face_index] = []; end + + sheet_faces[sheet_face_index][is_right] = file + is_right = 1 - is_right + + sheet_face_index += is_descending + + if sheet_face_index == number_of_sheet_faces + sheet_face_index -= 1 + is_descending = -1 + end + end + + tmp_pdf_files = [] + 0.upto(sheet_faces.length - 1) do |i| + f = @config['target'] + "-#{i}.pdf" + process_pagination(f, i, sheet_faces.length, *sheet_faces[i]) + tmp_pdf_files << f + end + + system("pdfjoin #{tmp_pdf_files.collect { |f| "\"#{f}\"" }.join(" ")} --outfile \"#{@config['target']}\"") + end +end + +# +# Convert a bitmap to a single page print for proofing +# +class TempBitmapToPrint < OutputFilter + include PrintHandling + def build(input, output) + build_for_print(input, "tmp2.pbm") + + system("sam2p -c:lzw -m:dpi:#{(72.0 * (72.0 / @config['dpi'].to_f))} tmp2.pbm PDF:\"#{output}\""); + system("rm tmp2.pbm") + end +end + +# +# Convert bitmap files to a paginated print-ready file +# +class TempBitmapToPaginatedPrint < OutputFilter + include PrintHandling, Pagination + def build(input, output, side = "none") + build_for_print(input, output, side) + end + + def targets(info) + (@config['spread'] == true) ? [ filename(info) + "-left.png", "tmp-right.png" ] : filename(info) + end + + def process_pagination(output, face, total_faces, left, right) + page_width, page_height = calculate_page_size + + commands = [ + "-size #{page_width * 2}x#{page_height}", + "xc:white", + "-gravity Northwest" + ] + + left_creep = 0 + right_creep = 0 + if @config['page_thickness'] + max_creep = (total_faces / 2) + + left_creep = ((max_creep - (face - max_creep).abs) * @config['page_thickness'].to_f * @config['dpi'].to_i).floor + right_creep = ((max_creep - (total_faces - face - max_creep).abs) * @config['page_thickness'].to_f * @config['dpi'].to_i).floor + end + + if left + commands << "-draw 'image Over -#{left_creep.to_i},0 0,0 \"#{left}\"'" + end + if right + commands << "-draw 'image Over #{(page_width + right_creep).to_i},0 0,0 \"#{right}\"'" + end + commands << "-channel RGB -depth 8 tmp2.png" + + convert(commands) + system("sam2p -c:lzw -m:dpi:#{(72.0 * (72.0 / @config['dpi'].to_f))} tmp2.png PDF:\"#{output}\""); + system("rm tmp2.png") + end +end + +any_rebuilt = false +any_rsync = false + +config = YAML::load(File.open(ARGV[0], "r")) +global = config['Global'] + +if !global['path']; exit 1; end + +page_index_format = global['page_index_format'] ? global['page_index_format'] : "%0#{Math.log10(Dir[global['path']].length).ceil}d" + +page_index = 1 +fileinfo_by_file = {} + +if global['pages'] + re = nil + + files = global['pages'].collect do |f| + result = nil + case f.class.to_s + when 'String' + result = global['path'] + f + if f == "blank" + fileinfo_by_file[result] = { 'file' => f } + end + when 'Hash' + if f['file'] + case f['file'].class.to_s + when 'String' + result = global['path'] + f['file'] + fileinfo_by_file[result] = f + when 'Array' + result = f['file'].collect { |sub_f| global['path'] + sub_f } + fileinfo_by_file[result.join(",")] = f + end + else + result = f + end + end + result + end +else + re = Regexp.new(global['match']) + + files = Dir[global['path']].sort.collect do |filename| + if matches = re.match(filename) + filename + end + end +end + +paginated_source_files = {} +rsync_files_by_target = {} + +files.each do |filename| + ok = true; matches = nil; fileinfo = {} + + if filename.instance_of? Hash + if filename['blank'] + ok = false + config.each do |type, info| + if info['is_paginated'] + if !paginated_source_files[type]; paginated_source_files[type] = []; end + paginated_source_files[type] << nil + end + end + page_index += 1 + else + fileinfo = filename + filename = fileinfo['file'] + end + else + if re; ok = matches = re.match(filename); end + end + + if ok + filename_display = (filename.instance_of? Array) ? filename.join(", ") : filename + + puts "Examining #{filename_display}..." + + filename_parts = { + 'page_index' => sprintf(page_index_format, page_index) + } + + if matches + all, index, title = matches.to_a + else + index = page_index - 1 + title = "" + end + + if global['title']; title = global['title'].gsub("{index}", index).gsub("{title}", title); end + filename_parts['index'] = index + filename_parts['title'] = title + + config.each do |type, info| + if type != "Global" + input = nil; output = nil + + fileinfo_key = (filename.instance_of? Array) ? filename.join(",") : filename + + file_fileinfo = (fileinfo_by_file[fileinfo_key]) ? fileinfo_by_file[fileinfo_key] : {} + + extension = File.extname((filename.instance_of? Array) ? filename[0] : filename).downcase + + case extension + when ".svg" + case File.extname(config[type]['target']).downcase + when ".jpg", ".jpeg", ".png", ".gif" + input = SVGToTempBitmap + output = TempBitmapToWeb + when ".pdf" + input = SVGToTempBitmap + output = (info['is_paginated']) ? TempBitmapToPaginatedPrint : TempBitmapToPrint + end + end + + if !input; raise "No input handler for #{extension} defined"; end + if !output; raise "No output handler for #{File.extname(config[type]['target']).downcase} defined"; end + + input_obj = input.instance + input_obj.config = info.dup.merge(fileinfo).merge(file_fileinfo) + + output_obj = output.instance + output_obj.config = info.dup.merge(fileinfo).merge(file_fileinfo) + + if info['is_paginated'] + output_obj.config['target'] += "-{page_index}.png" + end + + targets = output_obj.targets(filename_parts) + + rebuild = false + + [ targets ].flatten.each do |t| + if !File.exists?(t) + rebuild = true + else + [ filename ].flatten.each do |f| + if File.mtime(f) > File.mtime(t) + rebuild = true + end + end + end + end + + if rebuild + any_rebuilt = true + + puts "Rebuilding #{filename_display} (#{type})..." + + tmp_files = input_obj.build(filename) + + output_files = [] + case tmp_files.class.to_s + when "String" + output_obj.build(tmp_files, targets) + output_files << targets + when "Array" + [0,1].each do |i| + output_obj.build(tmp_files[i], targets[i], (i == 0) ? "left" : "right") + output_files << targets[i] + end + end + + input_obj.cleanup + end + if info['is_paginated'] + if !paginated_source_files[type]; paginated_source_files[type] = []; end + paginated_source_files[type] << targets + end + if info['rsync'] + if !rsync_files_by_target[info['rsync']]; rsync_files_by_target[info['rsync']] = []; end + rsync_files_by_target[info['rsync']] << targets + end + end + end + + page_index += 1 + end +end + +config.each do |type, info| + if info['is_paginated'] + output = TempBitmapToPaginatedPrint + + output_obj = output.instance + output_obj.config = info.dup + + output_obj.paginate(paginated_source_files[type].flatten) + end + + if info['rsync'] + system("echo '#{rsync_files_by_target[info['rsync']].join("\n")}' | rsync -vru --files-from=- . #{info['rsync']}") + end +end + +if global['use_git'] + system("git add .") + system("git commit -a") +end