Annotation of parser3/src/types/pa_vobject.h, revision 1.28
1.6 paf 1: /** @file
1.7 paf 2: Parser: @b object class decl.
3:
1.15 paf 4: Copyright (c) 2001, 2002 ArtLebedev Group (http://www.artlebedev.com)
1.16 paf 5: Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru)
1.1 paf 6: */
7:
8: #ifndef PA_VOBJECT_H
9: #define PA_VOBJECT_H
10:
1.28 ! paf 11: static const char* IDENT_VOBJECT_H="$Date: 2002/08/13 14:14:44 $";
1.19 paf 12:
1.1 paf 13: #include "pa_vjunction.h"
14: #include "pa_vclass.h"
1.4 paf 15: #include "pa_vstateless_object.h"
1.6 paf 16:
1.23 paf 17: // defines
18:
19: #define CLASS_NAME "CLASS"
20: #define BASE_NAME "BASE"
21:
1.26 paf 22: /** parser class instance, stores
23: - class VObject::fclass;
24: - fields VObject::ffields (dynamic, not static, which are stored in class).
25: - links to base/derived instances [VObject::fbase, VObject::fderived]
1.6 paf 26: */
1.24 paf 27: class VObject: public VStateless_object {
1.1 paf 28: public: // Value
29:
1.24 paf 30: const char *type() const { return fclass.name_cstr(); }
31: Value *as(const char *atype) { return fclass.as(atype); }
1.12 parser 32:
1.25 paf 33: /// VObject: fclass
34: VStateless_class *get_class() { return &fclass; }
35: /// VObject: fbase
36: /*override*/ Value *base_object() { return fbase; }
1.26 paf 37: /// VObject: true, todo: z base table can be 33
1.12 parser 38: Value *as_expr_result(bool) { return NEW VBool(pool(), as_bool()); }
1.26 paf 39: /// VObject: true, todo: z base table can be false
1.12 parser 40: bool as_bool() const { return true; }
1.1 paf 41:
1.26 paf 42: /// VObject: (field)=value;(CLASS)=vclass;(method)=method_ref
43: Value *get_element(const String& aname, Value *aself) {
44: // gets element from last_derivate upwards
45: if(aself) {
46: // $CLASS
47: if(aname==CLASS_NAME)
48: return get_class();
49:
50: // for first call, pass call to last derived VObject
51: return get_last_derived()->get_element(aname, 0/*mark this call as 'not first'*/);
52: }
53:
54: // $method, $CLASS_field
55: {
56: Temp_base temp_base(*get_class(), 0);
57: if(Value *result=VStateless_object::get_element(aname, this))
1.25 paf 58: return result;
59: }
60:
1.9 paf 61: // $field=ffields.field
1.26 paf 62: if(Value *result=static_cast<Value *>(ffields.get(aname)))
1.9 paf 63: return result;
64:
1.26 paf 65: // up the tree...
66: if(fbase)
67: if(Value *result=fbase->get_element(aname, fbase))
68: return result;
69:
70: return 0;
1.1 paf 71: }
72:
1.26 paf 73: /// VObject: (field)=value
1.27 paf 74: /*override*/ bool put_element(const String& aname, Value *avalue, bool replace) {
1.26 paf 75: // replaces element to last_derivate upwards or stores it in self
1.1 paf 76: // speed1:
77: // will not check for '$CLASS(subst)' trick
78: // will hope that user ain't THAT self-hating person
79: // speed2:
80: // will not check for '$method_name(subst)' trick
81: // -same-
82:
1.26 paf 83: // downwards: same as upwards
84:
85: if(fderived && fderived->put_element(aname, avalue, true))
86: return true; // replaced in derived
87:
88: // upwards: copied from VClass::put_element...
89:
1.27 paf 90: try {
91: if(fbase && fbase->put_element(aname, avalue, true))
92: return true; // replaced in base
1.28 ! paf 93: } catch(Exception) {
! 94: /* ignore "can not store to table&co errors for nonexistent elements */
! 95: bool error;
! 96: try {
! 97: error=get_element(aname, this)!=0;
! 98: } catch(Exception) {
! 99: error=false;
! 100: }
! 101: if(error)
! 102: /*re*/throw;
! 103: }
1.26 paf 104:
105: if(replace)
106: return ffields.put_replace(aname, avalue);
107: else {
108: ffields.put(aname, avalue);
109: return false;
110: }
1.1 paf 111: }
112:
1.24 paf 113: /// VObject: remember derived [the only client] */
1.26 paf 114: /*override*/ VObject *set_derived(VObject *aderived) {
115: VObject *result=fderived;
1.25 paf 116: fderived=aderived;
117: return fderived;
1.24 paf 118: }
119:
1.1 paf 120: public: // creation
121:
1.24 paf 122: VObject(Pool& apool, VStateless_class& aclass) : VStateless_object(apool),
123: fclass(aclass),
124: ffields(apool),
125: fderived(0),
1.25 paf 126: fbase(fclass.base_class()?fclass.base_class()->create_new_value(apool):0) {
1.24 paf 127: if(fbase)
1.25 paf 128: fbase->set_derived(this);
1.1 paf 129: }
130:
131: private:
132:
1.26 paf 133: Value *get_last_derived() {
134: return fderived?fderived->get_last_derived():this;
135: }
136:
137: private:
138:
1.24 paf 139: VStateless_class& fclass;
1.1 paf 140: Hash ffields;
1.26 paf 141: VObject *fderived;
1.24 paf 142: Value *fbase;
1.25 paf 143: };
144:
145: class Temp_derived {
146: Value& fvalue;
1.26 paf 147: VObject *fsaved_derived;
1.25 paf 148: public:
1.26 paf 149: Temp_derived(Value& avalue, VObject *aderived) :
1.25 paf 150: fvalue(avalue),
151: fsaved_derived(avalue.set_derived(aderived)) {}
152: ~Temp_derived() { fvalue.set_derived(fsaved_derived); }
1.1 paf 153: };
154:
155: #endif
E-mail: