views:

300

answers:

2

I have an ItemRenderer that is shared by several applications (inside a DataGrid), and I would like to add a context menu to it (rather than in each application). The renderer is derived from the Canvas class, and the code to create the context menu looks something like:

            var menuItem:ContextMenuItem = new ContextMenuItem("Test");
            menuItem.addEventListener(ContextMenuEvent.MENU_ITEM_SELECT, contextCallback);

            var customContextMenu:ContextMenu = new ContextMenu();              
            customContextMenu.hideBuiltInItems(); //hide flash menu
            customContextMenu.customItems.push(menuItem);               
            this.contextMenu = customContextMenu;

However, when I right click on the cell in the datagrid, I get the default Flash Context Menu. Is this not possible?

Edit: Here is a fully runnable example, which doesn't show the context menu's:

Application.mxml:

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"
    layout="vertical"
    verticalAlign="middle"
    backgroundColor="white">

<mx:Model id="items">
    <items>
        <item label="Item 1" data="i001" propertyA="Item 1.A" propertyB="Item 1.B" />
        <item label="Item 2" data="i002" propertyA="Item 2.A" propertyB="Item 2.B" />
        <item label="Item 3" data="i003" propertyA="Item 3.A" propertyB="Item 3.B" />
        <item label="Item 4" data="i004" propertyA="Item 4.A" propertyB="Item 4.B" />
        <item label="Item 5" data="i005" propertyA="Item 5.A" propertyB="Item 5.B" />
        <item label="Item 6" data="i006" propertyA="Item 6.A" propertyB="Item 6.B" />
        <item label="Item 7" data="i007" propertyA="Item 7.A" propertyB="Item 7.B" />
        <item label="Item 8" data="i008" propertyA="Item 8.A" propertyB="Item 8.B" />
    </items>
</mx:Model>

<mx:DataGrid id="dataGrid" width="400" dataProvider="{items.item}">
    <mx:columns>
        <mx:DataGridColumn headerText="No Menu" dataField="label" />
        <mx:DataGridColumn headerText="Menu" dataField="propertyA" itemRenderer="canvasRenderer"/>
    </mx:columns>
</mx:DataGrid>

canvasRenderer.mxml:

<mx:Canvas xmlns:mx="http://www.adobe.com/2006/mxml" width="100%" verticalScrollPolicy="off" horizontalScrollPolicy="off" resize="this.setSize()" creationComplete="init()" implements="mx.controls.listClasses.IDropInListItemRenderer">
<mx:Script>
    <![CDATA[
        import mx.controls.listClasses.ListData;
        import mx.controls.dataGridClasses.DataGridListData;
        import mx.controls.listClasses.BaseListData;
        import mx.events.FlexEvent;

        private var _listData:DataGridListData;

        private function init():void {
            var menuItem:ContextMenuItem = new ContextMenuItem("Copy", true);
            menuItem.addEventListener(ContextMenuEvent.MENU_ITEM_SELECT, function():void {
                trace("selected");
            });

            var menu:ContextMenu = new ContextMenu();   
            menu.hideBuiltInItems(); //hide flash menu  
            menu.customItems.push(menuItem);    
            this.contextMenu = menu;
        }

        public override function set data(value:Object):void {
            super.data = value;
            if(_listData && myLabel) {
                var text:String = _listData.label ? _listData.label : value[_listData.dataField];
                myLabel.text = text;
                myLabel.toolTip = text;
            } 
        }

        public function get listData():BaseListData { return _listData; }
        public function set listData(value:BaseListData):void { _listData = DataGridListData(value); }
        public function setSize():void { myLabel.width = this.width; }
    ]]>
</mx:Script>

<mx:Label id="myLabel" truncateToFit="true"/>

+1  A: 

I've added the context menu to an itemRenderer in a List with the Flextras DataSorter. However, I don't see any difference between what I did and what you're doing.

Source code here: http://www.flextras.com/DataSorter/Samples/RightClick/srcview/index.html

and the running Sample http://www.flextras.com/DataSorter/Samples/RightClick/

Can you share more code? What is the "this" that you're adding the context menu too? Is your temRenderer in-line or a separate component?

www.Flextras.com
I've provided a runnable example. It seems like it has something to do with it being wrapped inside the canvas, but thats just a guess
gmoniey
Change the text from "Copy" to something else and your code starts magically working. "Copy" must be a reserved word or something similar.
www.Flextras.com
Ahh...very frustrating. Suppose its my fault. I actually changed the text in my first sample, but never tried running that code.Thanks for catching that.
gmoniey
+2  A: 

This works:

<?xml version="1.0" encoding="utf-8"?>
<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"
        layout="vertical"
        verticalAlign="middle"
        backgroundColor="white"
        creationComplete="init()">

    <mx:Script>
        <![CDATA[
            import mx.controls.Alert;

            [Bindable] public var cm1:ContextMenu;
            [Bindable] public var cm2:ContextMenu;

            private function init():void {
                var cmi1:ContextMenuItem = new ContextMenuItem("View item in column 1...", true);
                cmi1.addEventListener(ContextMenuEvent.MENU_ITEM_SELECT, contextMenuItem_menuItemSelect);

                cm1 = new ContextMenu();
                cm1.hideBuiltInItems();
                cm1.customItems = [cmi1];
                cm1.addEventListener(ContextMenuEvent.MENU_SELECT, contextMenu_menuSelect);

                var cmi2:ContextMenuItem = new ContextMenuItem("View item in column 2...", true);
                cmi2.addEventListener(ContextMenuEvent.MENU_ITEM_SELECT, contextMenuItem_menuItemSelect);

                cm2 = new ContextMenu();
                cm2.hideBuiltInItems();
                cm2.customItems = [cmi2];
                cm2.addEventListener(ContextMenuEvent.MENU_SELECT, contextMenu_menuSelect);
            }

            private function contextMenu_menuSelect(evt:ContextMenuEvent):void {
                dataGrid.selectedIndex = lastRollOverIndex;
            }

            private function contextMenuItem_menuItemSelect(evt:ContextMenuEvent):void {
                var obj:Object = dataGrid.selectedItem;
                Alert.show("Property A: " + obj.propertyA + "\n" + "Property B: " + obj.propertyB, obj.label, Alert.OK);
            }
        ]]>
    </mx:Script>

    <mx:Model id="items">
        <items>
            <item label="Item 1" data="i001" propertyA="Item 1.A" propertyB="Item 1.B" />
            <item label="Item 2" data="i002" propertyA="Item 2.A" propertyB="Item 2.B" />
            <item label="Item 3" data="i003" propertyA="Item 3.A" propertyB="Item 3.B" />
            <item label="Item 4" data="i004" propertyA="Item 4.A" propertyB="Item 4.B" />
            <item label="Item 5" data="i005" propertyA="Item 5.A" propertyB="Item 5.B" />
            <item label="Item 6" data="i006" propertyA="Item 6.A" propertyB="Item 6.B" />
            <item label="Item 7" data="i007" propertyA="Item 7.A" propertyB="Item 7.B" />
            <item label="Item 8" data="i008" propertyA="Item 8.A" propertyB="Item 8.B" />
        </items>
    </mx:Model>

    <mx:Number id="lastRollOverIndex" />

    <mx:DataGrid id="dataGrid" width="400" dataProvider="{items.item}">
        <mx:columns>
            <mx:DataGridColumn headerText="Label">
              <mx:itemRenderer>
                <mx:Component>
                  <mx:Label text="{data.label}" contextMenu="{outerDocument.cm1}"/>
                </mx:Component>
              </mx:itemRenderer>
            </mx:DataGridColumn>

            <mx:DataGridColumn headerText="Property A">
              <mx:itemRenderer>
                <mx:Component>
                  <mx:Label text="{data.propertyA}" contextMenu="{outerDocument.cm2}"/>
                </mx:Component>
              </mx:itemRenderer>
            </mx:DataGridColumn>
        </mx:columns>
    </mx:DataGrid>

    <mx:Label text="{dataGrid.selectedItem.label}" />

</mx:Application>
James Ward
The question is why! does your sample seem to work with an in-line renderer; but his sample not seem to work with an external renderer. I couldn't get his sample to work either.
www.Flextras.com
Got it! "Copy" won't show up. It must be a reserved word or something similar. Changing the text of his ContextMenuItem made things magically work.
www.Flextras.com
Ahh. Yes. Flash Player does not permit certain words in a custom context menu. Possible workaround: http://www.themorphicgroup.com/blog/2009/07/06/contextmenu-copy-paste-delete/
James Ward
One point for the work around link. ;)
www.Flextras.com