Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Component presenter class #26

Closed
wants to merge 65 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
65 commits
Select commit Hold shift + click to select a range
28386a4
Initial presenter development
tombeynon Sep 17, 2015
8181e18
TODO: Dummy component class
tombeynon Sep 17, 2015
b8e8cba
Require component classes in dummy
tombeynon Sep 17, 2015
2e30bc5
Define attributes in class
tombeynon Sep 18, 2015
53fb802
Tests
tombeynon Sep 18, 2015
89eed1e
Readme update
tombeynon Sep 18, 2015
9da1b65
Added defaults example
tombeynon Oct 16, 2015
2ca764e
Moved component lookup to presenter class
tombeynon Oct 16, 2015
a7dadc3
Remove rails helpers
tombeynon Oct 16, 2015
861e9d1
Added convenience method for properties
tombeynon Oct 19, 2015
4dd2cb3
Rails autoload and .descendants doesn't work
tombeynon Oct 19, 2015
1a5727a
Attribute defaults should pre-populate hash
tombeynon Oct 19, 2015
a0adcec
Deep merge attribute defaults
tombeynon Oct 19, 2015
c7021f3
Public methods will be exposed to views
tombeynon Oct 19, 2015
683f9c2
Remove broken #attributes tests (now private)
tombeynon Oct 22, 2015
00f75d6
Handle autoloading in engine initializer
tombeynon Oct 22, 2015
c64c8e5
Fix Hound issues
tombeynon Oct 22, 2015
e0eab2d
Cleaned up attributes
tombeynon Oct 22, 2015
422a279
Single to double quotes
tombeynon Oct 22, 2015
ca3bf20
Redundant self
tombeynon Oct 22, 2015
bf12ab0
Line length
tombeynon Oct 22, 2015
728363a
Trailing whitespace, align params..
tombeynon Oct 22, 2015
df31920
Use camelize instead of classify
tombeynon Oct 26, 2015
383a056
Rename Attributes to Properties
tombeynon Oct 30, 2015
cdf1845
Better `component_for` method
tombeynon Oct 30, 2015
1cf37a4
Presenters now create a view context to render
tombeynon Oct 30, 2015
692f606
Update CardComponent
tombeynon Nov 2, 2015
070e4bf
Tidy presenter and define property methods
tombeynon Apr 14, 2016
17cafd5
Use controller view context over self
tombeynon Apr 14, 2016
b4f6748
Fix load issues
tombeynon Apr 14, 2016
a15192a
Expose component public methods
tombeynon Apr 14, 2016
9aa58a0
Tidy
tombeynon Apr 14, 2016
6036a8d
Some actual presenter tests
tombeynon Apr 14, 2016
4c24333
Initial presenter development
tombeynon Sep 17, 2015
286d77b
TODO: Dummy component class
tombeynon Sep 17, 2015
9882966
Require component classes in dummy
tombeynon Sep 17, 2015
84d614e
Define attributes in class
tombeynon Sep 18, 2015
bdb2c9c
Tests
tombeynon Sep 18, 2015
3dd5821
Readme update
tombeynon Sep 18, 2015
b46365a
Added defaults example
tombeynon Oct 16, 2015
dabb6c7
Moved component lookup to presenter class
tombeynon Oct 16, 2015
1c59089
Remove rails helpers
tombeynon Oct 16, 2015
9dcaabb
Added convenience method for properties
tombeynon Oct 19, 2015
baf1c08
Rails autoload and .descendants doesn't work
tombeynon Oct 19, 2015
897a5c4
Attribute defaults should pre-populate hash
tombeynon Oct 19, 2015
8bc298c
Deep merge attribute defaults
tombeynon Oct 19, 2015
16662fe
Public methods will be exposed to views
tombeynon Oct 19, 2015
13eeeaa
Remove broken #attributes tests (now private)
tombeynon Oct 22, 2015
636b6b2
Handle autoloading in engine initializer
tombeynon Oct 22, 2015
6502d0d
Fix Hound issues
tombeynon Oct 22, 2015
1ade24d
Cleaned up attributes
tombeynon Oct 22, 2015
c80984f
Single to double quotes
tombeynon Oct 22, 2015
e255d33
Redundant self
tombeynon Oct 22, 2015
682021a
Line length
tombeynon Oct 22, 2015
678f022
Trailing whitespace, align params..
tombeynon Oct 22, 2015
2176069
Use camelize instead of classify
tombeynon Oct 26, 2015
2a458b3
Rename Attributes to Properties
tombeynon Oct 30, 2015
0698f5f
Better `component_for` method
tombeynon Oct 30, 2015
33b208b
Merge pull request #4 from tombeynon/presenter-rendering
tombeynon Apr 15, 2016
c6e6ea2
Fix hound issues
tombeynon Apr 15, 2016
2449dcc
Update README with new presenter behaviour
tombeynon Apr 15, 2016
a7625d8
Use to_partial_path now we control the view context
tombeynon Apr 17, 2016
166906e
Protect Presenter public methods more reliably
tombeynon Apr 17, 2016
c0c9ea5
Attempt to fix Rails 3 error
tombeynon Apr 18, 2016
cde8031
Rollback to_partial_path work
tombeynon Apr 18, 2016
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
34 changes: 32 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ app/
header.css
header.js
header.yml
header_component.rb # optional
```

Keep in mind that you can also use `scss`, `coffeescript`, `haml`, or any other
Expand All @@ -60,11 +61,40 @@ coffee-script as long as you have these preprocessors running on your app.
```erb
<!-- app/components/header/_header.html.erb -->
<div class="header">
<h1>This is a header component with the title: <%= properties[:title] %></h1>
<h3>And subtitle <%= properties[:subtitle] %></h3>
<h1>This is a header component with the title: <%= title %></h1>
<h3>And subtitle <%= subtitle %></h3>
<% if show_links? %>
<ul>
<% links.each do |link| %>
<li><%= link %></li>
<% end %>
</ul>
<% end %>
</div>
```

```ruby
# app/components/header/header_component.rb
class HeaderComponent < MountainView::Presenter
properties :title, :subtitle
property :links, default: []

def title
properties[:title].titleize
end

def show_links?
links.any?
end
end
```

Including a component class is optional, but it helps avoid polluting your
views and helpers with presenter logic. Public methods in your component class
will be made available to the view, along with any properties you define.
You can also access all properties using the `properties` method in your
component class and views. You can even define property defaults.

### Using components on your views
You can then call your components on any view by using the following
helper:
Expand Down
3 changes: 2 additions & 1 deletion app/helpers/mountain_view/component_helper.rb
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
module MountainView
module ComponentHelper
def render_component(slug, properties = {})
render "#{slug}/#{slug}", properties: properties
component = MountainView::Presenter.component_for(slug, properties)
component.render(controller.view_context)
end
end
end
2 changes: 2 additions & 0 deletions lib/mountain_view.rb
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
require "mountain_view/version"
require "mountain_view/configuration"
require "mountain_view/presenter"
require "mountain_view/component"

module MountainView
def self.configuration
Expand Down
7 changes: 6 additions & 1 deletion lib/mountain_view/engine.rb
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
require "rails"
require "mountain_view/component"

module MountainView
class Engine < ::Rails::Engine
Expand All @@ -11,6 +10,12 @@ class Engine < ::Rails::Engine
end
end

initializer "mountain_view.load_component_classes",
before: :set_autoload_paths do |app|
component_paths = "#{MountainView.configuration.components_path}/{*}"
app.config.autoload_paths += Dir[component_paths]
end

initializer "mountain_view.assets" do |app|
Rails.application.config.assets.paths <<
MountainView.configuration.components_path
Expand Down
77 changes: 77 additions & 0 deletions lib/mountain_view/presenter.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
module MountainView
class Presenter
class_attribute :_properties, instance_accessor: false
self._properties = {}

attr_reader :slug, :properties

def initialize(slug, properties = {})
@slug = slug
@properties = default_properties.deep_merge(properties)
end

def render(context)
context.extend ViewContext
context.inject_component_context self
context.render partial: partial
end
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What do you think of naming this method to_partial_path? Ref: https://robots.thoughtbot.com/rendering-collections-in-rails

It's a bit more Rails-y, and that way we can use render component, locals: component.locals in the render_component method.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Very nice solution, hadn't even thought of that!

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, unfortunately Rails does something a bit special with to_partial_path. If the controller is namespaced (such as MountainView::StyleguideController) then it'll namespace the to_partial_path with the controller namespace, resulting in mountain_view/component/component/. See the following method..

https://github.com/rails/rails/blob/20559834e4959531f02f85cac433127c2cc52b20/actionview/lib/action_view/renderer/partial_renderer.rb#L482

The only way I can see to prevent this behaviour would be to override partial_path entirely. Not awful, but maybe overkill for what we're trying to achieve.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh, didn't know about that!
Yeah, overriding partial_path is definitively overkill. As is, partial, should be perfectly fine then.

Thanks for the research!


def partial
"#{slug}/#{slug}"
end

private

def default_properties
self.class._properties.inject({}) do |sum, (k, v)|
sum[k] = v[:default]
sum
end
end

class << self
def component_for(*args)
klass = "#{args.first.to_s.camelize}Component".safe_constantize
klass ||= self
klass.new(*args)
end

def properties(*args)
opts = args.extract_options!
properties = args.inject({}) do |sum, name|
sum[name] = opts
sum
end
define_property_methods(args)
self._properties = _properties.merge(properties)
end
alias_method :property, :properties

private
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Trailing whitespace detected.


def define_property_methods(names = [])
names.each do |name|
next if method_defined?(name)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Trailing whitespace detected.

define_method name do
properties[name.to_sym]
end
end
end
end

module ViewContext
attr_reader :_component
delegate :properties, to: :_component

def inject_component_context(component)
@_component = component
protected_methods = MountainView::Presenter.public_methods(false)
methods = component.public_methods(false) - protected_methods
methods.each do |meth|
next if self.class.method_defined?(meth)
self.class.delegate meth, to: :_component
end
end
end
end
end
7 changes: 4 additions & 3 deletions test/dummy/app/components/card/_card.html.erb
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,12 @@
<% end %>
<div class="card__content">
<h3 class="card__content__title">
<a href="<%= properties[:link] %>"><%= properties[:title] %></a>
<a href="<%= properties[:link] %>"><%= title %></a>
</h3>
<%- if properties[:description] %>
<%- if has_description? %>
<p><%= properties[:description] %></p>
<%- end %>
<p>Location: <%= location %></p>
<div class="card__content__data">
<%- if properties[:data] && properties[:data].any? %>
<%- properties[:data].each do |data| %>
Expand All @@ -23,4 +24,4 @@
<% end %>
</div>
</div>
</div>
</div>
6 changes: 4 additions & 2 deletions test/dummy/app/components/card/card.yml
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
-
:title: "Aspen Snowmass"
:title: "Snowmass"
:location: "Aspen"
:description: "Aspen Snowmass is a winter resort complex located in Pitkin County in western Colorado in the United States. Owned and operated by the Aspen Skiing Company it comprises four skiing/snowboarding areas on four adjacent mountains in the vicinity of the towns of Aspen and Snowmass Village."
:link: "http://google.com"
:image_url: "http://i.imgur.com/QzuIJTo.jpg"
Expand All @@ -13,7 +14,8 @@


-
:title: "Breckenridge, Colorado"
:title: "Breckenridge"
:location: "Colorado"
:link: "http://facebook.com"
:image_url: "http://i.imgur.com/w7ZyWPg.jpg"
:data:
Expand Down
14 changes: 14 additions & 0 deletions test/dummy/app/components/card/card_component.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
class CardComponent < MountainView::Presenter
include ActionView::Helpers::TagHelper

properties :title, :description, :link, :image_url, :location
property :data, default: []

def title
[location, properties[:title]].compact.join(", ")
end

def has_description?
description.present?
end
end
38 changes: 38 additions & 0 deletions test/mountain_view/presenter_test.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
require "test_helper"

class InheritedPresenter < MountainView::Presenter
properties :title, :description
property :data, default: []

def title
"Foo#{properties[:title].downcase}"
end
end

class MountainView::PresenterTest < ActiveSupport::TestCase
test "returns the correct partial path" do
presenter = MountainView::Presenter.new("header")
assert_equal "header/header", presenter.partial
end

test "exposes properties as provided" do
properties = {foo: "bar", hello: "world"}
presenter = MountainView::Presenter.new("header", properties)
assert_equal properties, presenter.properties
end

test "inherited presenter returns the correct title" do
presenter = InheritedPresenter.new("inherited", title: "Bar")
assert_equal "Foobar", presenter.title
end

test "inherited presenter responds to #data" do
presenter = InheritedPresenter.new("inherited", data: ["Foobar"])
assert_equal ["Foobar"], presenter.data
end

test "inherited presenter returns the default value for #data" do
presenter = InheritedPresenter.new("inherited", {})
assert_equal [], presenter.data
end
end