10

我正在编写一个库,其中包含通过 Thor 启动的嵌入式 Sinatra 应用程序。我想挂载Sprockets::Environmentat/css/js的实例,并将主应用程序映射到/. 这很容易Rack::URLMapconfig.ru文件中使用,但在这种情况下没有文件,因为我正在以编程方式启动 Sinatra 应用程序Sinatra::Application.run!。我怎样才能做到这一点?

4

3 回答 3

13

其实,这并不难。您需要做的就是将 的实例分配给Sprockets::EnvironmentSinatra 配置变量并定义一些路径来查找您感兴趣的资产。

这是一个基本示例:

require "sass"
require "haml"
require "erubis"
require "sinatra"
require "sprockets"

set :assets, Sprockets::Environment.new

# Configure sprockets
settings.assets.append_path "app/javascripts"
settings.assets.append_path "app/stylesheets"

# For compressed JS and CSS output
require "yui/compressor"
settings.assets.js_compressor  = YUI::JavaScriptCompressor.new
settings.assets.css_compressor = YUI::CssCompressor.new

get "/" do
  haml :index
end

get "/javascripts/:file.js" do
  content_type "application/javascript"
  settings.assets["#{params[:file]}.js"]
end

get "/stylesheets/:file.css" do
  content_type "text/css"
  settings.assets["#{params[:file]}.css"]
end

快乐链轮!

于 2012-05-20T01:50:05.223 回答
5

我最终通过编写一个具有Rack::URLMap. 它看起来大致是这样的:

require "sprockets"
require "sinatra/base"

class SprocketsMiddleware
  attr_reader :app, :prefix, :sprockets

  def initialize(app, prefix)
    @app = app
    @prefix = prefix
    @sprockets = Sprockets::Environment.new

    yield sprockets if block_given?
  end

  def call(env)
    path_info = env["PATH_INFO"]
    if path_info =~ prefix
      env["PATH_INFO"].sub!(prefix, "")
      sprockets.call(env)
    else
      app.call(env)
    end
  ensure
    env["PATH_INFO"] = path_info
  end
end

class App < Sinatra::Base
  use SprocketsMiddleware, %r{/assets} do |env|
    env.append_path "assets/css"
    env.append_path "assets/js"
  end
end

App.run!
于 2012-05-21T05:41:44.303 回答
2

以下是我如何将 Sprockets 集成到 Sinatra 中,并使用类似 Rails 的目录布局、帮助程序和 JS 和 CSS 的缩小。

我选择编写一个 Sinatra 扩展。此扩展封装了 sprockets 的配置(路径、缩小、助手),并且可以由应用程序注册。

module Sinatra
  module Assets
    extend Sinatra::Extension

    configure do
      set :assets, Sprockets::Environment.new(root).tap { |assets|
        %w(assets vendor/assets).each do |base|
          %w(images javascripts stylesheets).each do |type|
            assets.append_path File.join(base, type)
          end
        end
        if production?
          assets.js_compressor = Closure::Compiler.new
          assets.css_compressor = YUI::CssCompressor.new
          uid = Digest::MD5.hexdigest(File.dirname(__FILE__))[0,8]
          assets.cache = Sprockets::Cache::FileStore.new("/tmp/sinatra-#{uid}")
        else
          assets.cache = nil
        end
      }
    end

    get "/assets/*" do
      env["PATH_INFO"].sub!(%r{^/assets}, "")
      expires Time.now + (365*24*60*60) if settings.production?
      settings.assets.call(env)
    end

    helpers do
      include Sprockets::Helpers

      Sprockets::Helpers.configure do |config|
        config.expand = development?
        config.digest = production?
      end

      def assets_environment
        settings.assets
      end
    end
  end
end

在您的应用程序中使用扩展很简单:

class App < Sinatra::Base
  register Sinatra::Assets
  # ...
end

资产可以放置在assetsvendor/assets中。例如vendor/assets/jquery.js可以通过逻辑名称引用,即http://localhost/assets/jquery.js.

在上面的示例中,我使用了sprockets-helpers,它提供了诸如javascript_tag. 上面给出的配置假设在开发中,您希望扩展引用资产所需的资产(导致每个资产有多个标签)。

于 2013-01-08T09:40:17.600 回答