3

I would like to have a proxied object with methods and private variables attached to it.

That is to say, all normal object properties:

foo = {}
foo.bar = "baz"
foo.boo = "hoo"

with some prototypes:

foo.setPrivateThings = function(value){ if (value) private = value; return private; }

where enumeration skips private variables/functions:

console.log(foo); // { bar: "baz", boo: "hoo" }

and get/set will be run through a magic getter/setter:

foo.doesntexist = "..." = function(key){ console.log "Setting "+key; return new Date(); }

So far I have this hokey Coffeescript using node-proxy. Any better answers?

class Data
    constructor: (obj) ->
        @proxy = require "node-proxy"

        p = @proxy.create
            has: (name) ->
                name of obj
            hasOwn: (name) ->
                ({}).hasOwnProperty.call obj, name
            get: (receiver, name) ->                
                p = obj.transform()[name]
                if typeof p != 'undefined' && p != '__lookupGetter__' && p != '__lookupSetter__'
                    return p()
                # We could do magic getting here
                obj[name]
            set: (receiver, name, val) ->
                # We could do magic setting here
                obj[name] = val
                true
            enumerate: ->
                result = []
                for name of obj
                    result.push name if typeof obj[name] != 'function'
                result
            keys: ->
                Object.keys obj
        , obj

        _transform = {}
        p.transform = (_t) ->
            _transform = _t if _t
            return _transform

        return p


d = new Data
    name: "Bill"
    colors: ["blue", "red"]

d.transform
    timer: ->
        return new Date()

console.log d.name, d.colors, d.timer

console.log d

produces

Bill [ 'blue', 'red' ] Sat, 15 Oct 2011 06:39:27 GMT
{ name: 'Bill', colors: [ 'blue', 'red' ] }
4

2 に答える 2

3

列挙はプライベート変数/関数をスキップします:

Object.defineProperty(foo, "name", {
  get: function () { "return magic"; },
  set: function (value) { setMagic(value); },
  writable: true,
  configurable: true
  enumerable: false
});

プロキシはやり過ぎです。列挙できないプロパティを定義するだけです

于 2011-10-15T12:14:20.243 に答える
0

@Raynos のおかげで、この解決策が適していると思います (さらに実験します)。

obj =
    firstname: "Bill"
    lastname: "Fell"
    colors: ["blue", "red"]

transform =
    name: ->
        "#{this.firstname} #{this.lastname}"
    colorstoo: "colors"

for key, val of transform   
    Object.defineProperty obj, key,
        get: if typeof val == "string" then new Function("return this.#{val}") else val
        enumerable: true

console.log obj.name, obj.colorstoo
console.log obj

生成:

Bill Fell [ 'blue', 'red' ]

{ firstname: 'Bill',
  lastname: 'Fell',
  colors: [ 'blue', 'red' ],
  name: [Getter],
  colorstoo: [Getter] }

このソリューションでは、動的な get/set は許可されず、リストとしてのみ定義されることに注意してください。

于 2011-10-15T18:00:44.443 に答える