| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470 | //jshint eqnull:truedefine(function () {    "use strict";    function defineMethod (klass, name, method) {        Object.defineProperty(klass.prototype, name, {            value: method,            enumerable: false, configurable: true, writable: true        });    }    function installJSMethod (obj, jsSelector, fn) {        Object.defineProperty(obj, jsSelector, {            value: fn,            enumerable: false, configurable: true, writable: true        });    }    RuntimeSelectorsBrik.deps = ["selectors", "selectorConversion", "smalltalkGlobals", "classes"];    function RuntimeSelectorsBrik (brikz, st) {        var selectors = brikz.selectors.selectors;        var globals = brikz.smalltalkGlobals.globals;        var nilAsClass = brikz.classes.nilAsClass;        var st2js = brikz.selectorConversion.st2js;        var jsSelectors = this.jsSelectors = [];        /* Method not implemented handlers */        function installNewSelectors (newSelectors, targetClasses) {            newSelectors.forEach(function (selector) {                var jsSelector = st2js(selector);                jsSelectors.push(jsSelector);                var fn = createDnuHandler(selector);                installJSMethod(nilAsClass.fn.prototype, jsSelector, fn);                targetClasses.forEach(function (target) {                    installJSMethod(target.fn.prototype, jsSelector, fn);                });            });        }        this.installNewSelectors = installNewSelectors;        /* Dnu handler method */        function createDnuHandler (stSelector) {            return function () {                return globals.Message._selector_arguments_notUnderstoodBy_(                    stSelector, [].slice.call(arguments), this                );            };        }        installNewSelectors(selectors, []);    }    RuntimeClassesBrik.deps = ["event", "runtimeSelectors", "behaviors", "classes", "runtimeMethods"];    function RuntimeClassesBrik (brikz, st) {        var jsSelectors = brikz.runtimeSelectors.jsSelectors;        var installNewSelectors = brikz.runtimeSelectors.installNewSelectors;        var installMethod = brikz.runtimeMethods.installMethod;        var traitsOrClasses = brikz.behaviors.traitsOrClasses;        var wireKlass = brikz.classes.wireKlass;        var emit = brikz.event.emit;        var detachedRootClasses = [];        function markClassDetachedRoot (klass) {            klass.detachedRoot = true;            detachedRootClasses = traitsOrClasses.filter(function (klass) {                return klass.detachedRoot;            });        }        emit.selectorsAdded = function (newSelectors) {            installNewSelectors(newSelectors, detachedRootClasses);        };        /* Initialize a class in its class hierarchy. Handle both classes and         metaclasses. */        function initClassAndMetaclass (klass) {            initClass(klass);            if (klass.a$cls && !klass.meta) {                initClass(klass.a$cls);            }        }        traitsOrClasses.forEach(function (traitOrClass) {            if (!traitOrClass.trait) initClassAndMetaclass(traitOrClass);        });        emit.classAdded = function (klass) {            initClassAndMetaclass(klass);            klass._enterOrganization();        };        emit.traitAdded = function (trait) {            trait._enterOrganization();        };        emit.classRemoved = function (klass) {            klass._leaveOrganization();        };        emit.traitRemoved = function (trait) {            trait._leaveOrganization();        };        function initClass (klass) {            wireKlass(klass);            if (klass.detachedRoot) {                copySuperclass(klass);            }            installMethods(klass);        }        function copySuperclass (klass) {            var myproto = klass.fn.prototype,                superproto = klass.superclass.fn.prototype;            jsSelectors.forEach(function (jsSelector) {                installJSMethod(myproto, jsSelector, superproto[jsSelector]);            });        }        function installMethods (klass) {            var methods = klass.methods;            Object.keys(methods).forEach(function (selector) {                installMethod(methods[selector], klass);            });        }        /* Manually set the constructor of an existing Smalltalk klass, making it a detached root class. */        st.setClassConstructor = this.setClassConstructor = function (klass, constructor) {            markClassDetachedRoot(klass);            klass.fn = constructor;            initClass(klass);        };    }    FrameBindingBrik.deps = ["smalltalkGlobals", "runtimeClasses"];    function FrameBindingBrik (brikz, st) {        var globals = brikz.smalltalkGlobals.globals;        var setClassConstructor = brikz.runtimeClasses.setClassConstructor;        setClassConstructor(globals.Number, Number);        setClassConstructor(globals.BlockClosure, Function);        setClassConstructor(globals.Boolean, Boolean);        setClassConstructor(globals.Date, Date);        setClassConstructor(globals.String, String);        setClassConstructor(globals.Array, Array);        setClassConstructor(globals.RegularExpression, RegExp);        setClassConstructor(globals.Error, Error);        setClassConstructor(globals.Promise, Promise);        this.__init__ = function () {            st.alias(globals.Array, "OrderedCollection");            st.alias(globals.Date, "Time");        }    }    RuntimeMethodsBrik.deps = ["event", "selectorConversion"];    function RuntimeMethodsBrik (brikz, st) {        var st2js = brikz.selectorConversion.st2js;        var emit = brikz.event.emit;        function installMethod (method, klass) {            var jsSelector = method.jsSelector;            if (!jsSelector) {                jsSelector = method.jsSelector = st2js(method.selector);            }            installJSMethod(klass.fn.prototype, jsSelector, method.fn);        }        this.installMethod = installMethod;        emit.behaviorMethodAdded = function (method, klass) {            installMethod(method, klass);            propagateMethodChange(klass, method, klass);        };        emit.behaviorMethodRemoved = function (method, klass) {            delete klass.fn.prototype[method.jsSelector];            propagateMethodChange(klass, method, null);        };        emit.methodReplaced = function (newMethod, oldMethod, traitOrBehavior) {            traitOrBehavior._methodOrganizationEnter_andLeave_(newMethod, oldMethod);        };        function propagateMethodChange (klass, method, exclude) {            var selector = method.selector;            var jsSelector = method.jsSelector;            st.traverseClassTree(klass, function (subclass, sentinel) {                if (subclass === exclude) return;                if (subclass.methods[selector]) return sentinel;                if (subclass.detachedRoot) {                    installJSMethod(subclass.fn.prototype, jsSelector, subclass.superclass.fn.prototype[jsSelector]);                }            });        }    }    PrimitivesBrik.deps = ["smalltalkGlobals"];    function PrimitivesBrik (brikz, st) {        var globals = brikz.smalltalkGlobals.globals;        /* Converts a JavaScript object to valid Smalltalk Object */        st.readJSObject = function (js) {            if (js == null) return null;            else if (Array.isArray(js)) return js.map(st.readJSObject);            else if (js.constructor !== Object) return js;            var pairs = [];            for (var i in js) {                pairs.push(i, st.readJSObject(js[i]));            }            return globals.Dictionary._newFromPairs_(pairs);        };        /* Boolean assertion */        st.assert = function (shouldBeBoolean) {            if (typeof shouldBeBoolean === "boolean") return shouldBeBoolean;            else if (shouldBeBoolean != null && typeof shouldBeBoolean === "object") {                shouldBeBoolean = shouldBeBoolean.valueOf();                if (typeof shouldBeBoolean === "boolean") return shouldBeBoolean;            }            globals.NonBooleanReceiver._signalOn_(shouldBeBoolean);        };    }    RuntimeBrik.deps = ["selectorConversion", "smalltalkGlobals", "runtimeClasses"];    function RuntimeBrik (brikz, st) {        var globals = brikz.smalltalkGlobals.globals;        var setClassConstructor = brikz.runtimeClasses.setClassConstructor;        function SmalltalkMethodContext (home, setup) {            // TODO lazy fill of .sendIdx            this.sendIdx = {};            // TODO very likely .senderContext, not .homeContext here            this.homeContext = home;            this.setup = setup;        }        // Fallbacks        SmalltalkMethodContext.prototype.supercall = false;        SmalltalkMethodContext.prototype.locals = Object.freeze({});        SmalltalkMethodContext.prototype.receiver = null;        SmalltalkMethodContext.prototype.selector = null;        SmalltalkMethodContext.prototype.lookupClass = null;        SmalltalkMethodContext.prototype.outerContext = null;        SmalltalkMethodContext.prototype.index = 0;        defineMethod(SmalltalkMethodContext, "fill", function (receiver, selector, locals, lookupClass) {            this.receiver = receiver;            this.selector = selector;            if (locals != null) this.locals = locals;            this.lookupClass = lookupClass;            if (this.homeContext) {                this.homeContext.evaluatedSelector = selector;            }        });        defineMethod(SmalltalkMethodContext, "fillBlock", function (locals, ctx, index) {            if (locals != null) this.locals = locals;            this.outerContext = ctx;            if (index) this.index = index;        });        defineMethod(SmalltalkMethodContext, "method", function () {            var method;            var lookup = this.lookupClass || this.receiver.a$cls;            while (!method && lookup) {                method = lookup.methods[st.js2st(this.selector)];                lookup = lookup.superclass;            }            return method;        });        setClassConstructor(globals.MethodContext, SmalltalkMethodContext);        /* This is the current call context object.         In Smalltalk code, it is accessible just by using 'thisContext' variable.         In JS code, use api.getThisContext() (see below).         */        var thisContext = null;        /*         Runs worker function so that error handler is not set up         if there isn't one. This is accomplished by unconditional         wrapping inside a context of a simulated `nil seamlessDoIt` call,         which then stops error handler setup (see st.withContext above).         The effect is, $core.seamless(fn)'s exceptions are not         handed into ST error handler and caller should process them.         */        st.seamless = function (worker) {            var oldContext = thisContext;            thisContext = new SmalltalkMethodContext(thisContext, function (ctx) {                ctx.fill(null, "seamlessDoIt", {}, globals.UndefinedObject);            });            var result = worker(thisContext);            thisContext = oldContext;            return result;        };        function resultWithErrorHandling (worker) {            try {                return worker(thisContext);            } catch (error) {                globals.ErrorHandler._handleError_(error);                thisContext = null;                // Rethrow the error in any case.                throw error;            }        }        /*         Standard way to run within context.         Sets up error handler if entering first ST context in a stack.         */        st.withContext = function (worker, setup) {            var oldContext = thisContext;            thisContext = new SmalltalkMethodContext(thisContext, setup);            var result = oldContext == null ? resultWithErrorHandling(worker) : worker(thisContext);            thisContext = oldContext;            return result;        };        /* Handle thisContext pseudo variable */        st.getThisContext = function () {            if (!thisContext) return null;            for (var frame = thisContext; frame; frame = frame.homeContext) {                frame.setup(frame);            }            return thisContext;        };    }    MessageSendBrik.deps = ["smalltalkGlobals", "selectorConversion", "root"];    function MessageSendBrik (brikz, st) {        var globals = brikz.smalltalkGlobals.globals;        var nilAsReceiver = brikz.root.nilAsReceiver;        /* Send message programmatically. Used to implement #perform: & Co. */        st.send2 = function (self, selector, args, klass) {            if (self == null) {                self = nilAsReceiver;            }            var method = klass ? klass.fn.prototype[st.st2js(selector)] : self.a$cls && self[st.st2js(selector)];            return method != null ?                method.apply(self, args || []) :                globals.Message._selector_arguments_notUnderstoodBy_(                    selector, [].slice.call(args), self.a$cls ? self : wrapJavaScript(self)                );        };        function wrapJavaScript (o) {            return globals.JSObjectProxy._on_(o);        }        st.wrapJavaScript = wrapJavaScript;        /* If the object property is a function, then call it, except if it starts with         an uppercase character (we probably want to answer the function itself in this         case and send it #new from Amber).         */        st.accessJavaScript = function (self, propertyName, args) {            var propertyValue = self[propertyName];            if (typeof propertyValue === "function" && !/^[A-Z]/.test(propertyName)) {                return propertyValue.apply(self, args || []);            } else if (args.length === 0) {                return propertyValue;            } else {                self[propertyName] = args[0];                return self;            }        };    }    function SelectorConversionBrik (brikz, st) {        /* Convert a Smalltalk selector into a JS selector */        st.st2js = this.st2js = function (string) {            return '_' + string                .replace(/:/g, '_')                .replace(/[\&]/g, '_and')                .replace(/[\|]/g, '_or')                .replace(/[+]/g, '_plus')                .replace(/-/g, '_minus')                .replace(/[*]/g, '_star')                .replace(/[\/]/g, '_slash')                .replace(/[\\]/g, '_backslash')                .replace(/[\~]/g, '_tild')                .replace(/%/g, '_percent')                .replace(/>/g, '_gt')                .replace(/</g, '_lt')                .replace(/=/g, '_eq')                .replace(/,/g, '_comma')                .replace(/[@]/g, '_at');        };        /* Convert a string to a valid smalltalk selector.         if you modify the following functions, also change st2js         accordingly */        st.js2st = function (selector) {            if (selector.match(/^__/)) {                return binaryJsToSt(selector);            } else {                return keywordJsToSt(selector);            }        };        function keywordJsToSt (selector) {            return selector.replace(/^_/, '').replace(/_/g, ':');        }        function binaryJsToSt (selector) {            return selector                .replace(/^_/, '')                .replace(/_and/g, '&')                .replace(/_or/g, '|')                .replace(/_plus/g, '+')                .replace(/_minus/g, '-')                .replace(/_star/g, '*')                .replace(/_slash/g, '/')                .replace(/_backslash/g, '\\')                .replace(/_tild/g, '~')                .replace(/_percent/g, '%')                .replace(/_gt/g, '>')                .replace(/_lt/g, '<')                .replace(/_eq/g, '=')                .replace(/_comma/g, ',')                .replace(/_at/g, '@');        }        st.st2prop = function (stSelector) {            var colonPosition = stSelector.indexOf(':');            return colonPosition === -1 ? stSelector : stSelector.slice(0, colonPosition);        };    }    StartImageBrik.deps = ["smalltalkGlobals"];    function StartImageBrik (brikz, st) {        var globals = brikz.smalltalkGlobals.globals;        this.run = function () {            globals.AmberBootstrapInitialization._run();        };    }    /* Making smalltalk that can run */    function configureWithRuntime (brikz) {        brikz.runtimeSelectors = RuntimeSelectorsBrik;        brikz.runtimeClasses = RuntimeClassesBrik;        brikz.frameBinding = FrameBindingBrik;        brikz.runtimeMethods = RuntimeMethodsBrik;        brikz.messageSend = MessageSendBrik;        brikz.runtime = RuntimeBrik;        brikz.primitives = PrimitivesBrik;        brikz.selectorConversion = SelectorConversionBrik;        brikz.startImage = StartImageBrik;        brikz.rebuild();    }    return configureWithRuntime;});
 |