|
=begin |
|
================================= |
|
PROXY VERSION 3 |
|
================================= |
|
|
|
|
|
DEFINITION: |
|
I want a simple wrapper which will wrap the Application class and down. |
|
|
|
Criterium: |
|
* Each class must be extendable directly via the proxy class: |
|
class ProxyApplication |
|
def method() |
|
#... |
|
end |
|
end |
|
class ProxyCurrent |
|
def method() |
|
#... |
|
end |
|
end |
|
#... |
|
* All methods of delegated class should be accessible from Proxy<<Class>> |
|
* Proxy object will first call on Proxy class to try to handle routines, before reverting to original object. |
|
* All original object methods will return wrapped `Proxy` objects. |
|
if Application.open_current #=> Current instance then |
|
ProxyApplication.open_current #=> ProxyCurrent instance |
|
if #<Current>.get_row(1) #=> Row instance |
|
#<ProxyCurrent>.get_row(1) #=> ProxyRow instance |
|
#... |
|
* Ideally would work even if within console |
|
* If is class, override original class name with Proxy object |
|
=end |
|
|
|
|
|
|
|
class Proxy |
|
#============================= |
|
# PUBLIC CLASS METHODS |
|
#============================= |
|
def self.enwrap(objToWrap) |
|
#Get klass |
|
klass = objToWrap.is_a?(Class) ? objToWrap : objToWrap.class |
|
|
|
#Refuse to enwrap these values |
|
if ::Proxy.isStandardLibraryClass(klass) |
|
return objToWrap |
|
end |
|
|
|
#Extension class name |
|
newClassName = "Proxy" + klass.to_s |
|
|
|
#If class already defined then receive it, else define it |
|
if !(::Object.constants.include?(newClassName.to_sym)) |
|
#Define a class dynamically |
|
classDef = ::Object.const_set(newClassName.to_sym, Class.new(ProxyOverride)) |
|
else |
|
classDef = ::Object.const_get(newClassName.to_sym) |
|
end |
|
|
|
#Wrap object |
|
override = classDef.new(objToWrap) |
|
retVal = Proxy.new(override, objToWrap) |
|
|
|
#If object is a class then overwrite the class name |
|
if objToWrap.is_a?(Class) |
|
::Object.const_set(klass.name.to_sym, retVal) |
|
end |
|
|
|
#Return new instance of wrapped(objToWrap) |
|
return retVal |
|
end |
|
|
|
#Test whether an object should be wrapped or not |
|
def self.isStandardLibraryClass(klass) |
|
@@classes ||= nil |
|
if !@@classes |
|
#It is anticipated that all classes up till RubyVM are default ruby classes |
|
classes = ::Object.constants.select {|n| ::Object.const_get(n).class.to_s == "Class"} |
|
@@classes = classes.first(classes.find_index(:RubyVM)+1) |
|
end |
|
|
|
# |
|
return @@classes.include?(klass.name.to_sym) |
|
end |
|
|
|
#============================= |
|
# PUBLIC INSTANCE METHODS |
|
#============================= |
|
|
|
def initialize(override,delegated) |
|
@override = override |
|
@delegated = delegated |
|
end |
|
|
|
def method_missing(m,*args,&block) |
|
#Wrap block arguments |
|
if block_given? |
|
newBlock = Proc.new do |*args| |
|
args = args.map {|arg| ::Proxy.enwrap(arg)} |
|
block.call(*args) |
|
end |
|
else |
|
newBlock = Proc.new {} |
|
end |
|
|
|
#If delegate is a class and override class responds to method then... |
|
if @delegated.is_a?(Class) && @override.class.respond_to?(m) |
|
#Get value to return |
|
retVal = @override.class.__send__(m,*args,&newBlock) |
|
#If override instance responds to method then... |
|
elsif @override.respond_to?(m) |
|
#Get value to return |
|
retVal = @override.__send__(m,*args,&newBlock) |
|
#Assume delegate instance responds to method. |
|
else |
|
#Get value to return |
|
retVal = @delegated.__send__(m,*args,&newBlock) |
|
end |
|
|
|
#Ensure retVal is wrapped in a Proxy |
|
return ::Proxy.enwrap(retVal) |
|
end |
|
|
|
#Use delegated class for `class()` as this is more commonly used |
|
def class |
|
@delegated.is_a?(Class) ? @delegated : @delegated.class |
|
end |
|
end |
|
|
|
|
|
class ProxyOverride |
|
def initialize(obj) |
|
#Work around for |
|
@obj = obj |
|
end |
|
end |