如何使用Typescript进行Cytoscape初始化?

6

我使用 TypeScript 构建了一个 React 组件,该组件使用 cytoscape(以及其types)作为无头模型。我的目标是创建一个 NPM 包,以便我可以直接在其他项目中导入它。

我的库:

  1. 当我将 cytoscape.js 作为静态资源导入到我的 index.html 中时工作
  2. 但是,当我在新的 es6 应用程序中 npm install 我的组件时,它不起作用: Uncaught ReferenceError: cytoscape is not defined at new WbsLayout。尽管在安装我的组件时 cytoscape 已自动安装到 node_modules。

Typescript源代码

WbsLayout.tsx:

export class WbsLayout  {
    //...
    public cy: Cy.Core;
    constructor(graph: any, Options?: Options) {           
        this.cy = cytoscape({ // <-- Works or fails, see cases #1 and #2 explained above
            headless: true,
            elements: graph
        });
        //...
    }
    //...
}

请注意,我在组件源代码中没有任何地方导入cytoscape(或任何类似的方式),因为我似乎不需要使用Typescript和cytoscape类型定义的方式使我的组件与在html中导入的cytoscape一起工作。也许这是一个NPN软件包的问题...

Wbs.tsx:

一个简单的React组件,接收WbsLayout属性。
cytoscape的类型定义(仅相关部分)
declare namespace Cy {
    //...
    interface Core { }

    //...
    interface Static {
        (options?: Cy.CytoscapeOptions): Core;
        (extensionName: string, foo: string, bar: any): Core;
        version: string;
    }
}

declare module "cytoscape" {
    export = cytoscape;
}
declare var cytoscape: Cy.Static;

Webpack 2配置

Wbs.tsxWbsLayout.tsx等文件会使用Webpack 2打包成@nodeject/wbs-react

module.exports = {
  entry: {
    'test/index': './src/test.tsx',
    'dist/index': './src/index.tsx'
  },

  output: {
    filename: "./[name].js",
    libraryTarget: 'umd',
    library: 'wbs-react',
    umdNamedDefine: true
  },

  devtool: "source-map",

  resolve: {
    extensions: [".webpack.js", ".web.js", ".ts", ".tsx", ".js"]
  },

  node: {
    fs: "empty",
    child_process: "empty"
  },

  module: {
    rules: [    
      {
        enforce: 'pre',
        test: /\.js$/,
        loader: "source-map-loader"
      },
           
      {
        test: /\.tsx?$/,
        loader: "ts-loader"
      },
      {
        test: /\.css$/,
        loader: "style-loader!css-loader"
      },
    ]
  },
  
  externals: {
    "react": {
      root: 'React',
      commonjs2: 'react',
      commonjs: 'react',
      amd: 'react'
    },

    "react-dom": {
      root: 'ReactDOM',
      commonjs2: 'react-dom',
      commonjs: 'react-dom',
      amd: 'react-dom'
    },

    "cytoscape": {
      root: 'Cy',
      commonjs2: 'cytoscape',
      commonjs: 'cytoscape',
      amd: 'cytoscape'
    },

    "chai": {
      root: 'chai',
      commonjs2: 'chai',
      commonjs: 'chai',
      amd: 'chai'
    }
  }
};

tsconfig.json

{
  "compilerOptions": {
    "outDir": "./temp/",
    "sourceMap": false,
    "noImplicitAny": true,
    "module": "commonjs",
    "target": "es2015",
    "allowSyntheticDefaultImports": true,
    "moduleResolution": "node",
    "jsx": "react"
  },
  "exclude": [
    "node_modules",
    "dist",
    "temp",
    "test"
  ]
}

新的es6应用程序源代码

import React from 'react';
import ReactDOM from 'react-dom';
import {Wbs, WbsLayout} from '@nodeject/wbs-react';

let graph = {
  nodes: [
    { data: { id: '1' } },
    { data: { id: '1.1', parent: '1' } }
  ],
  edges: [
    { data: { id: 'e1', source: '1', target: '1.1' } }
  ]
};

layout: new WbsLayout(graph); // <-- Fails here (case #2)

ReactDOM.render(<Wbs layout={layout}/>, document.querySelector('#wbs_example'));

由于typings导出了一个模块,所以import * as cytoscape from "cytoscape";应该可以工作。 - Saravana
谢谢回答。我之前尝试在WbsLayout.tsx中添加,但在同一位置(this.cy = cytoscape({ .....}))出现了Uncaught TypeError: cytoscape is not a function错误。 - Greg Forel
你尝试过将module目标更改为es2015,将moduleResolution选项更改为node并将allowSyntheticDefaultImports选项设置为true吗? - Daniel Rosenwasser
嘿@DanielRosenwasser,又出现了同样的错误。这可能与webpack有关吗?我阅读了一些选项,如“library=your-app-name”和其他选项,但我也没有成功。 - Greg Forel
2个回答

1

我注意到如果从webpack的外部完全移除cytoscape,问题就会得到解决。以下是我需要做的:

"cytoscape": {
  root: 'Cy',
  commonjs2: 'cytoscape',
  commonjs: 'cytoscape',
  amd: 'cytoscape'
 }

had to be:

"cytoscape": {
  root: 'cytoscape', <--------
  commonjs2: 'cytoscape',
  commonjs: 'cytoscape',
  amd: 'cytoscape'
 }

0

你尝试过将它导入为..吗?

const cytoscape = require('cytoscape');

如果可以的话,您能否同时发布您的tsconfig.json文件。

你是指 import cytoscape = require('cytoscape') 吗?同样的错误,它甚至不再作为静态资源工作:cytoscape 不是一个函数。我会编辑我的第一篇帖子以添加 tsconfig.json。 - Greg Forel

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接