Class: Volt::AttributeBinding

Inherits:
BaseBinding show all
Defined in:
lib/volt/page/bindings/attribute_binding.rb

Instance Attribute Summary

Attributes inherited from BaseBinding

#binding_name, #context, #target

Instance Method Summary collapse

Methods inherited from BaseBinding

#dom_section

Constructor Details

#initialize(page, target, context, binding_name, attribute_name, getter, setter) ⇒ AttributeBinding

Returns a new instance of AttributeBinding.



6
7
8
9
10
11
12
13
14
# File 'lib/volt/page/bindings/attribute_binding.rb', line 6

def initialize(page, target, context, binding_name, attribute_name, getter, setter)
  super(page, target, context, binding_name)

  @attribute_name = attribute_name
  @getter         = getter
  @setter         = setter

  setup
end

Instance Method Details

#changed(event = nil) ⇒ Object



41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
# File 'lib/volt/page/bindings/attribute_binding.rb', line 41

def changed(event = nil)
  case @attribute_name
    when 'value'
      current_value = element.value
    else
      current_value = element.is(':checked')
  end

  if @is_radio
    if current_value
      # if it is a radio button and its checked
      @context.instance_exec(@selected_value, &@setter)
    end
  else
    @context.instance_exec(current_value, &@setter)
  end
end

#elementObject



59
60
61
# File 'lib/volt/page/bindings/attribute_binding.rb', line 59

def element
  Element.find('#' + binding_name)
end

#removeObject



115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
# File 'lib/volt/page/bindings/attribute_binding.rb', line 115

def remove
  # Unbind events, leave the element there since attribute bindings
  # aren't responsible for it being there.
  case @attribute_name
    when 'value'
      element.off('input.attrbind', nil)
    when 'checked'
      element.off('change.attrbind', nil)
  end

  @string_template_renderer.remove if @string_template_renderer
  @string_template_renderer_computation.stop if @string_template_renderer_computation
  @computation.stop if @computation

  # Clear any references
  @target  = nil
  @context = nil
  @getter  = nil
end

#remove_anchorsObject



135
136
137
# File 'lib/volt/page/bindings/attribute_binding.rb', line 135

def remove_anchors
  fail 'attribute bindings do not have anchors, can not remove them'
end

#setupObject



16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
# File 'lib/volt/page/bindings/attribute_binding.rb', line 16

def setup
  # Listen for changes
  @computation = -> do
    begin
      update(@context.instance_eval(&@getter))
    rescue => e
      Volt.logger.error("AttributeBinding Error: #{e.inspect}")
      update('')
    end
  end.watch!

  @is_radio = element.is('[type=radio]')
  if @is_radio
    @selected_value = element.attr('value')
  end

  # Bind so when this value updates, we update
  case @attribute_name
    when 'value'
      element.on('input.attrbind') { changed }
    when 'checked'
      element.on('change.attrbind') { |event| changed(event) }
  end
end

#update(new_value) ⇒ Object



63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
# File 'lib/volt/page/bindings/attribute_binding.rb', line 63

def update(new_value)
  if @attribute_name == 'checked'
    update_checked(new_value)
    return
  end

  # Stop any previous reactive template computations
  @string_template_renderer_computation.stop if @string_template_renderer_computation
  @string_template_renderer.remove if @string_template_renderer

  if new_value.is_a?(StringTemplateRender)
    # We don't need to refetch the whole reactive template to
    # update, we can just depend on it and update directly.
    @string_template_renderer = new_value

    @string_template_renderer_computation = -> do
      self.value = @string_template_renderer.html
    end.watch!
  else
    if new_value.is_a?(NilMethodCall) || new_value.nil?
      new_value = ''
    end

    self.value = new_value
  end
end

#update_checked(value) ⇒ Object



103
104
105
106
107
108
109
110
111
112
113
# File 'lib/volt/page/bindings/attribute_binding.rb', line 103

def update_checked(value)
  if value.is_a?(NilMethodCall) || value.nil?
    value = false
  end

  if @is_radio
    value = (@selected_value == value)
  end

  element.prop('checked', value)
end

#value=(val) ⇒ Object



90
91
92
93
94
95
96
97
98
99
100
101
# File 'lib/volt/page/bindings/attribute_binding.rb', line 90

def value=(val)
  case @attribute_name
    when 'value'
      # TODO: only update if its not the same, this keeps it from moving the
      # cursor in text fields.
      if val != element.value
        element.value = val
      end
    else
      element[@attribute_name] = val
  end
end