All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.tapestry5.tree.js Maven / Gradle / Ivy

/* Copyright 2011 The Apache Software Foundation
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

T5.extend(T5, {
    tree : {

        /**
         * Approximate time per pixel for the hide and reveal animations. The
         * idea is to have small (few children) and large (many childen)
         * animations operate at the same visible rate, even though they will
         * take different amounts of time.
         */
        ANIMATION_RATE : .005,

        /**
         * Maximum animation time, in seconds. This is necessary for very large
         * animations, otherwise its looks visually odd to see the child tree
         * nodes whip down the screen.
         */
        MAX_ANIMATION_DURATION : .5,

        /** Type of Scriptaculous effect to hide/show child nodes. */
        TOGGLE_TYPE : 'blind',

        /**
         * Name of Scriptaculous effects queue to ensure that animations do not
         * overwrite each other.
         */
        QUEUE_NAME : 't-tree-updates'
    }
});

T5.extendInitializers(function() {

    var cfg = T5.tree;

    function doAnimate(element) {
        var sublist = $(element).up('li').down("ul");

        var dim = sublist.getDimensions();

        var duration = Math.min(dim.height * cfg.ANIMATION_RATE,
            cfg.MAX_ANIMATION_DURATION)

        new Effect.toggle(sublist, cfg.TOGGLE_TYPE, {
            duration : duration,
            queue : {
                position : 'end',
                scope : cfg.QUEUE_NAME
            }
        });
    }

    function animateRevealChildren(element) {
        $(element).addClassName("t-tree-expanded");

        doAnimate(element);
    }

    function animateHideChildren(element) {
        $(element).removeClassName("t-tree-expanded");

        doAnimate(element);
    }

    function initializer(spec) {
        var loaded = spec.expanded || spec.leaf;
        var expanded = spec.expanded;
        var loading = false;

        if (expanded) {
            $(spec.clientId).addClassName("t-tree-expanded")
        }


        function successHandler(reply) {
            // Remove the Ajax load indicator
            $(spec.clientId).update("");
            $(spec.clientId).removeClassName("t-empty-node");

            var response = reply.responseJSON;

            Tapestry.loadScriptsInReply(response, function() {
                var element = $(spec.clientId).up("li");
                var label = element.down("span.t-tree-label");

                label.insert({
                    after : response.content
                });

                // Hide the new sublist so that we can animate revealing it.
                element.down("ul").hide();

                animateRevealChildren(spec.clientId);

                loading = false;
                loaded = true;
                expanded = true;
            });

        }

        function doLoad() {
            if (loading)
                return;

            loading = true;

            $(spec.clientId).addClassName("t-empty-node");

            $(spec.clientId).update("");

            Tapestry.ajaxRequest(spec.expandChildrenURL, successHandler);
        }

        $(spec.clientId).observe("click", function(event) {
            event.stop();

            if (!loaded && spec.expandChildrenURL) {

                doLoad();

                return;
            }

            // Children have been loaded, just a matter of toggling
            // between showing or hiding the children.

            var f = expanded ? animateHideChildren : animateRevealChildren;

            f.call(null, spec.clientId);

            var url = expanded ? spec.markCollapsedURL : spec.markExpandedURL;

            // Send request, ignore response.

            Tapestry.ajaxRequest(url, {});

            expanded = !expanded;
        });


        if (spec.selectURL) {

            var selected = spec.selected;

            var label = $(spec.clientId).next("span.t-tree-label");

            label.addClassName("t-selectable");

            if (selected) {
                label.addClassName("t-selected-leaf-node-label");
            }

            label.observe("click", function(event) {
                event.stop();


                selected = ! selected;


                if (selected) {
                    label.addClassName("t-selected-leaf-node-label");
                }
                else {
                    label.removeClassName("t-selected-leaf-node-label");
                }

                // TODO: In the future, we may want to select children when a parent is selected,
                // or vice-versa. There's a lot of use cases. These will be directed from new methods
                // on the TreeSelectionModel interface and encoded into the response. For now,
                // the response is empty and ignored.

                Tapestry.ajaxRequest(spec.selectURL, { parameters: { "t:selected": selected } });
            });
        }
    }

    return {
        treeNode : initializer
    };
});




© 2015 - 2025 Weber Informatics LLC | Privacy Policy