How to use the “number_to_currency” helper method in the model rather than view?

后端 未结 11 1530
有刺的猬
有刺的猬 2021-01-30 03:59

I would like to use to_dollar method in my model like this:

module JobsHelper      
  def to_dollar(amount)
    if amount < 0
      number_to_cur         


        
相关标签:
11条回答
  • 2021-01-30 04:23

    Helper methods are generally used for View files. It is not a good practice to use these methods in Model class. But if you want to use then Sam's answer is ok. OR I suggest you can write your own custom method.

    0 讨论(0)
  • 2021-01-30 04:29

    @fguillen's way is good, though here's a slightly cleaner approach, particular given that the question makes two references to to_dollar. I'll first demonstrate using Ryan Bates' code (http://railscasts.com/episodes/132-helpers-outside-views).

    def description
      "This category has #{helpers.pluralize(products.count, 'product')}."
    end
    
    def helpers
      ActionController::Base.helpers
    end
    

    Notice the call helpers.pluralize. This is possible due to the method definition (def helpers), which simply returns ActionController::Base.helpers. Therefore helpers.pluralize is short for ActionController::Base.helpers.pluralize. Now you can use helpers.pluralize multiple times, without repeating the long module paths.

    So I suppose the answer to this particular question could be:

    class Job < ActiveRecord::Base
      include JobsHelper
      def details
        return "Only " + helpers.to_dollar(part_amount_received) + 
               " out of " + helpers.to_dollar(price) + " received."
      end
    
      def helpers
        ActionView::Helpers::NumberHelper
      end
    end
    
    0 讨论(0)
  • 2021-01-30 04:31

    It’s not available because its use in a model (typically) violates MVC (and it does seem to in your case). You're taking data and manipulating it for presentation. This, by definition, belongs in the view, not the model.

    Here are some solutions:

    • Use a presenter or view model object to mediate between the model and view. This almost definitely requires more initial work than other solutions, but is almost always a better design. Using helpers in a presenter/view-model doesn’t violate MVC, as they reside in the view layer, replacing traditional custom Rails helpers and logic-filled views.

    • Explicitly include ActionView::Helpers::NumberHelper in JobsHelper instead of depending on Rails to have magically loaded it for you. This is still not great, as you shouldn’t access a helper from a model.

    • Violate MVC & SRP. See fguillen’s answer for how to do this. I won’t echo it here because I don’t agree with it. Even more so, though, do I disagree with polluting your model with presentation methods as in Sam’s answer.

    If you think “but I really need this to write my to_csv & to_pdf methods in my model!”, then your entire premise is wrong—after all, you don’t have a to_html method, do you? And yet your object is very often rendered as HTML. Consider creating a new class for generating your output instead of making your data model know what a CSV is (because it shouldn’t).

    As for using helpers for ActiveModel validation errors in the model, well, I’m sorry but ActiveModel/Rails has screwed us all there by forcing error messages to be realized in the data layer, rather than returning the semantic idea of an error to be realized later—sigh. You can get around this, but it basically means not using ActiveModel::Errors anymore. I’ve done it, it works well.

    As an aside, here’s a useful way to include helpers in a presenter/view-model without polluting its set of methods (because being able to do e.g. MyPresenterOrViewModel.new.link_to(...) makes no sense):

    class MyPresenterOrViewModel
      def some_field
        helper.number_to_currency(amount, :precision => 0)
      end
    
      private
    
      def helper
        @helper ||= Class.new do
          include ActionView::Helpers::NumberHelper
        end.new
      end
    end
    
    0 讨论(0)
  • 2021-01-30 04:31

    I agree with all of you that this could be breaking the MVC pattern but there is always reasons to break a pattern, in my case I needed these currency formatter methods to use them in a template filter (Liquid in my case).

    At the end I found out I could access to these currency formatter methods using things like this:

    ActionController::Base.helpers.number_to_currency
    
    0 讨论(0)
  • 2021-01-30 04:31

    You need to also include the ActionView::Helpers::NumberHelper

    class Job < ActiveRecord::Base
      include ActionView::Helpers::NumberHelper
      include JobsHelper
      def details
        return "Only " + to_dollar(part_amount_received) + 
               " out of " + to_dollar(price) + " received."
      end
    end
    
    0 讨论(0)
提交回复
热议问题