(function(requirejs, require, define) {
    define(
    ['js/capa/drag_and_drop/state',
     'js/capa/drag_and_drop/config_parser', 'js/capa/drag_and_drop/container',
     'js/capa/drag_and_drop/base_image', 'js/capa/drag_and_drop/scroller',
     'js/capa/drag_and_drop/draggables', 'js/capa/drag_and_drop/targets',
     'js/capa/drag_and_drop/update_input'],
    function(State, configParser, Container, BaseImage, Scroller, Draggables, Targets, updateInput) {
        return Main;

        function Main() {
        // https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/Array/every
        //
        // Array.prototype.every is a recent addition to the ECMA-262 standard; as such it may not be present in
        // other implementations of the standard.
            if (!Array.prototype.every) {
                Array.prototype.every = function(fun /* , thisp */) {
                    var thisp, t, len, i;

                    if (this == null) {
                        throw new TypeError();
                    }

                    t = Object(this);
                    len = t.length >>> 0;
                    if (typeof fun != 'function') {
                        throw new TypeError();
                    }

                    thisp = arguments[1];

                    for (i = 0; i < len; i++) {
                        if (i in t && !fun.call(thisp, t[i], i, t)) {
                            return false;
                        }
                    }

                    return true;
                };
            }

            $('.drag_and_drop_problem_div').each(processProblem);
        }

    // $(value) - get the element of the entire problem
        function processProblem(index, value) {
            var problemId, config, state;

            if ($(value).attr('data-problem-processed') === 'true') {
            // This problem was already processed by us before, so we will
            // skip it.

                return;
            }
            $(value).attr('data-problem-processed', 'true');

            problemId = $(value).attr('data-plain-id');
            if (typeof problemId !== 'string') {
                console.log('ERROR: Could not find the ID of the problem DOM element.');

                return;
            }

            try {
                config = JSON.parse($('#drag_and_drop_json_' + problemId).html());
            } catch (err) {
                console.log('ERROR: Could not parse the JSON configuration options.');
                console.log('Error message: "' + err.message + '".');

                return;
            }

            state = State(problemId);

            if (configParser(state, config) !== true) {
                console.log('ERROR: Could not make sense of the JSON configuration options.');

                return;
            }

            Container(state);
            BaseImage(state);

            (function addContent() {
                if (state.baseImageLoaded !== true) {
                    setTimeout(addContent, 50);

                    return;
                }

                Targets.initializeBaseTargets(state);
                Scroller(state);
                Draggables.init(state);

                state.updateArrowOpacity();

            // Update the input element, checking first that it is not filled with
            // an answer from the server.
                if (updateInput.check(state) === false) {
                    updateInput.update(state);
                }
            }());
        }
    }); // End-of: define(
}(RequireJS.requirejs, RequireJS.require, RequireJS.define)); // End-of: (function (requirejs, require, define) {