From 03f4c23c04b83b218bc892d65f5155522b58c37a Mon Sep 17 00:00:00 2001 From: Scott Davis Date: Thu, 24 Feb 2011 01:02:31 -0500 Subject: [PATCH] abstracted out engine for chunkypng --- Gemfile.lock | 2 +- .../sass_extensions/functions/sprites.rb | 58 +---- lib/compass/sass_extensions/sprites/base.rb | 229 ++++++++++-------- .../sprites/engines/chunky_png_engine.rb | 36 +++ 4 files changed, 171 insertions(+), 154 deletions(-) create mode 100644 lib/compass/sass_extensions/sprites/engines/chunky_png_engine.rb diff --git a/Gemfile.lock b/Gemfile.lock index 253f6ef0..c4f9afa6 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -1,7 +1,7 @@ PATH remote: . specs: - compass (0.11.beta.2.153582f) + compass (0.11.beta.2.1c71df3) chunky_png (~> 0.12.0) sass (>= 3.1.0.alpha.218) diff --git a/lib/compass/sass_extensions/functions/sprites.rb b/lib/compass/sass_extensions/functions/sprites.rb index 3ff566d6..71f64540 100644 --- a/lib/compass/sass_extensions/functions/sprites.rb +++ b/lib/compass/sass_extensions/functions/sprites.rb @@ -1,6 +1,7 @@ require 'digest/md5' require 'compass/sass_extensions/sprites/image' require 'compass/sass_extensions/sprites/base' +require 'compass/sass_extensions/sprites/engines/chunky_png_engine' module Compass::SassExtensions::Functions::Sprites ZERO = Sass::Script::Number::new(0) @@ -14,63 +15,8 @@ module Compass::SassExtensions::Functions::Sprites end end - class SpriteMap < Compass::SassExtensions::Sprites::Base - # Calculates the overal image dimensions - # collects image sizes and input parameters for each sprite - def compute_image_metadata! - @width = 0 - init_images - compute_image_positions! - @height = @images.last.top + @images.last.height - end - - def init_images - @images = image_names.collect do |relative_file| - image = Compass::SassExtensions::Sprites::Image.new(relative_file, options) - @width = [ @width, image.width + image.offset ].max - image - end - end - - def compute_image_positions! - @images.each_with_index do |image, index| - image.left = image.position.unit_str == "%" ? (@width - image.width) * (image.position.value / 100) : image.position.value - next if index == 0 - last_image = @images[index-1] - image.top = last_image.top + last_image.height + [image.spacing, last_image.spacing].max - end - end + class SpriteMap < Compass::SassExtensions::Sprites::ChunkyPngEngine - def image_for(name) - @images.detect { |img| img.name == name} - end - - def require_png_library! - begin - require 'oily_png' - rescue LoadError - require 'chunky_png' - end - end - - # Returns a PNG object - def construct_sprite - require_png_library! - output_png = ChunkyPNG::Image.new(width, height, ChunkyPNG::Color::TRANSPARENT) - images.each do |image| - input_png = ChunkyPNG::Image.from_file(image.file) - if image.repeat == "no-repeat" - output_png.replace input_png, image.left, image.top - else - x = image.left - (image.left / image.width).ceil * image.width - while x < width do - output_png.replace input_png, x, image.top - x += image.width - end - end - end - output_png - end end # Creates a SpriteMap object. A sprite map, when used in a property is the same diff --git a/lib/compass/sass_extensions/sprites/base.rb b/lib/compass/sass_extensions/sprites/base.rb index f5f7080f..70f7ee94 100644 --- a/lib/compass/sass_extensions/sprites/base.rb +++ b/lib/compass/sass_extensions/sprites/base.rb @@ -1,121 +1,156 @@ module Compass module SassExtensions module Sprites - class Base < Sass::Script::Literal - # Changing this string will invalidate all previously generated sprite images. - # We should do so only when the packing algorithm changes - SPRITE_VERSION = "1" - - attr_accessor :image_names, :path, :name, :options - attr_accessor :images, :width, :height - - def self.from_uri(uri, context, kwargs) - path, name = Compass::Sprites.path_and_name(uri.value) - sprites = Compass::Sprites.discover_sprites(uri.value).map do |sprite| - sprite.gsub(Compass.configuration.images_path+"/", "") + module Base + module ClassMethods + def from_uri(uri, context, kwargs) + path, name = Compass::Sprites.path_and_name(uri.value) + sprites = Compass::Sprites.discover_sprites(uri.value).map do |sprite| + sprite.gsub(Compass.configuration.images_path+"/", "") + end + new(sprites, path, name, context, kwargs) end - new(sprites, path, name, context, kwargs) end + #instance Methods + module InstanceMethods + + def self.included(base) + base.extend(ClassMethods) + end + + # Changing this string will invalidate all previously generated sprite images. + # We should do so only when the packing algorithm changes + SPRITE_VERSION = "1" - def initialize(image_names, path, name, context, options) - @image_names, @path, @name, @options = image_names, path, name, options - @images = nil - @width = nil - @height = nil - @evaluation_context = context - validate! - compute_image_metadata! - end + attr_accessor :image_names, :path, :name, :options + attr_accessor :images, :width, :height - # Calculate the size of the sprite - def size - [width, height] - end - def sprite_names - image_names.map{|f| Compass::Sprites.sprite_name(f) } - end + def initialize(image_names, path, name, context, options) + @image_names, @path, @name, @options = image_names, path, name, options + @images = nil + @width = nil + @height = nil + @evaluation_context = context + validate! + compute_image_metadata! + end - def validate! - for sprite_name in sprite_names - unless sprite_name =~ /\A#{Sass::SCSS::RX::IDENT}\Z/ - raise Sass::SyntaxError, "#{sprite_name} must be a legal css identifier" + # Calculate the size of the sprite + def size + [width, height] + end + + # Calculates the overal image dimensions + # collects image sizes and input parameters for each sprite + def compute_image_metadata! + @width = 0 + init_images + compute_image_positions! + @height = @images.last.top + @images.last.height + end + + def init_images + @images = image_names.collect do |relative_file| + image = Compass::SassExtensions::Sprites::Image.new(relative_file, options) + @width = [ @width, image.width + image.offset ].max + image end end - end - - # The on-the-disk filename of the sprite - def filename - File.join(Compass.configuration.images_path, "#{path}-#{uniqueness_hash}.png") - end - - # Calculates the overal image dimensions - # collects image sizes and input parameters for each sprite - def compute_image_metadata! - end - - # Generate a sprite image if necessary - def generate - if generation_required? - sprite_data = construct_sprite - save!(sprite_data) - Compass.configuration.run_callback(:sprite_generated, sprite_data) + # Calculates the overal image dimensions + # collects image sizes and input parameters for each sprite + def compute_image_positions! + @images.each_with_index do |image, index| + image.left = image.position.unit_str == "%" ? (@width - image.width) * (image.position.value / 100) : image.position.value + next if index == 0 + last_image = @images[index-1] + image.top = last_image.top + last_image.height + [image.spacing, last_image.spacing].max + end end - end - def generation_required? - !File.exists?(filename) || outdated? - end + def image_for(name) + @images.detect { |img| img.name == name} + end - def uniqueness_hash - @uniqueness_hash ||= begin - sum = Digest::MD5.new - sum << SPRITE_VERSION - sum << path - images.each do |image| - [:relative_file, :height, :width, :repeat, :spacing, :position, :digest].each do |attr| - sum << image.send(attr).to_s + def sprite_names + image_names.map{|f| Compass::Sprites.sprite_name(f) } + end + + def validate! + for sprite_name in sprite_names + unless sprite_name =~ /\A#{Sass::SCSS::RX::IDENT}\Z/ + raise Sass::SyntaxError, "#{sprite_name} must be a legal css identifier" end end - sum.hexdigest[0...10] end - @uniqueness_hash - end - # saves the sprite for later retrieval - def save!(output_png) - saved = output_png.save filename - Compass.configuration.run_callback(:sprite_saved, filename) - saved - end + # The on-the-disk filename of the sprite + def filename + File.join(Compass.configuration.images_path, "#{path}-#{uniqueness_hash}.png") + end - # All the full-path filenames involved in this sprite - def image_filenames - image_names.map do |image_name| - File.join(Compass.configuration.images_path, image_name) + # Generate a sprite image if necessary + def generate + if generation_required? + sprite_data = construct_sprite + save!(sprite_data) + Compass.configuration.run_callback(:sprite_generated, sprite_data) + end + end + + def generation_required? + !File.exists?(filename) || outdated? + end + + def uniqueness_hash + @uniqueness_hash ||= begin + sum = Digest::MD5.new + sum << SPRITE_VERSION + sum << path + images.each do |image| + [:relative_file, :height, :width, :repeat, :spacing, :position, :digest].each do |attr| + sum << image.send(attr).to_s + end + end + sum.hexdigest[0...10] + end + @uniqueness_hash + end + + # saves the sprite for later retrieval + def save!(output_png) + saved = output_png.save filename + Compass.configuration.run_callback(:sprite_saved, filename) + saved + end + + # All the full-path filenames involved in this sprite + def image_filenames + image_names.map do |image_name| + File.join(Compass.configuration.images_path, image_name) + end + end + + def inspect + to_s + end + + def to_s(options = self.options) + sprite_url(self).value + end + + def respond_to?(meth) + super || @evaluation_context.respond_to?(meth) + end + + def method_missing(meth, *args, &block) + if @evaluation_context.respond_to?(meth) + @evaluation_context.send(meth, *args, &block) + else + super + end end end - - def inspect - to_s - end - - def to_s(options = self.options) - sprite_url(self).value - end - - def respond_to?(meth) - super || @evaluation_context.respond_to?(meth) - end - - def method_missing(meth, *args, &block) - if @evaluation_context.respond_to?(meth) - @evaluation_context.send(meth, *args, &block) - else - super - end - end - end end end diff --git a/lib/compass/sass_extensions/sprites/engines/chunky_png_engine.rb b/lib/compass/sass_extensions/sprites/engines/chunky_png_engine.rb new file mode 100644 index 00000000..40db7adc --- /dev/null +++ b/lib/compass/sass_extensions/sprites/engines/chunky_png_engine.rb @@ -0,0 +1,36 @@ +module Compass + module SassExtensions + module Sprites + class ChunkyPngEngine < Sass::Script::Literal + include Compass::SassExtensions::Sprites::Base::InstanceMethods + + def require_png_library! + begin + require 'oily_png' + rescue LoadError + require 'chunky_png' + end + end + + # Returns a PNG object + def construct_sprite + require_png_library! + output_png = ChunkyPNG::Image.new(width, height, ChunkyPNG::Color::TRANSPARENT) + images.each do |image| + input_png = ChunkyPNG::Image.from_file(image.file) + if image.repeat == "no-repeat" + output_png.replace input_png, image.left, image.top + else + x = image.left - (image.left / image.width).ceil * image.width + while x < width do + output_png.replace input_png, x, image.top + x += image.width + end + end + end + output_png + end + end + end + end +end \ No newline at end of file