Slide 1

Slide 1 text

Presenters and Decorators A Code Tour

Slide 2

Slide 2 text

No content

Slide 3

Slide 3 text

Mike Moore

Slide 4

Slide 4 text

@blowmage

Slide 5

Slide 5 text

Disclaimer

Slide 6

Slide 6 text

Presenters should ease PAIN

Slide 7

Slide 7 text

Components...

Slide 8

Slide 8 text

<%= render_component(@component) %> app/views/dashboard/index.html.erb

Slide 9

Slide 9 text

def render_component(component) render :partial => "shared/component", :locals => {:component => component} end app/controllers/application_controller.rb

Slide 10

Slide 10 text

<%= render_component(component) %> app/views/shared/_component.html.erb

Slide 11

Slide 11 text

module ComponentHelper def render_component(component) return "" if component.nil? if component.respond_to?(:render) raw component.render(self) else raw component.to_s end end end app/helpers/component_helper.rb

Slide 12

Slide 12 text

class DashboardController < ApplicationController before_filter :require_user def index @component = DashboardComponentBuilder.build(current_user) end end app/controller/dashboard_controller.rb

Slide 13

Slide 13 text

class DashboardComponentBuilder def self.build(user) my_posts = user.posts.recent DashboardComponent.new( :post_items => PostItemComponentBuilder.build_list(my_posts), :author_stats_component => AuthorAnalyticsComponentBuilder.build(user) ) end end lib/dashboard_component_builder.rb

Slide 14

Slide 14 text

class PostItemComponentBuilder < ComponentBuilder def self.build(post) post_path = UrlGenerator.post_path(post) PostItemComponent.new( :title_link => Link.new(post.name, post_path), :description => post.description, :views => post.view_count, :post_thumbnail => PostThumbnailComponentBuilder.build(post), :comments => post.comments.published.count, :edit_link => Link.new("Edit Post", UrlGenerator.edit_post_path(post)), :status_class => build_status_class(post), :status_text => build_status_text(post), :delete_button => DeleteButtonComponentBuilder.build(post) ) end #... end lib/post_item_component_builder.rb

Slide 15

Slide 15 text

class ComponentBuilder def self.build_list(items,*args) return [] if items.nil? or items.empty? if self.respond_to?(:build) items.map do |item| self.build(item,*args) end.compact else raise "ComponentBuilder must implement .build" end end end app/components/component_builder.rb

Slide 16

Slide 16 text

class PostItemComponentBuilder < ComponentBuilder def self.build(post) post_path = UrlGenerator.post_path(post) PostItemComponent.new( :title_link => Link.new(post.name, post_path), :description => post.description, :views => post.view_count, :post_thumbnail => PostThumbnailComponentBuilder.build(post), :comments => post.comments.published.count, :edit_link => Link.new("Edit Post", UrlGenerator.edit_post_path(post)), :status_class => build_status_class(post), :status_text => build_status_text(post), :delete_button => DeleteButtonComponentBuilder.build(post) ) end #... end lib/post_item_component_builder.rb

Slide 17

Slide 17 text

PostItemComponent = PartialComponent.define( "posts/post_item", :title_link, :description, :views, :comments, :delete_button, :edit_link, :post_thumbnail, :status_class, :status_text) app/components/post_item_component.rb

Slide 18

Slide 18 text

class PartialComponent def self.define(partial, *var_names) component_class = ConstructorStruct.new( *[var_names + [:partial]].flatten) component_class.send(:define_method, :locals) do locals = {} var_names.each do |k| locals[k] = instance_variable_get("@#{k}") end locals end component_class.send(:define_method, :render) do |template| template.send(:render, :partial => @partial || partial, :locals => locals) end component_class end end app/components/partial_component.rb

Slide 19

Slide 19 text

PostItemComponent = PartialComponent.define( "posts/post_item", :title_link, :description, :views, :comments, :delete_button, :edit_link, :post_thumbnail, :status_class, :status_text) app/components/post_item_component.rb

Slide 20

Slide 20 text

.post_item{ :class => status_class } = render_component(post_thumbnail) .text %h5 - if status_text %span= status_text = render_link(title_link) .description= description .stats_and_actions .item_stats %p.views.stat_display %strong.stat_count= views Views %p.comments.stat_display %strong.stat_count= comments Comments .item_actions .edit %a.edit{:href => edit_link.url} %span Edit .delete = render_component(delete_button) app/views/posts/_post_item.html.haml

Slide 21

Slide 21 text

Welcome to the team!

Slide 22

Slide 22 text

Won’t somebody please think of the new developers?!?!

Slide 23

Slide 23 text

View ⬌ Bikeshed

Slide 24

Slide 24 text

Typical Summary Example...

Slide 25

Slide 25 text

<% published_count = current_user.posts.published.count + current_user.videos.published.count %> <% if published_count > 0 %>
  • Published: <%= published_count %>
  • <% end %> <% draft_count = current_user.posts.unpublished.count + current_user.videos.unpublished.count %> <% if draft_count > 0 %>
  • Drafted: <%= draft_count %>
  • <% end %> app/shared/nav/_author_details.html.erb

    Slide 26

    Slide 26 text

    <% if current_user.admin? %> <% approval_count = current_user.organization.posts.unapproved.count %> <% if approval_count.to_i > 0 %>
  • Needing Approval: <%= approval_count %>
  • <% end %> <% end %> app/shared/nav/_author_details.html.erb (cont)

    Slide 27

    Slide 27 text

    Solutions?

    Slide 28

    Slide 28 text

    class User < ActiveRecord::Base def published_count self.posts.published.count + self.videos.published.count end def draft_count self.posts.unpublished.count + self.videos.unpublished.count end def approval_count if self.admin? self.organization.posts.unapproved.count else 0 end end # ... end app/models/user.rb

    Slide 29

    Slide 29 text

    class AuthorSummary def initialize(author) @author = author end def published_count @author.posts.published.count + @author.videos.published.count end def draft_count @author.posts.unpublished.count + @author.videos.unpublished.count end def approval_count if @author.admin? @author.organization.posts.unapproved.count else 0 end end end lib/author_summary.rb

    Slide 30

    Slide 30 text

    <% summary = AuthorSummary.new(current_user) %> <% if summary.published_count > 0 %>
  • Published: <%= summary.published_count %>
  • <% end %> <% if summary.draft_count > 0 %>
  • Drafted: <%= summary.draft_count %>
  • <% end %> <% if summary.approval_count > 0 %>
  • Needing Approval: <%= summary.approval_count %>
  • <% end %> app/shared/nav/_author_details.html.erb

    Slide 31

    Slide 31 text

    ActiveDecorator github.com/amatsuda/active_decorator

    Slide 32

    Slide 32 text

    module UserDecorator def published_count self.posts.published.count + self.videos.published.count end def draft_count self.posts.unpublished.count + self.videos.unpublished.count end def approval_count if self.admin? self.organization.posts.unapproved.count else 0 end end end app/decorators/user_decorator

    Slide 33

    Slide 33 text

    <% if current_user.published_count > 0 %>
  • Published: <%= current_user.published_count %>
  • <% end %> <% if current_user.draft_count > 0 %>
  • Drafted: <%= current_user.draft_count %>
  • <% end %> <% if current_user.approval_count > 0 %>
  • Needing Approval: <%= current_user.approval_count %>
  • <% end %> app/shared/nav/_author_details.html.erb

    Slide 34

    Slide 34 text

    It helped!

    Slide 35

    Slide 35 text

    Typical Serialization Example...

    Slide 36

    Slide 36 text

    class PostController < ApplicationController before_filter :require_user def show @post = current_organization.posts.find(params[:id]) respond_to do |format| format.html format.json { render :json => @post } end end end app/controllers/posts_controller.rb

    Slide 37

    Slide 37 text

    object @post attributes :id, :title # look up author_name on the model, but use author in the JSON attributes :author_name => :author if current_user.admin? # only show views, popularity to admins attributes :view_count => :views, :popularity_index => :popularity end app/views/posts/show.json.rabl

    Slide 38

    Slide 38 text

    class PostController < ApplicationController before_filter :require_user def show @post = current_organization.posts.find(params[:id]) respond_to do |format| format.html format.json end end end app/controllers/posts_controller.rb

    Slide 39

    Slide 39 text

    class PostSerializer def initialize(post, user) @post, @user = post, user end def to_json if @user && @user.admin? full_detail else summary end end def full_detail summary.merge({ :views => @post.view_count, :popularity => @post.popularity_index }) end def summary { :id => @post.id, :title => @post.title, :author => @post.author_name } end end lib/post_serializer.rb

    Slide 40

    Slide 40 text

    class PostController < ApplicationController before_filter :require_user def show @post = current_organization.posts.find(params[:id]) respond_to do |format| format.html format.json { s = PostSerializer.new(@post, current_user) render :json => s.to_json } end end end app/controllers/posts_controller.rb

    Slide 41

    Slide 41 text

    ActiveModel:: Serializers github.com/josevalim/ active_model_serializers

    Slide 42

    Slide 42 text

    class PostSerializer < ActiveModel::Serializer attributes :id, :title # look up author_name on the model, but use author in the JSON attribute :author_name, :key => :author # @scoped is set to current_user by default if @scoped.admin? attribute :view_count, :key => :views attribute :popularity_index, :key => :popularity end end app/serializers/post_serializer.rb

    Slide 43

    Slide 43 text

    class PostController < ApplicationController before_filter :require_user def show @post = current_organization.posts.find(params[:id]) respond_to do |format| format.html format.json { render :json => @post } end end end app/controllers/posts_controller.rb

    Slide 44

    Slide 44 text

    Pretty cool!

    Slide 45

    Slide 45 text

    But...

    Slide 46

    Slide 46 text

    Is that it?

    Slide 47

    Slide 47 text

    Presenter Definition?

    Slide 48

    Slide 48 text

    Design Patterns

    Slide 49

    Slide 49 text

    Decorator Design Pattern Attach additional responsibilities to an object dynamically. Decorators provide a flexible alternative to subclassing for extending functionality.

    Slide 50

    Slide 50 text

    Mediator Design Pattern Define an object that encapsulates how a set of objects interact. Mediator promotes loose coupling by keeping objects from referring to each other explicitly, and lets you vary their interaction independently.

    Slide 51

    Slide 51 text

    Presenter Spectrum

    Slide 52

    Slide 52 text

    Presenter Spectrum Models Views Decorators Presenters?

    Slide 53

    Slide 53 text

    Rails: Presenter Pattern blog.jayfields.com/2007/03/rails- presenter-pattern.html

    Slide 54

    Slide 54 text

    Presentation Model martinfowler.com/eaaDev/ PresentationModel.html

    Slide 55

    Slide 55 text

    Presentation Model Presentation Model may interact with several domain objects, but Presentation Model is not a GUI friendly facade to a specific domain object...

    Slide 56

    Slide 56 text

    Presentation Model Instead it is easier to consider Presentation Model as an abstract of the view that is not dependent on a specific GUI framework...

    Slide 57

    Slide 57 text

    Presentation Model While several views can utilize the same Presentation Model, each view should require only one Presentation Model...

    Slide 58

    Slide 58 text

    Presentation Model In the case of composition a Presentation Model may contain one or many child Presentation Model instances, but each child control will also have only one Presentation Model.

    Slide 59

    Slide 59 text

    Definition?

    Slide 60

    Slide 60 text

    “Representation of the State of the View”

    Slide 61

    Slide 61 text

    Presenter == Presentation Model?

    Slide 62

    Slide 62 text

    ViewModel == Presentation Model

    Slide 63

    Slide 63 text

    A Slightly Better Example...

    Slide 64

    Slide 64 text

    <% if @course.available? && @course.self_enrollment && @course.open_enrollment && (!@course_enrollment || !@course_enrollment.active?) && !session["role_course_#{@course.id}"] %> <%= render :partial => "join_course", :object => @course %> <% elsif @course_enrollment && @course_enrollment.self_enrolled && @course_enrollment.active? && (!session["role_course_#{@course.id}"]) %> <%= render :partial => "drop_course", :locals => { :course => @course, :enrollment => @course_enrollment } %> <% elsif temp_type = session["role_course_#{@course.id}"] %> <%= render :partial => "temp_access", :object => temp_type %> <% end %> app/views/course.html.erb

    Slide 65

    Slide 65 text

    <% if @view.show_join_course? %> <%= render :partial => "join_course", :object => @view.course %> <% elsif @view.show_drop_course? %> <%= render :partial => "drop_course", :locals => { :course => @view.course, :enrollment => @view.enrollment } %> <% elsif @view.show_temp_access? %> <%= render :partial => "temp_access", :object => @view.temp_access %> <% end %> app/views/course.html.erb

    Slide 66

    Slide 66 text

    class CourseShowPresenter attr_accessor :course, :enrollment, :user, :session def show_join_course? @course.available? && @course.self_enrollment && @course.open_enrollment && (!@enrollment || !@enrollment.active?) && !show_temp_access? end def show_drop_course? @enrollment && @enrollment.self_enrolled && @enrollment.active? && !show_temp_access? end def show_temp_access? @session["role_course_#{@course.id}"] end alias :temp_access, :show_temp_access? end app/presenters/course_show_presenter.rb

    Slide 67

    Slide 67 text

    class CourseShowPresenter attr_accessor :course, :enrollment, :user, :session def show_join_course? @course.can_user_join?(@user) && !show_temp_access? end def show_drop_course? @course.can_user_drop?(@user) && !show_temp_access? end def show_temp_access? @session["role_course_#{@course.id}"] end alias :temp_access, :show_temp_access? end app/presenters/course_show_presenter.rb

    Slide 68

    Slide 68 text

    <% if @view.show_join_course? %> <%= render :partial => "join_course", :object => @view.course %> <% elsif @view.show_drop_course? %> <%= render :partial => "drop_course", :locals => { :course => @view.course, :enrollment => @view.enrollment } %> <% elsif @view.show_temp_access? %> <%= render :partial => "temp_access", :object => @view.temp_access %> <% end %> app/views/course.html.erb

    Slide 69

    Slide 69 text

    A Good Example...

    Slide 70

    Slide 70 text

    No content

    Slide 71

    Slide 71 text

    No content

    Slide 72

    Slide 72 text

    <% if current_organization.feature?(:categories) %> <% if current_site.setting?(:categories) %> <% if user_logged_in? || current_site.setting?(:show_all_content) %> categories = current_site.top_categories <% else %> categories = current_site.top_public_categories <% end %> <% if categories.any? %>
    Top Categories
      <%= render :partial => "category_item", :collection => categories %>
    <% end %> <% end %> <% end %> app/views/home/_sidebar.html.erb

    Slide 73

    Slide 73 text

    class HomepagePresenter attr_accessor :organization, :site, :user # Categories def show_categories? @organization.feature?(:categories) && @site.setting?(:categories) && self.categories.any? end def categories if @user.present? || @site.setting?(:show_all_content) @categories ||= @organization.top_categories else @categories ||= @organization.top_public_categories end end #... end app/presenters/homepage_presenter.rb

    Slide 74

    Slide 74 text

    <% if current_organization.feature?(:categories) %> <% if current_site.setting?(:categories) %> <% if user_logged_in? || current_site.setting?(:show_all_content) %> categories = current_site.top_categories <% else %> categories = current_site.top_public_categories <% end %> <% if categories.any? %>
    Top Categories
      <%= render :partial => "category_item", :collection => categories %>
    <% end %> <% end %> <% end %> app/views/home/_sidebar.html.erb

    Slide 75

    Slide 75 text

    <% if @view.show_categories? %>
    Top Categories
      <%= render :partial => "category_item", :collection => @view.categories %>
    <% end %> app/views/home/_sidebar.html.erb

    Slide 76

    Slide 76 text

    class HomepagePresenter attr_accessor :organization, :site, :user # Categories def show_categories? @organization.feature?(:categories) && @site.setting?(:categories) && self.categories.any? end def categories if @user.present? || @site.setting?(:show_all_content) @categories ||= @organization.top_categories else @categories ||= @organization.top_public_categories end end #... end app/presenters/homepage_presenter.rb

    Slide 77

    Slide 77 text

    class HomepagePresenter #... def categories if show_all_content? @categories ||= @organization.top_categories else @categories ||= @organization.top_public_categories end end protected def show_all_content? @user.present? || @site.setting?(:show_all_content) end #... end app/presenters/homepage_presenter.rb

    Slide 78

    Slide 78 text

    # Presenter when user is logged in class HomepageUserPresenter #... end # Presenter when site shows only public content class HomepagePublicPresenter #... end # Presenter when site shows all content, all the time class HomepagePermissivePresenter #... end app/presenters/homepage_*_presenter.rb

    Slide 79

    Slide 79 text

    class HomepageController < ApplicationController def index @view = build_homepage_presenter respond_to do |format| format.html end end def build_homepage_presenter view = if user_logged_in? HomepageUserPresenter.new elsif current_site.setting?(:show_all_content) HomepagePermissivePresenter.new else HomepagePublicPresenter.new end view.organization = current_organization view.site = current_site view.user = current_user view end end app/controllers/homepage_controller.rb

    Slide 80

    Slide 80 text

    Warning!

    Slide 81

    Slide 81 text

    No content

    Slide 82

    Slide 82 text

    No content

    Slide 83

    Slide 83 text

    No content

    Slide 84

    Slide 84 text

    Thank you!

    Slide 85

    Slide 85 text

    HIRE ME!!! blowmage.com Questions?