module Lolita
module Adapter
module CommonHelper
class Record
def initialize(adapter, orm_record)
@adapter = adapter
@record = orm_record
end
def title
if @record.respond_to?(:title)
@record.title
elsif @record.respond_to?(:name)
@record.name
elsif content_field = @adapter.fields.detect{|f| f.type.to_s=="string"}
@record.send(content_field.name)
else
"#{@record.class.lolita_model_name.human} #{@record.id}"
end
end
end
class PaginationBuilder
def initialize adapter,page,per,options
@adapter = adapter
@page = page
@per = per
@options = options || {}
end
def params
request.respond_to?(:params) && request.params || {}
end
def request
@options[:request]
end
def nested_criteria
nested_hsh = params[:nested]
if params[:nested] && !params[:nested][:association]
nested_hsh = nested_hsh.reject{|k,v| [:parent,:path].include?(k.to_sym)}
@adapter.klass.where(nested_hsh)
else
@adapter.klass.unscoped
end
end
def ability_criteria
if @adapter.klass.respond_to?(:accessible_by)
@adapter.klass.accessible_by(current_ability)
else
@adapter.klass.unscoped
end
end
def relation
if params[:nested] && params[:nested][:association]
@adapter.find(hsh[:nested][:id]).send(hsh[:nested][:association])
else
@adapter.klass.unscoped
end
end
def custom_criteria
if @options[:pagination_method]
if @options[:pagination_method].respond_to?(:each)
@options[:pagination_method].each do |method_name|
@options[:previous_scope] = scope
if new_criteria = pagination_criteria_for_klass(method_name,@page,@per,@options)
@custom_criteria = @custom_criteria ? @custom_criteria.merge(new_criteria) : new_criteria
end
end
else
@custom_criteria = pagination_scope_for_klass(@options[:pagination_method],@page,@per,@options)
end
unless @custom_criteria
raise ArgumentError, "Didn't generate any scope from #{@options} page:{page} per:#{@per}"
else
@custom_criteria
end
else
@adapter.klass.unscoped
end
end
def pagination_scope_for_klass(method_name,page,per,options)
if @adapter.klass.respond_to?(method_name)
@adapter.klass.send(method_name,page,per,options)
end
end
def create_page
page_criteria = relation.merge(nested_criteria).merge(ability_criteria).merge(custom_criteria)
unless page_criteria.respond_to?(:current_page)
page_criteria = page_criteria.order(sorting).page(@page).per(@per)
end
page_criteria
end
def current_ability
controller = request.headers["action_controller.instance"]
if controller && controller.respond_to?(:current_ability)
controller.current_ability
end
end
def sorting
params[:s] ? params[:s].gsub(',',' ').gsub('|',',') : nil
end
end
def record(orm_record)
Record.new(self,orm_record)
end
# Return all association class names
def associations_class_names
self.associations.map{|name,association|
association.class_name
}
end
def association_by_klass(given_klass)
associations.select{|name,association|
association.klass == given_klass
}.values.first
end
def filter attributes={}
klass.where(attributes.reject{|k,v| v.blank? })
end
# Detect if class reflect on association by name
def reflect_on_association(name)
if orm_association = klass.reflect_on_association(name)
self.class.const_get(:Association).new(orm_association,self)
end
end
def by_id(id)
klass.where(klass.primary_key => id)
end
def find_by_id(id)
self.klass.unscoped.merge(by_id(id)).first
end
# This method is used to paginate, main reason is for list and for index action.
# Method accepts three arguments
# page - page that should be shown (integer)
# per - how many records there should be in page
# options - Hash with optional information.
# By default, Lolita::Configuration::List passes request, with current request information.
# Also it passes :pagination_method that is used to detect if there is special method(-s) in model
# that should be used for creating page.
def paginate(page, per, options = {})
pagination_builder = PaginationBuilder.new(self, page, per, options)
pagination_builder.create_page
end
def switch_record_state(record, state = nil)
set_state_for(record)
if state
record.send(:"#{state}_state!")
elsif !record.have_state?
if record.new_record?
record.create_state!
else
record.update_state!
end
end
record
end
def set_state_for(record)
unless record.respond_to?(:read_state!)
class << record
def set_state(new_state)
@state_set = true
@state = new_state
end
def have_state?
@state_set
end
def read_state!
set_state :read
end
def create_state!
set_state :create
end
def update_state!
set_state :update
end
def state
set_state(:read) unless @state
@state
end
def in_read_state?
state == :read
end
def in_create_state?
state == :create
end
def in_update_state?
state == :update
end
end
end
record
end
end
end
end