问题描述
我想编写一个简单的 erb 模板生成器,以使用 Generator
模块从视图中解析存储的 erb 模板。我从 rails 控制器调用 Generator
以生成它的单例实例,并通过 WallController
指针将 self
传递给它。
require 'generator'
class WallController < ApplicationController
def index
header = File.read 'app/views/application-header.html'.freeze
@instances = {header: header}
# Load view generators
Generator.generate_instances self
end
end
Generator.generate_instances
实际上尝试做的第一件事是复制 WallController
实例变量(因此是 self 指针)以执行 erb 模板的正确解析。然后它生成返回 erb 结果文本的方法。
require 'erb'
module Generator
def self.generate_instances environment
# Mimic class environment
if environment.superclass == ApplicationController
environment.instance_variables.each do |v|
puts "copy instance variable '#{v}' from #{environment.name} to #{self.name}"
value = environment.instance_variable_get(v)
self.send :instance_variable_set,v,value
end
end
# Parse the ERB templates
templates = @instances
return 0 if !templates.is_a?(Hash) or templates.empty?
templates.keys.each.with_index do |key,index|
define_singleton_method key do
ERB.new(templates.values[index]).result
end
end
end
end
Generator
界面的用法如下所示:
<%=== Generator.header %>
我是 rails 新手,但我发现 rails 控制器的包含文件仅限于单个静态结构。我没有设法覆盖可能有用的 class Object
或 class Class
单例方法。
但是,在运行上述示例后,WallController
的实例变量返回 WallController
类地址而不是 WallController.index
定义的值。
undefined method `empty?' for #<WallController:0x000000000a1f90>
是否有在其他控制器之间分配 rails 控制器实例变量的正确方法?如果没有,为什么常规实例复制不起作用?
如果我必须用 ruby 编写它,那会很容易:
module Y
def self.generate_environment environment
environment.instance_variables.each do |v|
puts "copy #{v} from #{environment.name} to #{self.name}"
value = environment.instance_variable_get v
self.instance_variable_set(v,value)
end if environment.class == Class
puts "Retrived string: #{@hello}"
end
end
class X
def self.index
@hello = 'Hello,World!'
Y.generate_environment self
end
end
X.index
解决方法
这个问题可以用 viewcomponent 解决,它允许视图控制器使用标准的 ruby 代码。也解决了在合理的速度下将视图代码划分为更小的可重用组件的问题。
要使用视图组件 gem,首先将它包含到您的 Gemfile 中。
gem 'view_component',require: 'view_component/engine'
使用 bundle install
更新 gem 后,如果服务器正在运行,您还需要重新启动服务器,以应用新 gem。
然后生成组件的用法和其他rails生成器类似。第一个参数是组件名称,第二个参数是组件参数。
rails generate component Header site_id
现在我专注于 app/component
目录中生成的文件、视图和控制器代码。这将只是创建视图标题片段的控制器。
app/component/header_component.rb
内部可以封装WallController
中所有与header view相关的代码。
class HeaderComponent < ViewComponent::Base
def initialize(site_id:)
puts "Rendering header component for site: #{site_id}"
# Load site elements
@site = Site.find site_id
@menu_items = []
Site.all.each.with_index do |site,index|
@menu_items.push site.title => site.link
end
end
end
同理,将所有的header view erb代码放到app/component/header.html.erb
中。
可以使用 rails render 从视图中生成完成的组件:
<%= render HeaderComponent.new(site_id: 1) %>