require 'volt/reactive/eventable' module Volt class ReactiveArray include Eventable def initialize(array = []) @array = array @array_deps = [] @size_dep = Dependency.new @old_size = 0 end # Forward any missing methods to the array def method_missing(method_name, *args, &block) @array.send(method_name, *args, &block) end def ==(*args) @array.==(*args) end # At the moment, each just passes through. def each(&block) @array.each(&block) end def empty? @size_dep.depend @array.empty? end def count(&block) if block count = 0 size.times do |index| if block.call(self[index]).true? count += 1 end end count else size end end def select result = [] size.times do |index| val = self[index] if yield(val).true? result << val end end result end def any? if block_given? size.times do |index| val = self[index] if yield(val).true? return true end end false else @array.any? end end def all? if block_given? size.times do |index| val = self[index] unless yield(val).true? return false end end true else @array.all? end end # TODO: Handle a range def [](index) # Handle a negative index, depend on size index = size + index if index < 0 # Get or create the dependency dep = (@array_deps[index] ||= Dependency.new) # Track the dependency dep.depend # Return the index @array[index] end def []=(index, value) # Assign the new value @array[index] = value trigger_for_index!(index) trigger_size_change! end def size @size_dep.depend @array.size end alias_method :length, :size def delete_at(index) # Handle a negative index index = size + index if index < 0 model = @array.delete_at(index) # Remove the dependency for that cell, and #remove it index_deps = @array_deps.delete_at(index) index_deps.remove if index_deps trigger_removed!(index) # Trigger a changed event for each element in the zone where the # delete would change index.upto(size + 1) do |position| trigger_for_index!(position) end trigger_size_change! @persistor.removed(model) if @persistor model end def delete(val) index = @array.index(val) if index delete_at(index) else # Sometimes the model isn't loaded at the right state yet, so we # just remove it from the persistor @persistor.removed(val) if @persistor end end def clear old_size = @array.size deps = @array_deps @array_deps = [] # Trigger remove for each cell old_size.times do |index| trigger_removed!(old_size - index - 1) end # Trigger on each cell since we are clearing out the array if deps deps.each do |dep| dep.changed! if dep end end # clear the array @array = [] end # alias :__old_append :<< def <<(value) result = (@array << value) trigger_for_index!(size - 1) trigger_added!(size - 1) trigger_size_change! result end def +(array) fail 'not implemented yet' old_size = size # TODO: += is funky here, might need to make a .plus! method result = ReactiveArray.new(@array.dup + array) old_size.upto(result.size - 1) do |index| trigger_for_index!('changed', index) trigger_added!(old_size + index) end trigger_size_change! result end def insert(index, *objects) result = @array.insert(index, *objects) # All objects from index to the end have "changed" index.upto(result.size) do |index| trigger_for_index!(index) end objects.size.times do |count| trigger_added!(index + count) end trigger_size_change! result end def inspect # TODO: should also depend on items in array @size_dep.depend "#<#{self.class}:#{object_id} #{@array.inspect}>" end private # Check to see if the size has changed, trigger a change on size if it has def trigger_size_change! new_size = @array.size if new_size != @old_size @old_size = new_size @size_dep.changed! end end def trigger_for_index!(index) # Trigger a change for the cell dep = @array_deps[index] dep.changed! if dep end def trigger_added!(index) trigger!('added', index) end def trigger_removed!(index) trigger!('removed', index) end end end