/*Demonstrates class meta data.
Basically, we make use of the compiler's
data about a class.  We can add to the
compilers data arbitrary items, either string or
or int or F64 (if you typecast).

This stuff is not high performance.
Don't get carried away -- it might be slow.
*/

U0 RankOut(I64 i)
{
    " %z", i, "Cadet\0Ensign\0Captain\0Admiral\0President\0";
}

class Test1Struct
{
    I64 age         print_str "%2d" default_val 38;
    I64 color       default_val RED; //Accepts expressions
    I64 rank        print_str "%1d" default_val 6/2 output_fun &RankOut;
};

class Test2Struct
{
    I64 age             print_str "%2d" default_val 38 percentile 54.20;
    I64 rank            print_str "%1d" default_val 5;
    I64 serial_num      print_str "%6d" default_val 123456;
};

U0 DumpStruct(U8 *_d, U8 *class_name=lastclass)
{//lastclass is keyword.    See ::/Demo/LastClass.CC.
    CHashClass  *tmpc = HashFind(class_name, Fs->hash_table, HTT_CLASS);
    U8          *print_str;
    I64         *q, default_val;
    U0         (*fp_output_fun)(I64 i);
    F64          percentile;
    if (!tmpc)
        return;
    CMemberList *ml;

    ml = tmpc->member_list_and_root;
    while (ml)
    {
        "%s:", ml->str;

        //All our items are I64's.  If you want, you can check
        //the data type of the member variable.  See ClassRep().
        q = _d + ml->offset;

        if (print_str = MemberMetaData("print_str", ml))
            "" print_str, *q;

            //This is slightly ambiguous -- if no meta is present it will return zero.
        if (default_val = MemberMetaData("default_val", ml))
            " default:%d", default_val;

            //This corrects for the ambiguity, allowing zero percentile.
        if (MemberMetaFind("percentile", ml))
        {//check if it exists
            //We could use the CMemberListMeta structure returned by
            //MemberMetaFind() and save a search.
            percentile = MemberMetaData("percentile", ml)(F64);
            " percentile: %5.2f", percentile;
        }

        if (fp_output_fun = MemberMetaData("output_fun", ml))
            (*fp_output_fun)(*q);
        '\n';
        ml = ml->next;
    }
}

Test1Struct t1;
t1.age  = 44;
t1.rank = 3;

DumpStruct(&t1);

Test2Struct t2;
t2.age          = 22;
t2.rank         = 2;
t2.serial_num   = 55555;

DumpStruct(&t2);