如何在handsontable的上下文菜单中添加具有默认项目的自定义项目

时间:2018-05-26 00:42:57

标签: javascript jquery handsontable

我尝试使用handsontable,并希望将自定义项添加到上下文菜单中。

有很多教程可以实现自定义菜单,但它会忽略默认项目。

所以我添加了所有项目的键,但其中一些不起作用。

我的设置如下。

var basicSettings = {
    minRows: 1,
    minCols: 1,
    rowHeaders: false,
    colHeaders: false,
    hiddenColumns: true,
    contextMenu: {
        items: {
            row_above: {},
            row_below: {},
            "hsep1": "---------",
            col_left: {},
            col_right: {},
            "hsep2": "---------",
            remove_row: {},
            remove_col: {},
            "hsep3": "---------",
            undo: {},
            redo: {},
            "hsep4": "---------",
            make_read_only: {},
            "hsep5": "---------",
            alignment: {},
            "hsep6": "---------",
            copy: {},
            cut: {},
            "paste": {
                name: 'Paste',
                callback: function (key, option) {
                    this.copyPaste.triggerPaste();
                }
            },
            "hsep7": "---------",
            mergeCells: {
                name: "Merge"
            },
            "hsep8": "---------",
            // Custom menu item to set color of cells
            set_color: {
                key: "color",
                name: "Color",
                "submenu": {
                    "items": [
                        {
                            key: "color:1",
                            "name": "Black",
                            callback: function(key, options) {
                                for (var i = options[0].start.row; i <= options[0].end.row; i ++){
                                    for (var j = options[0].start.col; j <= options[0].end.col; j ++){
                                        this.getCell(i, j).className = "color-black";
                                    }
                                }
                            }
                        }, {
                            key: "color:2",
                            "name": "White",
                            callback: function(key, options) {
                                for (var i = options[0].start.row; i <= options[0].end.row; i ++){
                                    for (var j = options[0].start.col; j <= options[0].end.col; j ++){
                                        $(this.getCell(i, j)).removeClass("color-*").addClass("color-white");
                                    }
                                }
                                this.render();
                            }
                        }, {
                            key: "color:3",
                            "name": "Red",
                            callback: function(key, options) {
                                for (var i = options[0].start.row; i <= options[0].end.row; i ++){
                                    for (var j = options[0].start.col; j <= options[0].end.col; j ++){
                                        this.getCell(i, j).style.backgroundColor = "red";
                                    }
                                }
                                this.render();
                            }
                        }
                    ]
                }
            }
        }
    },
    manualRowResize: true,
    manualColumnResize: true,
    contextMenuCopyPaste: {
        swfPath: '/bower_components/zeroclipboard/dist/ZeroClipboard.swf'
    },
    copyPaste: true,
    mergeCells: true,
    search: true,
    stretchH: 'all',
    autoColumnSize: {useHeaders: true},
    autoRowSize: {syncLimit: 300},
    width: 1000,
    height: window.innerHeight - 100,
    licenseKey: "xxxxx-xxxxx-xxxxx-xxxxx-xxxxx"
};

菜单看起来像这样。

Current Menu

问题1:

有没有办法添加所有默认菜单项的自定义项?如果是这样,我不需要问题3和问题4的答案。

问题2:

让我问这个问题最重要的部分是自定义菜单,即“set_color”。 单击“黑色”或“红色”后,它会变成该颜色,但在我更改单元格的值后,它会返回。 如何防止细胞返回背景颜色?

问题3:

除了启用所有功能外,我还想要其他自定义项目。但我找不到“合并”项目的正确键。当前的“合并”功能非常奇怪。 如何使功能正常工作?

问题4:

我尝试https://github.com/handsontable/handsontable/issues/2853来实现粘贴功能,但我看到了这个错误。

Uncaught TypeError: Cannot read property 'triggerPaste' of undefined

请帮助我掌握这些用法。提前谢谢。

1 个答案:

答案 0 :(得分:2)

问题1:

  

有没有办法添加所有默认菜单项的自定义项?   如果是这样,我不需要问题3和问题4的答案。

  • Handsontable设置为contextMenu的情况下初始化true。例如:

      let
        example3 = document.getElementById('example3'),
        settings3,
        hot3;
    
      settings3 = {
        data: [...],
        rowHeaders: true,
        colHeaders: true,
        contextMenu: true // set to `true`..
      };
      hot3 = new Handsontable(example3, settings3);
    

  • 然后像这样更新contextMenu设置:

      let cm = hot3.getPlugin('ContextMenu');
      hot3.updateSettings({
        contextMenu: {
            // Clone the pre-defined items and add your custom items.
          items: Object.assign({}, cm.itemsFactory.predefinedItems, {
            'hsep1': '---------',
            'set_color': {
                key: 'color',
                name: 'Color',
              submenu: {
                items: [{
                  key: 'color:red',
                  name: 'Red',
                  callback: setCellColor
                }, {
                  key: 'color:blue',
                  name: 'Blue',
                  callback: setCellColor
                }]
              }
            }
          })
        }
      });
    

  • 问题2:

      

    让我问这个问题最重要的部分是自定义菜单,   那就是“set_color”。单击“黑色”或“红色”后,它变为   那种颜色,但在我改变一个单元格的值后,它会回头。怎么能   我阻止细胞变回背景颜色?

    我不确定如何阻止;但是,这是恢复相应单元格的颜色(或任何其他自定义/元数据..)的一种方法。

      // This is my callback for the 'set_color' context menu items.
      // Sample `key`: 'color:red'
      function setCellColor(key, opt) {
        let color = key.substring(6);
        for (let i = opt[0].start.row; i <= opt[0].end.row; i++) {
          for (let j = opt[0].start.col; j <= opt[0].end.col; j++) {
            this.getCell(i, j).style.color = color;
            this.setCellMeta(i, j, 'color', color); // Save the color
          }
        }
      }
    
      // Listen to the `beforeRenderer` event, and restore the cell's color
      // before the "renderer" starting rendering the cell
      Handsontable.hooks.add('beforeRenderer', function(td, r, c, p, pv, cp){
        if (cp.color) {
            td.style.color = cp.color;
        }
      }, hot3);
    

    演示

    点击此处的完整示例:http://jsfiddle.net/y9j3m29c/1/,基于https://docs.handsontable.com/3.0.0/demo-context-menu.html#page-custom

    对于粘贴功能:

  • 加载SheetClip()

    <script src="https://unpkg.com/sheetclip"></script>
    

  • 添加必要的变量:

    let clipboardCache = '';
    const sheetclip = new SheetClip();
    

  • 添加必要的回调:

      settings3 = {
        ...
        afterCopy: function(changes){
            clipboardCache = sheetclip.stringify(changes);
        },
        afterCut: function(changes){
            clipboardCache = sheetclip.stringify(changes);
        },
        afterPaste: function(changes){
            clipboardCache = sheetclip.stringify(changes);
        }
      };
    

  • 添加上下文菜单项:

      let cm = hot3.getPlugin('ContextMenu');
      hot3.updateSettings({
        contextMenu: {
            // Clone the pre-defined items and add your custom items.
          items: Object.assign({}, cm.itemsFactory.predefinedItems, {
            ...
            'paste': {
                name: 'Paste',
              disabled: function(){
                return clipboardCache.length === 0;
              },
              callback: function(){
                var plugin = this.getPlugin('copyPaste');
    
                this.listen();
                plugin.paste(clipboardCache);
              }
            }
          })
        }
      });
    

  • 关于https://docs.handsontable.com/3.0.0/demo-copy-paste.html#paste-in-context-menu的详细信息以及http://jsfiddle.net/y9j3m29c/2/上的演示 - 或http://jsfiddle.net/y9j3m29c/4/附带“清除剪贴板”功能。