Introduction
Welcome to RDKit.js for React. If you didn't do it already, we first recommend to go through the JavaScript examples for a low-level overview of the RDKit.js API.
Overview
You can draw any molecule from SMILES dynamically with RDKit.js .
import React from "react";
import MoleculeStructure from "../components/MoleculeStructure/MoleculeStructure";
import { SMILES_LIST } from "../utils/smiles";
class ExampleList extends React.Component {
render() {
return (
<div id="component-example-list" className="container">
<section className="hero">
<div className="hero-body">
<p className="title">Overview</p>
<p className="subtitle">
You can draw any molecule from SMILES dynamically with RDKit.js .
</p>
</div>
</section>
<div
id="structure-list"
className="columns is-desktop"
style={{ margin: "12px", overflowX: "scroll" }}
>
{SMILES_LIST.map((smiles) => (
<div className="column" key={smiles}>
<MoleculeStructure
id={smiles}
structure={smiles}
height={200}
width={200}
svgMode
/>
</div>
))}
</div>
</div>
);
}
}
export default ExampleList;
MoleculeStructure
The examples below will mostly use the React component MoleculeStructure
to render the examples. You can see the React implementation in the code below.
import React, { Component } from "react";
import _ from "lodash";
import PropTypes from "prop-types";
import "./MoleculeStructure.css";
import initRDKit from "../../utils/initRDKit";
class MoleculeStructure extends Component {
static propTypes = {
/**
* Generic properties
*/
id: PropTypes.string.isRequired,
className: PropTypes.string,
svgMode: PropTypes.bool,
width: PropTypes.number,
height: PropTypes.number,
/**
* RDKit-specific properties
*/
structure: PropTypes.string.isRequired,
subStructure: PropTypes.string,
extraDetails: PropTypes.object,
drawingDelay: PropTypes.number
};
static defaultProps = {
subStructure: "",
className: "",
width: 250,
height: 200,
svgMode: false,
extraDetails: {},
drawingDelay: undefined
};
constructor(props) {
super(props);
this.MOL_DETAILS = {
width: this.props.width,
height: this.props.height,
bondLineWidth: 1,
addStereoAnnotation: true,
...this.props.extraDetails
};
this.state = {
svg: undefined,
rdKitLoaded: false,
rdKitError: false
};
}
drawOnce = (() => {
let wasCalled = false;
return () => {
if (!wasCalled) {
wasCalled = true;
this.draw();
}
};
})();
draw() {
if (this.props.drawingDelay) {
setTimeout(() => {
this.drawSVGorCanvas();
}, this.props.drawingDelay);
} else {
this.drawSVGorCanvas();
}
}
drawSVGorCanvas() {
const mol = window.RDKit.get_mol(this.props.structure || "invalid");
const qmol = window.RDKit.get_qmol(this.props.subStructure || "invalid");
const isValidMol = this.isValidMol(mol);
if (this.props.svgMode && isValidMol) {
const svg = mol.get_svg_with_highlights(this.getMolDetails(mol, qmol));
this.setState({ svg });
} else if (isValidMol) {
const canvas = document.getElementById(this.props.id);
mol.draw_to_canvas_with_highlights(canvas, this.getMolDetails(mol, qmol));
}
/**
* Delete C++ mol objects manually
* https://emscripten.org/docs/porting/connecting_cpp_and_javascript/embind.html#memory-management
*/
mol?.delete();
qmol?.delete();
}
isValidMol(mol) {
return !!mol;
}
getMolDetails(mol, qmol) {
if (this.isValidMol(mol) && this.isValidMol(qmol)) {
const subStructHighlightDetails = JSON.parse(
mol.get_substruct_matches(qmol)
);
const subStructHighlightDetailsMerged = !_.isEmpty(
subStructHighlightDetails
)
? subStructHighlightDetails.reduce(
(acc, { atoms, bonds }) => ({
atoms: [...acc.atoms, ...atoms],
bonds: [...acc.bonds, ...bonds]
}),
{ bonds: [], atoms: [] }
)
: subStructHighlightDetails;
return JSON.stringify({
...this.MOL_DETAILS,
...(this.props.extraDetails || {}),
...subStructHighlightDetailsMerged
});
} else {
return JSON.stringify({
...this.MOL_DETAILS,
...(this.props.extraDetails || {})
});
}
}
componentDidMount() {
initRDKit()
.then(() => {
this.setState({ rdKitLoaded: true });
try {
this.draw();
} catch (err) {
console.log(err);
}
})
.catch((err) => {
console.log(err);
this.setState({ rdKitError: true });
});
}
componentDidUpdate(prevProps) {
if (
!this.state.rdKitError &&
this.state.rdKitLoaded &&
!this.props.svgMode
) {
this.drawOnce();
}
if (this.state.rdKitLoaded) {
const shouldUpdateDrawing =
prevProps.structure !== this.props.structure ||
prevProps.svgMode !== this.props.svgMode ||
prevProps.subStructure !== this.props.subStructure ||
prevProps.width !== this.props.width ||
prevProps.height !== this.props.height ||
!_.isEqual(prevProps.extraDetails, this.props.extraDetails);
if (shouldUpdateDrawing) {
this.draw();
}
}
}
render() {
if (this.state.rdKitError) {
return "Error loading renderer.";
}
if (!this.state.rdKitLoaded) {
return "Loading renderer...";
}
const mol = window.RDKit.get_mol(this.props.structure || "invalid");
const isValidMol = this.isValidMol(mol);
mol?.delete();
if (!isValidMol) {
return (
<span title={`Cannot render structure: ${this.props.structure}`}>
Render Error.
</span>
);
} else if (this.props.svgMode) {
return (
<div
title={this.props.structure}
className={"molecule-structure-svg " + (this.props.className || "")}
style={{ width: this.props.width, height: this.props.height }}
dangerouslySetInnerHTML={{ __html: this.state.svg }}
></div>
);
} else {
return (
<div
className={
"molecule-canvas-container " + (this.props.className || "")
}
>
<canvas
title={this.props.structure}
id={this.props.id}
width={this.props.width}
height={this.props.height}
></canvas>
</div>
);
}
}
}
export default MoleculeStructure;
SVG Rendering
You can render molecules using svg.
import MoleculeStructure from "../components/MoleculeStructure/MoleculeStructure";
function ExampleSVG() {
const caffeine = "CN1C=NC2=C1C(=O)N(C(=O)N2C)";
const aspirin = "CC(=O)Oc1ccccc1C(=O)O";
return (
<div id="component-example-svg" className="container">
<section className="hero">
<div className="hero-body">
<p className="title">SVG Rendering</p>
<p className="subtitle">You can render molecules using svg.</p>
</div>
</section>
<div className="columns is-desktop">
<div className="column">
<MoleculeStructure
id="structure-example-svg-caffeine"
structure={caffeine}
width={350}
height={300}
svgMode
/>
</div>
<div className="column">
<MoleculeStructure
id="structure-example-svg-aspirin"
structure={aspirin}
width={350}
height={300}
svgMode
/>
</div>
</div>
</div>
);
}
export default ExampleSVG;
Where to go next?
You can visit the JavaScript examples here, which presents the low-level API of RDKit.js.
You can visit the Vue examples here, which include more advanced and interactive examples.
You can visit the Angular examples here, which include more advanced and interactive examples.
You can also checkout the legacy getting started and demo.