Change the sprite utilities to create a sprite literal object that can retain information about the sprite for use in other function calls. This avoids the need for global storage and allows multiple sprites constructed from the same sprite folder with different values.
This commit is contained in:
parent
adcfcf4556
commit
027ebdd8c4
@ -0,0 +1,29 @@
|
|||||||
|
// Set the width and height of an element to the original
|
||||||
|
// dimensions of an image before it was included in the sprite.
|
||||||
|
@mixin sprite-dimensions($sprite, $name) {
|
||||||
|
height: image-height(sprite-file($sprite, $name));
|
||||||
|
width: image-width(sprite-file($sprite, $name));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the background position of the given `$sprite` to display the
|
||||||
|
// image of the given `$name`. You can move the image relative to its
|
||||||
|
// natural position by passing `$offset-x` and `$offset-y`.
|
||||||
|
@mixin sprite-position($sprite, $name, $offset-x: 0, $offset-y: 0) {
|
||||||
|
background-position: sprite-position($sprite, $name, $offset-x, $offset-y);
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin sprite($sprite, $name, $dimensions: false, $offset-x: 0, $offset-y: 0) {
|
||||||
|
@include sprite-position($sprite, $name, $offset-x, $offset-y);
|
||||||
|
@if $dimensions {
|
||||||
|
@include sprite-dimensions($sprite, $name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin sprites($sprite, $sprite-names, $base-class: false, $dimensions: false) {
|
||||||
|
@each $sprite-name in $sprite-names {
|
||||||
|
.#{sprite-name($sprite)}-#{$sprite-name} {
|
||||||
|
@if $base-class { @extend #{$base-class}; }
|
||||||
|
@include sprite($sprite, $sprite-name, $dimensions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
require 'chunky_png'
|
require 'chunky_png'
|
||||||
|
|
||||||
module Compass::SassExtensions::Functions::Sprites
|
module Compass::SassExtensions::Functions::Sprites
|
||||||
SASS_NULL = Sass::Script::Number::new(0)
|
ZERO = Sass::Script::Number::new(0)
|
||||||
|
|
||||||
# Provides a consistent interface for getting a variable in ruby
|
# Provides a consistent interface for getting a variable in ruby
|
||||||
# from a keyword argument hash that accounts for underscores/dash equivalence
|
# from a keyword argument hash that accounts for underscores/dash equivalence
|
||||||
@ -12,111 +12,253 @@ module Compass::SassExtensions::Functions::Sprites
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def generate_sprite_image(uri, kwargs = {})
|
class Sprite < Sass::Script::Literal
|
||||||
kwargs.extend VariableReader
|
|
||||||
path, name = Compass::Sprites.path_and_name(uri.value)
|
|
||||||
last_spacing = 0
|
|
||||||
width = 0
|
|
||||||
height = 0
|
|
||||||
|
|
||||||
# Get image metadata
|
attr_accessor :image_names, :path, :name, :options
|
||||||
Compass::Sprites.discover_sprites(uri.value).each do |file|
|
attr_accessor :images, :width, :height
|
||||||
Compass::Sprites.compute_image_metadata! file, path, name
|
|
||||||
|
def self.from_uri(uri, context, kwargs)
|
||||||
|
path, name = Compass::Sprites.path_and_name(uri.value)
|
||||||
|
new(Compass::Sprites.discover_sprites(uri.value), path, name, context, kwargs)
|
||||||
end
|
end
|
||||||
|
|
||||||
images = Compass::Sprites.sprites(path, name)
|
def initialize(image_names, path, name, context, options)
|
||||||
|
@image_names, @path, @name, @options = image_names, path, name, options
|
||||||
# Calculation
|
@images = nil
|
||||||
images.each do |image|
|
@width = nil
|
||||||
current_spacing = number_from_var(kwargs, "#{image[:name]}-spacing", 0)
|
@height = nil
|
||||||
if height > 0
|
@evaluation_context = context
|
||||||
height += [current_spacing, last_spacing].max
|
validate!
|
||||||
end
|
compute_image_metadata!
|
||||||
image[:y] = height
|
|
||||||
height += image[:height]
|
|
||||||
last_spacing = current_spacing
|
|
||||||
width = image[:width] if image[:width] > width
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Generation
|
def sprite_names
|
||||||
output_png = ChunkyPNG::Image.new(width, height, ChunkyPNG::Color::TRANSPARENT)
|
image_names.map{|f| Compass::Sprites.sprite_name(f) }
|
||||||
images.each do |image|
|
end
|
||||||
input_png = ChunkyPNG::Image.from_file(image[:file])
|
|
||||||
|
|
||||||
position = kwargs.get_var("#{image[:name]}-position") || Sass::Script::Number.new(0, ["%"])
|
def validate!
|
||||||
if position.unit_str == "%"
|
for sprite_name in sprite_names
|
||||||
image[:x] = (width - image[:width]) * (position.value / 100)
|
unless sprite_name =~ /\A#{Sass::SCSS::RX::IDENT}\Z/
|
||||||
else
|
raise Sass::SyntaxError, "#{sprite_name} must be a legal css identifier"
|
||||||
image[:x] = position.value
|
end
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
|
||||||
repeat = if (var = kwargs.get_var("#{image[:name]}-repeat"))
|
# Calculates the overal image dimensions
|
||||||
|
# collects image sizes and input parameters for each sprite
|
||||||
|
def compute_image_metadata!
|
||||||
|
@images = []
|
||||||
|
@width = 0
|
||||||
|
image_names.each do |file|
|
||||||
|
relative_file = file.gsub(Compass.configuration.images_path+"/", "")
|
||||||
|
width, height = Compass::SassExtensions::Functions::ImageSize::ImageProperties.new(file).size
|
||||||
|
sprite_name = Compass::Sprites.sprite_name(relative_file)
|
||||||
|
@width = [@width, width].max
|
||||||
|
@images << {
|
||||||
|
:name => sprite_name,
|
||||||
|
:file => file,
|
||||||
|
:relative_file => relative_file,
|
||||||
|
:height => height,
|
||||||
|
:width => width,
|
||||||
|
:repeat => repeat_for(sprite_name),
|
||||||
|
:spacing => spacing_for(sprite_name),
|
||||||
|
:position => position_for(sprite_name)
|
||||||
|
}
|
||||||
|
end
|
||||||
|
@images.each_with_index do |image, index|
|
||||||
|
if index == 0
|
||||||
|
image[:top] = 0
|
||||||
|
else
|
||||||
|
last_image = @images[index-1]
|
||||||
|
image[:top] = last_image[:top] + last_image[:height] + [image[:spacing], last_image[:spacing]].max
|
||||||
|
end
|
||||||
|
if image[:position].unit_str == "%"
|
||||||
|
image[:left] = (@width - image[:width]) * (image[:position].value / 100)
|
||||||
|
else
|
||||||
|
image[:left] = image[:position].value
|
||||||
|
end
|
||||||
|
end
|
||||||
|
@height = @images.last[:top] + @images.last[:height]
|
||||||
|
end
|
||||||
|
|
||||||
|
def position_for(name)
|
||||||
|
options.get_var("#{name}-position") || options.get_var("position") || Sass::Script::Number.new(0, ["px"])
|
||||||
|
end
|
||||||
|
|
||||||
|
def repeat_for(name)
|
||||||
|
if (var = options.get_var("#{name}-repeat"))
|
||||||
|
var.value
|
||||||
|
elsif (var = options.get_var("repeat"))
|
||||||
var.value
|
var.value
|
||||||
else
|
else
|
||||||
"no-repeat"
|
"no-repeat"
|
||||||
end
|
end
|
||||||
if repeat == "no-repeat"
|
end
|
||||||
output_png.replace input_png, image[:x], image[:y]
|
|
||||||
else
|
def spacing_for(name)
|
||||||
x = image[:x] - (image[:x] / image[:width]).ceil * image[:width]
|
(options.get_var("#{name}-spacing") ||
|
||||||
while x < width do
|
options.get_var("spacing") ||
|
||||||
output_png.replace input_png, x, image[:y]
|
ZERO).value
|
||||||
x += image[:width]
|
end
|
||||||
end
|
|
||||||
|
def image_for(name)
|
||||||
|
@images.detect{|img| img[:name] == name}
|
||||||
|
end
|
||||||
|
|
||||||
|
# Calculate the size of the sprite
|
||||||
|
def size
|
||||||
|
[width, height]
|
||||||
|
end
|
||||||
|
|
||||||
|
# Generate a sprite image if necessary
|
||||||
|
def generate
|
||||||
|
if generation_required?
|
||||||
|
save!(construct_sprite)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
output_png.save File.join(File.join(Compass.configuration.images_path, "#{path}.png"))
|
|
||||||
|
|
||||||
sprite_url(uri)
|
def generation_required?
|
||||||
end
|
!File.exists?(filename) || outdated?
|
||||||
Sass::Script::Functions.declare :generate_sprite_image, [:uri], :var_kwargs => true
|
end
|
||||||
|
|
||||||
|
# Returns a PNG object
|
||||||
|
def construct_sprite
|
||||||
|
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
|
||||||
|
|
||||||
|
# The on-the-disk filename of the sprite
|
||||||
|
def filename
|
||||||
|
File.join(File.join(Compass.configuration.images_path, "#{path}.png"))
|
||||||
|
end
|
||||||
|
|
||||||
|
# saves the sprite for later retrieval
|
||||||
|
def save!(output_png)
|
||||||
|
output_png.save filename
|
||||||
|
end
|
||||||
|
|
||||||
|
# All the full-path filenames involved in this sprite
|
||||||
|
def image_filenames
|
||||||
|
image_names.map do |image_name|
|
||||||
|
File.join(File.join(Compass.configuration.images_path, image_name))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Checks whether this sprite is outdated
|
||||||
|
def outdated?
|
||||||
|
last_update = self.mtime
|
||||||
|
image_filenames.each do |image|
|
||||||
|
return true if File.mtime(image) > last_update
|
||||||
|
end
|
||||||
|
false
|
||||||
|
end
|
||||||
|
|
||||||
|
def mtime
|
||||||
|
File.mtime(filename)
|
||||||
|
end
|
||||||
|
|
||||||
|
def inspect
|
||||||
|
to_s
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_s(options = self.options)
|
||||||
|
sprite_url(self).value
|
||||||
|
end
|
||||||
|
|
||||||
|
def method_missing(meth, *args, &block)
|
||||||
|
if @evaluation_context.respond_to?(meth)
|
||||||
|
@evaluation_context.send(meth, *args, &block)
|
||||||
|
else
|
||||||
|
super
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def sprite_image(uri, x_shift = SASS_NULL, y_shift = SASS_NULL, depricated_1 = nil, depricated_2 = nil)
|
|
||||||
check_spacing_deprecation uri, depricated_1, depricated_2
|
|
||||||
url = sprite_url(uri)
|
|
||||||
position = sprite_position(uri, x_shift, y_shift)
|
|
||||||
Sass::Script::String.new("#{url} #{position}")
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def sprite_url(uri)
|
def sprite(uri, kwargs = {})
|
||||||
path, name = Compass::Sprites.path_and_name(uri.value)
|
kwargs.extend VariableReader
|
||||||
image_url(Sass::Script::String.new("#{path}.png"))
|
Sprite.from_uri(uri, self, kwargs)
|
||||||
|
end
|
||||||
|
Sass::Script::Functions.declare :sprite, [:uri], :var_kwargs => true
|
||||||
|
|
||||||
|
def sprite_image(sprite, image = nil, x_shift = ZERO, y_shift = ZERO)
|
||||||
|
unless sprite.is_a?(Sprite)
|
||||||
|
missing_sprite!("sprite-image")
|
||||||
|
end
|
||||||
|
unless image && image.is_a?(Sass::Script::String)
|
||||||
|
raise Sass::SyntaxError, %Q(The second argument to sprite-image must be a sprite name. See http://beta.compass-style.org/help/tutorials/spriting/ for more information.)
|
||||||
|
end
|
||||||
|
url = sprite_url(sprite)
|
||||||
|
position = sprite_position(sprite, image, x_shift, y_shift)
|
||||||
|
Sass::Script::List.new([url, position], :space)
|
||||||
end
|
end
|
||||||
|
|
||||||
def sprite_position(uri, x_shift = SASS_NULL, y_shift = SASS_NULL, depricated_1 = nil, depricated_2 = nil)
|
def sprite_name(sprite)
|
||||||
check_spacing_deprecation uri, depricated_1, depricated_2
|
unless sprite.is_a?(Sprite)
|
||||||
path, name = Compass::Sprites.path_and_name(uri.value)
|
missing_sprite!("sprite-name")
|
||||||
image_name = File.basename(uri.value, '.png')
|
end
|
||||||
image = Compass::Sprites.sprites(path, name).detect{ |image| image[:name] == image_name }
|
Sass::Script::String.new(sprite.name)
|
||||||
|
end
|
||||||
|
|
||||||
|
def sprite_file(sprite, image_name)
|
||||||
|
unless sprite.is_a?(Sprite)
|
||||||
|
missing_sprite!("sprite-file")
|
||||||
|
end
|
||||||
|
if image = sprite.image_for(image_name.value)
|
||||||
|
Sass::Script::String.new(image[:relative_file])
|
||||||
|
else
|
||||||
|
missing_image!(sprite, image_name)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def sprite_url(sprite)
|
||||||
|
unless sprite.is_a?(Sprite)
|
||||||
|
missing_sprite!("sprite-url")
|
||||||
|
end
|
||||||
|
sprite.generate
|
||||||
|
image_url(Sass::Script::String.new("#{sprite.path}.png"))
|
||||||
|
end
|
||||||
|
|
||||||
|
def missing_image!(sprite, image_name)
|
||||||
|
raise Sass::SyntaxError, "No image called #{image_name} found in sprite #{sprite.path}/#{sprite.name}. Did you mean one of: #{sprite.sprite_names.join(", ")}"
|
||||||
|
end
|
||||||
|
|
||||||
|
def missing_sprite!(function_name)
|
||||||
|
raise Sass::SyntaxError, %Q(The first argument to #{function_name} must be a sprite. See http://beta.compass-style.org/help/tutorials/spriting/ for more information.)
|
||||||
|
end
|
||||||
|
|
||||||
|
def sprite_position(sprite, image_name = nil, x_shift = ZERO, y_shift = ZERO)
|
||||||
|
unless sprite.is_a?(Sprite)
|
||||||
|
missing_sprite!("sprite-position")
|
||||||
|
end
|
||||||
|
unless image_name && image_name.is_a?(Sass::Script::String)
|
||||||
|
raise Sass::SyntaxError, %Q(The second argument to sprite-image must be a sprite name. See http://beta.compass-style.org/help/tutorials/spriting/ for more information.)
|
||||||
|
end
|
||||||
|
image = sprite.image_for(image_name.value)
|
||||||
|
unless image
|
||||||
|
missing_image!(sprite, image_name)
|
||||||
|
end
|
||||||
if x_shift.unit_str == "%"
|
if x_shift.unit_str == "%"
|
||||||
x = x_shift.to_s
|
x = x_shift # CE: Shouldn't this be a percentage of the total width?
|
||||||
else
|
else
|
||||||
x = x_shift.value - image[:x]
|
x = x_shift.value - image[:left]
|
||||||
x = "#{x}px" unless x == 0
|
x = Sass::Script::Number.new(x, x == 0 ? [] : ["px"])
|
||||||
end
|
end
|
||||||
y = y_shift.value - image[:y]
|
y = y_shift.value - image[:top]
|
||||||
y = "#{y}px" unless y == 0
|
y = Sass::Script::Number.new(y, y == 0 ? [] : ["px"])
|
||||||
Sass::Script::String.new("#{x} #{y}")
|
Sass::Script::List.new([x, y],:space)
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def number_from_var(kwargs, var_name, default_value)
|
|
||||||
if number = kwargs.get_var(var_name)
|
|
||||||
assert_type number, :Number
|
|
||||||
number.value
|
|
||||||
else
|
|
||||||
default_value
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def check_spacing_deprecation(uri, spacing_before, spacing_after)
|
|
||||||
if spacing_before or spacing_after
|
|
||||||
path, name, image_name = Compass::Sprites.path_and_name(uri.value)
|
|
||||||
message = %Q(Spacing parameter is deprecated. ) +
|
|
||||||
%Q(Please add `$#{name}-#{image_name}-spacing: #{spacing_before};` ) +
|
|
||||||
%Q(before the `@import "#{path}/*.png";` statement.)
|
|
||||||
raise Compass::Error, message
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
@ -4,66 +4,41 @@ module Compass
|
|||||||
attr_accessor :path
|
attr_accessor :path
|
||||||
|
|
||||||
class << self
|
class << self
|
||||||
def reset
|
|
||||||
@@sprites = {}
|
|
||||||
end
|
|
||||||
|
|
||||||
def path_and_name(uri)
|
def path_and_name(uri)
|
||||||
if uri =~ %r{((.+/)?(.+))/(.+?)\.png}
|
if uri =~ %r{((.+/)?(.+))/(.+?)\.png}
|
||||||
[$1, $3, $4]
|
[$1, $3, $4]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def sprites(path, name, create = false)
|
|
||||||
if !defined?(@@sprites) || @@sprites.nil?
|
|
||||||
@@sprites = {}
|
|
||||||
end
|
|
||||||
index = "#{path}/#{name}"
|
|
||||||
images = @@sprites[index]
|
|
||||||
if images
|
|
||||||
images
|
|
||||||
elsif create
|
|
||||||
images = @@sprites[index] = []
|
|
||||||
else
|
|
||||||
raise Compass::Error, %Q(`@import` statement missing. Please add `@import "#{path}/*.png";`.)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def discover_sprites(uri)
|
def discover_sprites(uri)
|
||||||
glob = File.join(Compass.configuration.images_path, uri)
|
glob = File.join(Compass.configuration.images_path, uri)
|
||||||
Dir.glob(glob).sort
|
Dir.glob(glob).sort
|
||||||
end
|
end
|
||||||
|
|
||||||
def compute_image_metadata!(file, path, name)
|
def sprite_name(file)
|
||||||
width, height = Compass::SassExtensions::Functions::ImageSize::ImageProperties.new(file).size
|
File.basename(file, '.png')
|
||||||
sprite_name = File.basename(file, '.png');
|
|
||||||
unless sprite_name =~ /\A#{Sass::SCSS::RX::IDENT}\Z/
|
|
||||||
raise Sass::SyntaxError, "#{sprite_name} must be a legal css identifier"
|
|
||||||
end
|
|
||||||
sprites(path, name, true) << {
|
|
||||||
:name => sprite_name,
|
|
||||||
:file => file,
|
|
||||||
:height => height,
|
|
||||||
:width => width
|
|
||||||
}
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def images
|
def find_relative(*args)
|
||||||
Compass::Sprites.sprites(self.path, self.name, true)
|
nil
|
||||||
end
|
end
|
||||||
|
|
||||||
def find(uri, options)
|
def find(uri, options)
|
||||||
if uri =~ /\.png$/
|
if uri =~ /\.png$/
|
||||||
self.path, self.name = Compass::Sprites.path_and_name(uri)
|
self.path, self.name = Compass::Sprites.path_and_name(uri)
|
||||||
options.merge! :filename => name, :syntax => :scss, :importer => self
|
options.merge! :filename => name, :syntax => :scss, :importer => self
|
||||||
image_names = Compass::Sprites.discover_sprites(uri).map{|i| File.basename(i, '.png')}
|
sprite_files = Compass::Sprites.discover_sprites(uri)
|
||||||
|
image_names = sprite_files.map {|i| Compass::Sprites.sprite_name(i) }
|
||||||
Sass::Engine.new(content_for_images(uri, name, image_names), options)
|
Sass::Engine.new(content_for_images(uri, name, image_names), options)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def content_for_images(uri, name, images)
|
def content_for_images(uri, name, images)
|
||||||
<<-SCSS
|
<<-SCSS
|
||||||
|
@import "compass/utilities/sprites/base";
|
||||||
|
|
||||||
// General Sprite Defaults
|
// General Sprite Defaults
|
||||||
// You can override them before you import this file.
|
// You can override them before you import this file.
|
||||||
$#{name}-sprite-base-class: ".#{name}-sprite" !default;
|
$#{name}-sprite-base-class: ".#{name}-sprite" !default;
|
||||||
@ -82,50 +57,46 @@ $#{name}-#{sprite_name}-repeat: $#{name}-repeat !default;
|
|||||||
SCSS
|
SCSS
|
||||||
end.join}
|
end.join}
|
||||||
|
|
||||||
|
$#{name}-sprite: sprite("#{uri}",
|
||||||
|
#{images.map do |sprite_name|
|
||||||
|
%Q{ $#{sprite_name}-position: $#{name}-#{sprite_name}-position,
|
||||||
|
$#{sprite_name}-spacing: $#{name}-#{sprite_name}-spacing,
|
||||||
|
$#{sprite_name}-repeat: $#{name}-#{sprite_name}-repeat}
|
||||||
|
end.join(",\n")});
|
||||||
|
|
||||||
// All sprites should extend this class
|
// All sprites should extend this class
|
||||||
// The #{name}-sprite mixin will do so for you.
|
// The #{name}-sprite mixin will do so for you.
|
||||||
\#{$#{name}-sprite-base-class} {
|
\#{$#{name}-sprite-base-class} {
|
||||||
background: generate-sprite-image("#{uri}",
|
background: $#{name}-sprite no-repeat;
|
||||||
#{images.map do |sprite_name|
|
|
||||||
%Q{ $#{sprite_name}-position: $#{name}-#{sprite_name}-position,
|
|
||||||
$#{sprite_name}-spacing: $#{name}-#{sprite_name}-spacing,
|
|
||||||
$#{sprite_name}-repeat: $#{name}-#{sprite_name}-repeat}
|
|
||||||
end.join(",\n")}) no-repeat;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use this to set the dimensions of an element
|
// Use this to set the dimensions of an element
|
||||||
// based on the size of the original image.
|
// based on the size of the original image.
|
||||||
@mixin #{name}-sprite-dimensions($sprite) {
|
@mixin #{name}-sprite-dimensions($name) {
|
||||||
height: image-height("#{name}/\#{$sprite}.png");
|
@include sprite-dimensions($#{name}-sprite, $name)
|
||||||
width: image-width("#{name}/\#{$sprite}.png");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move the background position to display the sprite.
|
// Move the background position to display the sprite.
|
||||||
@mixin #{name}-sprite-position($sprite, $x: 0, $y: 0) {
|
@mixin #{name}-sprite-position($name, $offset-x: 0, $offset-y: 0) {
|
||||||
background-position: sprite-position("#{path}/\#{$sprite}.png", $x, $y);
|
@include sprite-position($#{name}-sprite, $name, $offset-x, $offset-y)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extends the sprite base class and set the background position for the desired sprite.
|
// Extends the sprite base class and set the background position for the desired sprite.
|
||||||
// It will also apply the image dimensions if $dimensions is true.
|
// It will also apply the image dimensions if $dimensions is true.
|
||||||
@mixin #{name}-sprite($sprite, $dimensions: $#{name}-sprite-dimensions, $x: 0, $y: 0) {
|
@mixin #{name}-sprite($name, $dimensions: $#{name}-sprite-dimensions, $offset-x: 0, $offset-y: 0) {
|
||||||
@extend \#{$#{name}-sprite-base-class};
|
@extend \#{$#{name}-sprite-base-class};
|
||||||
@include #{name}-sprite-position($sprite, $x, $y);
|
@include sprite($#{name}-sprite, $name, $dimensions, $offset-x, $offset-y)
|
||||||
@if $dimensions {
|
}
|
||||||
@include #{name}-sprite-dimensions($sprite);
|
|
||||||
}
|
@mixin #{name}-sprites($sprite-names, $dimensions: $#{name}-sprite-dimensions) {
|
||||||
|
@include sprites($#{name}-sprite, $sprite-names, $#{name}-sprite-base-class, $dimensions)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generates a class for each sprited image.
|
// Generates a class for each sprited image.
|
||||||
@mixin all-#{name}-sprites {
|
@mixin all-#{name}-sprites($dimensions: $#{name}-sprite-dimensions) {
|
||||||
#{images.map do |sprite_name|
|
@include #{name}-sprites(#{images.join(" ")}, $dimensions);
|
||||||
<<-SCSS
|
|
||||||
.#{name}-#{sprite_name} {
|
|
||||||
@include #{name}-sprite("#{sprite_name}");
|
|
||||||
}
|
|
||||||
SCSS
|
|
||||||
end.join}
|
|
||||||
}
|
}
|
||||||
SCSS
|
SCSS
|
||||||
end
|
end
|
||||||
|
|
||||||
def key(uri, options)
|
def key(uri, options)
|
||||||
|
@ -10,7 +10,6 @@ describe Compass::Sprites do
|
|||||||
FileUtils.cp_r @images_src_path, @images_tmp_path
|
FileUtils.cp_r @images_src_path, @images_tmp_path
|
||||||
Compass.configuration.images_path = @images_tmp_path
|
Compass.configuration.images_path = @images_tmp_path
|
||||||
Compass.configure_sass_plugin!
|
Compass.configure_sass_plugin!
|
||||||
Compass::Sprites.reset
|
|
||||||
end
|
end
|
||||||
|
|
||||||
after :each do
|
after :each do
|
||||||
@ -247,19 +246,21 @@ describe Compass::Sprites do
|
|||||||
|
|
||||||
it "should use position adjustments in functions" do
|
it "should use position adjustments in functions" do
|
||||||
css = render <<-SCSS
|
css = render <<-SCSS
|
||||||
$squares-position: 100%;
|
$squares-sprite: sprite("squares/*.png", $position: 100%);
|
||||||
@import "squares/*.png";
|
.squares-sprite {
|
||||||
|
background: $squares-sprite no-repeat;
|
||||||
|
}
|
||||||
|
|
||||||
.adjusted-percentage {
|
.adjusted-percentage {
|
||||||
background-position: sprite-position("squares/ten-by-ten.png", 100%);
|
background-position: sprite-position($squares-sprite, ten-by-ten, 100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.adjusted-px-1 {
|
.adjusted-px-1 {
|
||||||
background-position: sprite-position("squares/ten-by-ten.png", 4px);
|
background-position: sprite-position($squares-sprite, ten-by-ten, 4px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.adjusted-px-2 {
|
.adjusted-px-2 {
|
||||||
background-position: sprite-position("squares/twenty-by-twenty.png", -3px, 2px);
|
background-position: sprite-position($squares-sprite, twenty-by-twenty, -3px, 2px);
|
||||||
}
|
}
|
||||||
SCSS
|
SCSS
|
||||||
css.should == <<-CSS
|
css.should == <<-CSS
|
||||||
@ -289,15 +290,15 @@ describe Compass::Sprites do
|
|||||||
@import "squares/*.png";
|
@import "squares/*.png";
|
||||||
|
|
||||||
.adjusted-percentage {
|
.adjusted-percentage {
|
||||||
@include squares-sprite("ten-by-ten", $x: 100%);
|
@include squares-sprite("ten-by-ten", $offset-x: 100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.adjusted-px-1 {
|
.adjusted-px-1 {
|
||||||
@include squares-sprite("ten-by-ten", $x: 4px);
|
@include squares-sprite("ten-by-ten", $offset-x: 4px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.adjusted-px-2 {
|
.adjusted-px-2 {
|
||||||
@include squares-sprite("twenty-by-twenty", $x: -3px, $y: 2px);
|
@include squares-sprite("twenty-by-twenty", $offset-x: -3px, $offset-y: 2px);
|
||||||
}
|
}
|
||||||
SCSS
|
SCSS
|
||||||
css.should == <<-CSS
|
css.should == <<-CSS
|
||||||
@ -344,89 +345,46 @@ describe Compass::Sprites do
|
|||||||
image_md5('squares.png').should == '0187306f3858136feee87d3017e7f307'
|
image_md5('squares.png').should == '0187306f3858136feee87d3017e7f307'
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should use the sprite-image and sprite-url function as in lemonade" do
|
it "should provide a nice errors for lemonade's old users" do
|
||||||
css = render <<-SCSS
|
|
||||||
@import "squares/*.png";
|
|
||||||
|
|
||||||
.squares-1 {
|
|
||||||
background: sprite-image("squares/twenty-by-twenty.png") no-repeat;
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-2 {
|
|
||||||
background: sprite-image("squares/twenty-by-twenty.png", 100%) no-repeat;
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-3 {
|
|
||||||
background: sprite-image("squares/twenty-by-twenty.png", -4px, 3px) no-repeat;
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-4 {
|
|
||||||
background-image: sprite-url("squares/twenty-by-twenty.png");
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-5 {
|
|
||||||
background-image: sprite-url("squares/*.png");
|
|
||||||
}
|
|
||||||
SCSS
|
|
||||||
css.should == <<-CSS
|
|
||||||
.squares-sprite {
|
|
||||||
background: url('/squares.png') no-repeat;
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-1 {
|
|
||||||
background: url('/squares.png') 0 -10px no-repeat;
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-2 {
|
|
||||||
background: url('/squares.png') 100% -10px no-repeat;
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-3 {
|
|
||||||
background: url('/squares.png') -4px -7px no-repeat;
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-4 {
|
|
||||||
background-image: url('/squares.png');
|
|
||||||
}
|
|
||||||
|
|
||||||
.squares-5 {
|
|
||||||
background-image: url('/squares.png');
|
|
||||||
}
|
|
||||||
CSS
|
|
||||||
end
|
|
||||||
|
|
||||||
it "should raise deprecation errors for lemonade's spacing syntax" do
|
|
||||||
proc do
|
proc do
|
||||||
render <<-SCSS
|
render <<-SCSS
|
||||||
@import "squares/*.png";
|
|
||||||
|
|
||||||
.squares {
|
.squares {
|
||||||
background: sprite-image("squares/twenty-by-twenty.png", 0, 0, 11px) no-repeat;
|
background: sprite-url("squares/*.png") no-repeat;
|
||||||
}
|
}
|
||||||
SCSS
|
SCSS
|
||||||
end.should raise_error Compass::Error,
|
end.should raise_error Sass::SyntaxError,
|
||||||
%q(Spacing parameter is deprecated. Please add `$squares-twenty-by-twenty-spacing: 11px;` before the `@import "squares/*.png";` statement.)
|
%q(The first argument to sprite-url must be a sprite. See http://beta.compass-style.org/help/tutorials/spriting/ for more information.)
|
||||||
proc do
|
|
||||||
render <<-SCSS
|
|
||||||
@import "squares/*.png";
|
|
||||||
|
|
||||||
.squares {
|
|
||||||
background: sprite-position("squares/twenty-by-twenty.png", 0, 0, 11px) no-repeat;
|
|
||||||
}
|
|
||||||
SCSS
|
|
||||||
end.should raise_error Compass::Error,
|
|
||||||
%q(Spacing parameter is deprecated. Please add `$squares-twenty-by-twenty-spacing: 11px;` before the `@import "squares/*.png";` statement.)
|
|
||||||
end
|
|
||||||
|
|
||||||
it "should raise an error if @import is missing" do
|
|
||||||
proc do
|
proc do
|
||||||
render <<-SCSS
|
render <<-SCSS
|
||||||
.squares {
|
.squares {
|
||||||
background: sprite-image("squares/twenty-by-twenty.png") no-repeat;
|
background: sprite-image("squares/twenty-by-twenty.png") no-repeat;
|
||||||
}
|
}
|
||||||
SCSS
|
SCSS
|
||||||
end.should raise_error Compass::Error,
|
end.should raise_error Sass::SyntaxError,
|
||||||
%q(`@import` statement missing. Please add `@import "squares/*.png";`.)
|
%q(The first argument to sprite-image must be a sprite. See http://beta.compass-style.org/help/tutorials/spriting/ for more information.)
|
||||||
|
proc do
|
||||||
|
render <<-SCSS
|
||||||
|
@import "squares/*.png";
|
||||||
|
|
||||||
|
.squares {
|
||||||
|
background: sprite-position("squares/twenty-by-twenty.png") no-repeat;
|
||||||
|
}
|
||||||
|
SCSS
|
||||||
|
end.should raise_error Sass::SyntaxError,
|
||||||
|
%q(The first argument to sprite-position must be a sprite. See http://beta.compass-style.org/help/tutorials/spriting/ for more information.)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "should work even if @import is missing" do
|
||||||
|
actual_css = render <<-SCSS
|
||||||
|
.squares {
|
||||||
|
background: sprite-image(sprite("squares/*.png"), twenty-by-twenty) no-repeat;
|
||||||
|
}
|
||||||
|
SCSS
|
||||||
|
actual_css.should == <<-CSS
|
||||||
|
.squares {
|
||||||
|
background: url('/squares.png') 0 -10px no-repeat;
|
||||||
|
}
|
||||||
|
CSS
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
Loading…
Reference in New Issue
Block a user