I built a React component with Typescript that uses cytoscape (and its types) as a headless model. My objective is to create an NPM package so I can import it in other projects directly.
My library:
- Works when I import cytoscape.js in my index.html as a static asset
- Does not when I
npm install
my component in a new es6 app:Uncaught ReferenceError: cytoscape is not defined at new WbsLayout
. This, despite cytoscape being automatically installed in node_modules when installing my component.
Typescript source
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
});
//...
}
//...
}
Note that I don't import cytoscape from 'cytoscape'
(or any similar ways) anywhere in my component source, as I didn't have any as it seems that with Typescript and the way cytoscape's typings are defined, I didn't need it to make my component work with cytoscape imported in html. Maybe it's an issue for an NPN package though...
Wbs.tsx:
A simple React component that receives the prop WbsLayout.
cytoscape's typings (only relevant part)
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 config
Wbs.tsx
, WbsLayout.tsx
and others are bundled into @nodeject/wbs-react
using Webpack 2:
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"
]
}
New es6 app source
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'));
import * as cytoscape from "cytoscape";
should work. – BiplaneWbsLayout.tsx
, but got anUncaught TypeError: cytoscape is not a function
error at the very same spot (this.cy = cytoscape({ .....})
. – Deutziamodule
target toes2015
, yourmoduleResolution
option tonode
and setting theallowSyntheticDefaultImports
option totrue
? – Prestidigitationlibrary=your-app-name
and others, but I haven't been successful with those either. – Deutzia