1
votes

I'm currently trying to write a Rally app that will display the Predecessor/Successor hierarchy for a selected User Story. To illustrate, the user will select a User Story from a Chooser UI element. Then, a three-column Cardboard UI element will be generated--the leftmost column will contain all of the selected User Story's Predecessors (in card form), the middle column will contain the selected User Story's card, and the rightmost column will contain all of the selected User Story's Successors (in card form). From there, the Predecessor and Successor cards can be removed (denoting that they won't be Predecessors or Successors for the selected User Story) and new Predecessor/Successor cards can be added (denoting that they will become new Predecessors/Successors for the selected User Story).

However, my issue is this: the Cardboard UI was designed to display sets of different values for one particular attribute, but "Predecessor" and "Successor" don't fall into this category. Is there a possible way for me to display a User Story and then get its Predecessors and Successors and populate the rest of the board? I realize that it will take a substantial amount of modifications to the original board.

1

1 Answers

1
votes

I've found a way to hack it so that you can do the display. Not sure if it's worth it or not and what you want to do for adding/removing, but this might help set you on the right path.

In summary, the problem is that the cardboard/column classes are designed to work with single value fields and each column that's created does an individual query to Rally based on the column configuration. You'll need to override the rallycardboard and the rallycolumn. I'll give the full html that you can paste in below, but let's hit this one piece at a time.

If nothing else, this might be a good example of how to take the source code of rally classes and make something to override them.

Data:

The existing cardboard is given a record type and field and runs off to create a column for each valid value for the field. It gets this information by querying Rally for the stories and for the attribute definitions. But we want to use our data in a slightly different way, so we'll have to make a different data store and feed it in. So we want to use a wsapidatastore to go ahead and get the record we asked for (in this example, I have a story called US37 that has predecessors and successors). In a way, this is like doing a cross-tab in Excel. Instead of having one record (37) that's related to others, we want to make a data set of all the stories and define their relationship in a new field, which I called "_column." Like this:

Ext.create('Rally.data.WsapiDataStore', {
            model: "hierarchicalrequirement",
            autoLoad: true,
            fetch: ['Name','Predecessors','Successors','FormattedID','ObjectID','_ref'],
            filters: [ {
                property: 'FormattedID', operator: 'contains', value: '37'
            } ] /* get the record US37 */,
            listeners: { 
                load: function(store,data,success) {
                    if ( data.length === 1 ) {
                        var base_story = data[0].data;
                        var modified_records = [];

                        base_story._column = "base";
                        modified_records.push( base_story );

                        Ext.Array.each( base_story.Predecessors, function( story ) {
                            story._column = "predecessor";
                            modified_records.push( story );
                        } );

                        Ext.Array.each( base_story.Successors, function(story) {
                            story._column = "successor";
                            modified_records.push( story );
                        } );

We push the data into an array of objects with each having a new field to define which column it should go in. But this isn't quite enough, because we need to put the data into a store. The store needs a model -- and we have to define the fields in a way that the cardrenders know how to access the data. There doesn't seem to be an easy way to just add a field definition to an existing rally model, so this should do it (the rally model has unique field information and a method called getField(), so we have to add that:

                        Ext.define('CardModel', { 
                            extend: 'Ext.data.Model',
                            fields: [
                                { name: '_ref', type: 'string' },
                                { name: 'ObjectID', type: 'number'},
                                { name: 'Name', type: 'string', attributeDefinition: { AttributeType: 'STRING'} },
                                { name: 'FormattedID', type: 'string'},
                                { name: '_column', type: 'string' },
                                { name: 'ScheduleState', type: 'string' } ] ,
                            getField: function(name) {
                                if ( this.data[name] ) { 
                                    var return_field = null;
                                    Ext.Array.each( this.store.model.getFields(), function(field) {
                                        if ( field.name === name ) {
                                            return_field = field;
                                        }
                                    } );

                                    return return_field;
                                } else {
                                    return null;
                                }
                            } 
                        });

                        var cardStore = Ext.create('Ext.data.Store',{
                            model: 'CardModel',
                            data: modified_records
                        });

Now, we'll create a cardboard, but instead of the rally cardboard, we'll make one from a class we're going to define below ('DependencyCardboard'). In addition, we'll pass along a new definition for the columns that we'll also define below ('dependencycolumn').

                        var cardboard = Ext.create('DependencyCardboard', {
                            attribute: '_column',
                            store: cardStore, /* special to our new cardboard type */
                            height: 500,
                            columns: [{
                                xtype: 'dependencycolumn',
                                displayValue: 'predecessor',
                                value: 'predecessor',
                                store: cardStore
                            }, 
                            {
                                xtype: 'dependencycolumn',
                                displayValue: 'base',
                                value: 'base',
                                store: cardStore
                            },
                            {
                                xtype: 'dependencycolumn',
                                displayValue: 'successor',
                                value: 'successor',
                                store: cardStore
                            }]
                        });

Cardboard:

Mostly, the existing Rally cardboard can handle our needs because all the querying is done down in the columns themselves. But we still have to override it because there is one function that is causing us problems: _retrieveModels. This function normally takes the record type(s) (e.g., User Story) and based on that creates a data model that is based on the Rally definition. However, we're not using the UserStory records directly; we've had to define our own model so we could add the "_columns" field. So, we make a new definition (which we use in the create statement above for "DependencyCardboard").

(Remember, we can see the source code for all of the Rally objects in the API just by clicking on the title, so we can compare the below method to the one in the base class.)

We can keep all of the stuff that the Rally cardboard does and only override the one method by doing this:

Ext.define( 'DependencyCardboard', {
    extend: 'Rally.ui.cardboard.CardBoard',
    alias: 'widget.dependencycardboard',
    constructor: function(config) {
        this.mergeConfig(config);
        this.callParent([this.config]);
    },
    initComponent: function() {
        this.callParent(arguments);
    },
    _retrieveModels: function(success) {
        if ( this.store ) {
            this.models = [ this.store.getProxy().getModel() ];
            success.apply( this, arguments );
        }
    }

});

Column:

Each column normally goes off to Rally and says "give me all of the stories that have a field equal to the column name". But we're passing in the store to the cardboard, so we need to override _queryForData. In addition, there is something going on about defining the column height when we do this (I don't know why!) so I had to add a little catch in the getColumnHeightFromCards() method.

_queryForData: function() {
        var allRecords = [];

        var records = this.store.queryBy( function( record ) {
            if ( record.data._column === this.getValue() ) { allRecords.push( record ); }
        }, this);

        this.createAndAddCards( allRecords );
    },
    getColumnHeightFromCards: function() {
        var contentMinHeight = 500,
            bottomPadding = 30,
            cards = this.query(this.cardConfig.xtype),
            height = bottomPadding;

        for(var i = 0, l = cards.length; i < l; ++i) {
            if ( cards[i].el ) {
                 height += cards[i].getHeight();
            } else {
                height += 100;
            }

        }

        height = Math.max(height, contentMinHeight);
        height += this.down('#columnHeader').getHeight();

        return height;
    } 

Finish

So, if you add all those pieces together, you get one long html file that we can push into a panel (and that you can keep working on to figure out how to override dragging results and to add your chooser panel for the first item. (And you can make better abstracted into its own class)).

Full thing:

<!DOCTYPE html>
<html>
<head>
    <title>cardboard</title>

    <script type="text/javascript" src="/apps/2.0p3/sdk.js"></script>

    <script type="text/javascript">
        Rally.onReady(function() {
            /*global console, Ext */

            Ext.define( 'DependencyColumn', {
                extend: 'Rally.ui.cardboard.Column',
                alias: 'widget.dependencycolumn',
                constructor: function(config) {
                    this.mergeConfig(config);
                    this.callParent([this.config]);
                },
                initComponent: function() {
                    this.callParent(arguments);
                },
                _queryForData: function() {
                    var allRecords = [];

                    var records = this.store.queryBy( function( record ) {
                        if ( record.data._column === this.getValue() ) { allRecords.push( record ); }
                    }, this);

                    this.createAndAddCards( allRecords );
                },
                getColumnHeightFromCards: function() {
                    var contentMinHeight = 500,
                        bottomPadding = 30,
                        cards = this.query(this.cardConfig.xtype),
                        height = bottomPadding;

                    for(var i = 0, l = cards.length; i < l; ++i) {
                        if ( cards[i].el ) {
                             height += cards[i].getHeight();
                        } else {
                            height += 100;
                        }

                    }

                    height = Math.max(height, contentMinHeight);
                    height += this.down('#columnHeader').getHeight();

                    return height;
                } 

            });
            /*global console, Ext */

            Ext.define( 'DependencyCardboard', {
                extend: 'Rally.ui.cardboard.CardBoard',
                alias: 'widget.dependencycardboard',
                constructor: function(config) {
                    this.mergeConfig(config);
                    this.callParent([this.config]);
                },
                initComponent: function() {
                    this.callParent(arguments);
                },
                _retrieveModels: function(success) {
                    if ( this.store ) {
                        this.models = [ this.store.getProxy().getModel() ];
                        success.apply( this, arguments );
                    }
                }

            });
            /*global console, Ext */
            Ext.define('CustomApp', {
                extend: 'Rally.app.App',
                componentCls: 'app',
                items: [ { xtype: 'container', itemId: 'outer_box' }],
                launch: function() {
                    Ext.create('Rally.data.WsapiDataStore', {
                        model: "hierarchicalrequirement",
                        autoLoad: true,
                        fetch: ['Name','Predecessors','Successors','FormattedID','ObjectID','_ref'],
                        filters: [ {
                            property: 'FormattedID', operator: 'contains', value: '37'
                        } ],
                        listeners: { 
                            load: function(store,data,success) {
                                if ( data.length === 1 ) {
                                    var base_story = data[0].data;
                                    var modified_records = [];

                                    base_story._column = "base";
                                    modified_records.push( base_story );

                                    Ext.Array.each( base_story.Predecessors, function( story ) {
                                        story._column = "predecessor";
                                        modified_records.push( story );
                                    } );

                                    Ext.Array.each( base_story.Successors, function(story) {
                                        story._column = "successor";
                                        modified_records.push( story );
                                    } );

                                    Ext.define('CardModel', { 
                                        extend: 'Ext.data.Model',
                                        fields: [
                                            { name: '_ref', type: 'string' },
                                            { name: 'ObjectID', type: 'number'},
                                            { name: 'Name', type: 'string', attributeDefinition: { AttributeType: 'STRING'} },
                                            { name: 'FormattedID', type: 'string'},
                                            { name: '_column', type: 'string' },
                                            { name: 'ScheduleState', type: 'string' } ] ,
                                        getField: function(name) {
                                            if ( this.data[name] ) { 
                                                var return_field = null;
                                                Ext.Array.each( this.store.model.getFields(), function(field) {
                                                    if ( field.name === name ) {
                                                        return_field = field;
                                                    }
                                                } );

                                                return return_field;
                                            } else {
                                                return null;
                                            }
                                        } 
                                    });

                                    var cardStore = Ext.create('Ext.data.Store',{
                                        model: 'CardModel',
                                        data: modified_records
                                    });

                                    var cardboard = Ext.create('DependencyCardboard', {
                                        attribute: '_column',
                                        store: cardStore,
                                        height: 500,
                                        columns: [{
                                            xtype: 'dependencycolumn',
                                            displayValue: 'predecessor',
                                            value: 'predecessor',
                                            store: cardStore
                                        }, 
                                        {
                                            xtype: 'dependencycolumn',
                                            displayValue: 'base',
                                            value: 'base',
                                            store: cardStore
                                        },
                                        {
                                            xtype: 'dependencycolumn',
                                            displayValue: 'successor',
                                            value: 'successor',
                                            store: cardStore
                                        }]
                                    });

                                    this.down('#outer_box').add( cardboard );
                                }
                            },
                            scope: this
                        }
                    });
                }
            });

            Rally.launchApp('CustomApp', {
                name: 'cardboard'
            });
        });
    </script>

    <style type="text/css">
        .app {
             /* Add app styles here */
        }
    </style>
</head>
<body></body>
</html>