diff --git a/Gemfile b/Gemfile index 1ec9bada..c1f755d1 100644 --- a/Gemfile +++ b/Gemfile @@ -18,6 +18,7 @@ end if RUBY_VERSION >= "2.5.0" gem "rails", "~> 6.0" gem 'webrick' + gem "turbo-rails" if RUBY_VERSION >= "2.6.0" else gem "rails", "~> 5.0" end diff --git a/lib/draper/compatibility/broadcastable.rb b/lib/draper/compatibility/broadcastable.rb new file mode 100644 index 00000000..59f33f12 --- /dev/null +++ b/lib/draper/compatibility/broadcastable.rb @@ -0,0 +1,24 @@ +# frozen_string_literal: true + +module Draper + module Compatibility + # It would look consistent to use decorated objects inside templates broadcasted with + # Turbo::Broadcastable. + # + # This compatibility patch fixes the issue by overriding the original defaults to decorate the + # object, that's passed to the partial in a local variable. + module Broadcastable + private + + def broadcast_rendering_with_defaults(options) + return super unless decorator_class? + + # Add the decorated current instance into the locals (see original method for details). + options[:locals] = + (options[:locals] || {}).reverse_merge!(model_name.element.to_sym => decorate) + + super + end + end + end +end diff --git a/lib/draper/decoratable.rb b/lib/draper/decoratable.rb index 5a3aee39..3f72e74b 100644 --- a/lib/draper/decoratable.rb +++ b/lib/draper/decoratable.rb @@ -1,4 +1,5 @@ require 'draper/decoratable/equality' +require 'draper/compatibility/broadcastable' module Draper # Provides shortcuts to decorate objects directly, so you can do @@ -11,6 +12,10 @@ module Decoratable extend ActiveSupport::Concern include Draper::Decoratable::Equality + included do + prepend Draper::Compatibility::Broadcastable if defined? Turbo::Broadcastable + end + # Decorates the object using the inferred {#decorator_class}. # @param [Hash] options # see {Decorator#initialize} @@ -87,8 +92,6 @@ def decorator_class(called_on = self) def ===(other) super || (other.is_a?(Draper::Decorator) && super(other.object)) end - end - end end diff --git a/spec/dummy/app/models/post.rb b/spec/dummy/app/models/post.rb index 59b1f954..70c301f1 100644 --- a/spec/dummy/app/models/post.rb +++ b/spec/dummy/app/models/post.rb @@ -1,3 +1,5 @@ class Post < ApplicationRecord # attr_accessible :title, :body + + broadcasts if defined? Turbo::Broadcastable end diff --git a/spec/dummy/config/application.rb b/spec/dummy/config/application.rb index f0e1feaf..159f806c 100644 --- a/spec/dummy/config/application.rb +++ b/spec/dummy/config/application.rb @@ -9,6 +9,7 @@ def attempt_require(file) require 'draper' attempt_require 'mongoid' attempt_require 'devise' +attempt_require 'turbo-rails' require 'active_model_serializers' module Dummy diff --git a/spec/dummy/spec/models/post_spec.rb b/spec/dummy/spec/models/post_spec.rb index 1ddf8b56..8bc05a0f 100644 --- a/spec/dummy/spec/models/post_spec.rb +++ b/spec/dummy/spec/models/post_spec.rb @@ -5,4 +5,17 @@ it_behaves_like 'a decoratable model' it { should be_a ApplicationRecord } + + describe 'broadcasts' do + let(:modification) { described_class.create! } + + it 'passes a decorated object for rendering' do + expect do + modification + end.to have_enqueued_job(Turbo::Streams::ActionBroadcastJob).with { |stream, action:, target:, **rendering| + expect(rendering[:locals]).to include :post + expect(rendering[:locals][:post]).to be_decorated + } + end + end if defined? Turbo::Broadcastable end