diff --git a/components/index.js b/components/index.js
index 25ed0ee182..a008145e9f 100644
--- a/components/index.js
+++ b/components/index.js
@@ -253,6 +253,8 @@ export TrialBarButton from './trial-bar/button';
export SLDSTrialBarButton from './trial-bar/button';
export TrialBarDropdown from './trial-bar/dropdown';
export SLDSTrialBarDropdown from './trial-bar/dropdown';
+export SLDSTreeGrid from './tree-grid';
+export TreeGrid from './tree-grid';
export UNSAFE_DirectionSettings from './utilities/UNSAFE_direction';
export UtilityIcon from './utilities/utility-icon';
export SLDSUtilityIcon from './utilities/utility-icon';
diff --git a/components/tree-grid/__docs__/storybook-stories.jsx b/components/tree-grid/__docs__/storybook-stories.jsx
new file mode 100644
index 0000000000..29e48909cd
--- /dev/null
+++ b/components/tree-grid/__docs__/storybook-stories.jsx
@@ -0,0 +1,26 @@
+import React from 'react';
+import { storiesOf } from '@storybook/react';
+import { action } from '@storybook/addon-actions';
+
+import { TREE_GRID } from '../../../utilities/constants';
+import Default from '../__examples__/default';
+import Headless from '../__examples__/headless';
+
+storiesOf(TREE_GRID, module)
+ .addDecorator((getStory) => (
+
+
+ {
+ log({
+ action: this.props.action,
+ event,
+ eventName: 'More actions button of row clicked',
+ data,
+ });
+ }}
+ options={[
+ { label: 'Menu Item One', value: 'A0' },
+ { label: 'Menu Item Two', value: 'B0' },
+ ]}
+ value="A0"
+ />
+ }
+ >
+ {
+ log({
+ action: this.props.action,
+ event,
+ eventName:
+ 'More Actions Button of accountName column clicked',
+ data,
+ });
+ }}
+ options={[
+ { label: 'Menu Item One', value: 'A0' },
+ { label: 'Menu Item Two', value: 'B0' },
+ ]}
+ value="A0"
+ />
+ }
+ />
+ {
+ log({
+ action: this.props.action,
+ event,
+ eventName:
+ 'More Actions Button of employees column clicked',
+ data,
+ });
+ }}
+ options={[
+ { label: 'Menu Item One', value: 'A0' },
+ { label: 'Menu Item Two', value: 'B0' },
+ ]}
+ value="A0"
+ />
+ }
+ />
+ {
+ log({
+ action: this.props.action,
+ event,
+ eventName: 'More Actions Button of phone column clicked',
+ data,
+ });
+ }}
+ options={[
+ { label: 'Menu Item One', value: 'A0' },
+ { label: 'Menu Item Two', value: 'B0' },
+ ]}
+ value="A0"
+ />
+ }
+ />
+ {
+ log({
+ action: this.props.action,
+ event,
+ eventName:
+ 'More Actions Button of accountOwner column clicked',
+ data,
+ });
+ }}
+ options={[
+ { label: 'Menu Item One', value: 'A0' },
+ { label: 'Menu Item Two', value: 'B0' },
+ ]}
+ value="A0"
+ />
+ }
+ />
+ {
+ log({
+ action: this.props.action,
+ event,
+ eventName: 'More Actions Button of city column clicked',
+ data,
+ });
+ }}
+ options={[
+ { label: 'Menu Item One', value: 'A0' },
+ { label: 'Menu Item Two', value: 'B0' },
+ ]}
+ value="A0"
+ />
+ }
+ />
+
+
+
+ );
+ }
+}
+
+export default Example;
diff --git a/components/tree-grid/__examples__/headless.jsx b/components/tree-grid/__examples__/headless.jsx
new file mode 100644
index 0000000000..d60e9573c0
--- /dev/null
+++ b/components/tree-grid/__examples__/headless.jsx
@@ -0,0 +1,217 @@
+import React from 'react';
+
+import TreeGrid from '~/components/tree-grid';
+import Dropdown from '~/components/menu-dropdown';
+import TreeGridColumn from '~/components/tree-grid/column';
+import IconSettings from '~/components/icon-settings';
+
+import log from '~/utilities/log';
+
+const sampleData = {
+ 0: {
+ id: 0,
+ nodes: [1, 2, 3, 4],
+ },
+ 1: {
+ id: 1,
+ type: 'item',
+ name: '123555',
+ accountName: 'Rewis Inc',
+ },
+ 2: {
+ id: 2,
+ type: 'branch',
+ name: '123556',
+ accountName: 'Acme Corporation',
+ nodes: [5, 6],
+ },
+ 3: {
+ id: 3,
+ type: 'branch',
+ name: '123557',
+ accountName: 'Rhode Enterprises',
+ nodes: [7],
+ },
+ 4: {
+ id: 4,
+ type: 'branch',
+ name: '123558',
+ accountName: 'Tech Labs',
+ nodes: [8],
+ },
+ 5: {
+ id: 5,
+ type: 'branch',
+ name: '123556-A',
+ accountName: 'Acme Corporation (Bay Area)',
+ nodes: [9, 10],
+ },
+ 6: {
+ id: 6,
+ type: 'branch',
+ name: '123556-B',
+ accountName: 'Acme Corporation (East)',
+ nodes: [11, 12],
+ },
+ 7: {
+ id: 7,
+ type: 'item',
+ name: '123557-A',
+ accountName: 'Rhode Enterprises (UCA)',
+ },
+ 8: {
+ id: 8,
+ type: 'item',
+ name: '123558-A',
+ accountName: 'Opportunity Resources Inc',
+ },
+ 9: {
+ id: 9,
+ type: 'item',
+ name: '123556-A-A',
+ accountName: 'Acme Corporation (Oakland)',
+ },
+ 10: {
+ id: 10,
+ type: 'item',
+ name: '123556-A-B',
+ accountName: 'Acme Corporation (San Francisco)',
+ },
+ 11: {
+ id: 11,
+ type: 'item',
+ name: '123556-B-A',
+ accountName: 'Acme Corporation (NY)',
+ },
+ 12: {
+ id: 12,
+ type: 'branch',
+ name: '123556-B-B',
+ accountName: 'Acme Corporation (VA)',
+ nodes: [13],
+ },
+ 13: {
+ id: 13,
+ type: 'branch',
+ name: '123556-B-B-A',
+ accountName: 'Allied Technologies',
+ nodes: [14],
+ },
+ 14: {
+ id: 14,
+ type: 'item',
+ name: '123556-B-B-A-A',
+ accountName: 'Allied Technologies (UV)',
+ },
+};
+
+class Example extends React.Component {
+ static displayName = 'TreeGridExample';
+
+ state = {
+ nodes: this.props.nodes || sampleData,
+ };
+
+ getNodes = (node) =>
+ node.nodes ? node.nodes.map((id) => this.state.nodes[id]) : [];
+
+ handleExpansion = (event, data) => {
+ log({
+ action: this.props.action,
+ event,
+ eventName: 'Expand Branch',
+ data,
+ });
+ const { nodes } = this.state;
+ const updated = {
+ ...nodes,
+ ...{
+ [data.node.id]: {
+ ...data.node,
+ expanded: data.expanded,
+ },
+ },
+ };
+ this.setState({ nodes: updated });
+ };
+
+ findChildren = (node) => {
+ if (node.type === 'branch') {
+ let list = [];
+ node.nodes.forEach((child) => {
+ const c = this.findChildren(this.state.nodes[child]);
+ list = [...c, child, ...list];
+ });
+ return list;
+ }
+ return [];
+ };
+
+ handleSelection = (event, data) => {
+ log({
+ action: this.props.action,
+ event,
+ eventName: 'Select Branch',
+ data,
+ });
+ const curr = this.state.nodes;
+ curr[data.node.id].selected = data.selected;
+ const children = this.findChildren(data.node);
+ children.forEach((child) => {
+ curr[child].selected = data.selected;
+ });
+ this.setState({ nodes: curr });
+ };
+
+ render() {
+ return (
+
+
+ {
+ log({
+ action: this.props.action,
+ event,
+ eventName: 'More actions button of row clicked',
+ data,
+ });
+ }}
+ options={[
+ { label: 'Menu Item One', value: 'A0' },
+ { label: 'Menu Item Two', value: 'B0' },
+ ]}
+ value="A0"
+ />
+ }
+ >
+
+
+
+
+ );
+ }
+}
+
+export default Example;
diff --git a/components/tree-grid/column.jsx b/components/tree-grid/column.jsx
new file mode 100644
index 0000000000..a7168906ac
--- /dev/null
+++ b/components/tree-grid/column.jsx
@@ -0,0 +1,123 @@
+/* Copyright (c) 2015-present, salesforce.com, inc. All rights reserved */
+/* Licensed under BSD 3-Clause - see LICENSE.txt or git.io/sfdc-license */
+
+/* eslint-disable react/no-unused-prop-types */
+/* deepscan-disable REACT_USELESS_PROP_TYPES */
+
+// ### React
+import React from 'react';
+import PropTypes from 'prop-types';
+import classNames from 'classnames';
+import isFunction from 'lodash.isfunction';
+
+import Icon from '../icon';
+
+// ## Constants
+import { TREE_GRID_COLUMN } from '../../utilities/constants';
+
+/**
+ * Columns define the structure of the data displayed in the DataTable.
+ */
+const TreeGridColumn = (props) => {
+ let { sortDirection } = props;
+ if (props.isDefaultSortDescending && !sortDirection) sortDirection = 'desc';
+
+ return (
+