Skip to content

Instantly share code, notes, and snippets.

@timelyportfolio
Last active November 19, 2017 14:43
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save timelyportfolio/4acc0624527bfc9fa78ce15144b90b21 to your computer and use it in GitHub Desktop.
Save timelyportfolio/4acc0624527bfc9fa78ce15144b90b21 to your computer and use it in GitHub Desktop.
react + antd from R with reactR
license: mit
height: 600
scrolling: yes

Just because we use R does not mean we can't play with the awesome JavaScript tools React and antd-design.

Example

bl.ock

Code

nearly all JavaScript from this ant-design example

library(htmltools)
library(reactR)
library(pipeR)

antd <- htmlDependency(
  name = "antd",
  version = "2.13.10",
  src = c(href="https://unpkg.com/antd/dist/"),
  script = "antd.min.js",
  stylesheet = "antd.min.css"
)

### steps with button ####
steps_button <- list(
  list(
    title= 'Data',
    content= 'Raw Data'
  ),
  list(
    title= 'Model',
    content= 'Code for Model'
  ),
  list(
    title= 'Plot',
    content= 'Beautiful Plot'
  )
) %>>% jsonlite::toJSON(auto_unbox=TRUE)

content_data <- tags$pre(
  HTML(paste0(
    capture.output(str(iris,max.level=1)),
    collapse="<br/>"
  ))
)

content_model <- tags$pre("lm(Petal.Width~Petal.Length, data=iris)")

content_plot <- HTML(
  svglite::htmlSVG({plot(lm(Petal.Length~Petal.Width,data=iris),which=1)},standalone=FALSE)
)

tagList(
  tags$div(id="stepapp", style="width:30%;"),
  steps_button %>>%
    {
      sprintf(
'
const steps = %s;

steps[0].content = %s;
steps[1].content = %s;
steps[2].content = <img src="%s" />

class App extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      current: 0,
    };
  }
  next() {
    const current = this.state.current + 1;
    this.setState({ current });
  }
  prev() {
    const current = this.state.current - 1;
    this.setState({ current });
  }
  render() {
    const { current } = this.state;
    return (
      <div>
        <antd.Steps current={current} size="small">
          {steps.map(item => <antd.Steps.Step key={item.title} title={item.title} />)}
        </antd.Steps>
      <div className="steps-content">{steps[this.state.current].content}</div>
      <div className="steps-action">
        {
          this.state.current < steps.length - 1
          &&
          <antd.Button type="primary" onClick={() => this.next()}>Next</antd.Button>
        }
        {
          this.state.current === steps.length - 1
          &&
          <antd.Button type="primary" onClick={() => message.success("Processing complete!")}>Done</antd.Button>
        }
        {
          this.state.current > 0
          &&
          <antd.Button style={{ marginLeft: 8 }} type="ghost" onClick={() => this.prev()}>
            Previous
          </antd.Button>
        }
        </div>
      </div>
    );
  }
}

ReactDOM.render(<App />, document.querySelector("#stepapp"));
      ',
      .,
      content_data,
      content_model,
      base64enc::dataURI(rsvg::rsvg_png(charToRaw(content_plot)),mime="image/png")
    )
  } %>>%
    babel_transform() %>>%
    HTML %>>%
    tags$script()
) %>>%
  attachDependencies(
    list(
      html_dependency_react(offline=FALSE),
      antd
    )
  ) %>>%
  browsable()

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8"/>
<script src="//unpkg.com/react/umd/react.production.min.js"></script>
<script src="//unpkg.com/react-dom/umd/react-dom.production.min.js"></script>
<link href="https://unpkg.com/antd/dist/antd.min.css" rel="stylesheet" />
<script src="https://unpkg.com/antd/dist/antd.min.js"></script>
</head>
<body style="background-color:white;">
<div id="stepapp" style="width:30%;"></div>
<script>"use strict";
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
var steps = [{ "title": "Data", "content": "Raw Data" }, { "title": "Model", "content": "Code for Model" }, { "title": "Plot", "content": "Beautiful Plot" }];
steps[0].content = React.createElement(
"pre",
null,
"'data.frame': 150 obs. of 5 variables:",
React.createElement("br", null),
" $ Sepal.Length: num 5.1 4.9 4.7 4.6 5 5.4 4.6 5 4.4 4.9 ...",
React.createElement("br", null),
" $ Sepal.Width : num 3.5 3 3.2 3.1 3.6 3.9 3.4 3.4 2.9 3.1 ...",
React.createElement("br", null),
" $ Petal.Length: num 1.4 1.4 1.3 1.5 1.4 1.7 1.4 1.5 1.4 1.5 ...",
React.createElement("br", null),
" $ Petal.Width : num 0.2 0.2 0.2 0.2 0.2 0.4 0.3 0.2 0.2 0.1 ...",
React.createElement("br", null),
" $ Species : Factor w/ 3 levels \"setosa\",\"versicolor\",..: 1 1 1 1 1 1 1 1 1 1 ..."
);
steps[1].content = React.createElement(
"pre",
null,
"lm(Petal.Width~Petal.Length, data=iris)"
);
steps[2].content = React.createElement("img", { src: "" });
var App = function (_React$Component) {
_inherits(App, _React$Component);
function App(props) {
_classCallCheck(this, App);
var _this = _possibleConstructorReturn(this, (App.__proto__ || Object.getPrototypeOf(App)).call(this, props));
_this.state = {
current: 0
};
return _this;
}
_createClass(App, [{
key: "next",
value: function next() {
var current = this.state.current + 1;
this.setState({ current: current });
}
}, {
key: "prev",
value: function prev() {
var current = this.state.current - 1;
this.setState({ current: current });
}
}, {
key: "render",
value: function render() {
var _this2 = this;
var current = this.state.current;
return React.createElement(
"div",
null,
React.createElement(
antd.Steps,
{ current: current, size: "small" },
steps.map(function (item) {
return React.createElement(antd.Steps.Step, { key: item.title, title: item.title });
})
),
React.createElement(
"div",
{ className: "steps-content" },
steps[this.state.current].content
),
React.createElement(
"div",
{ className: "steps-action" },
this.state.current < steps.length - 1 && React.createElement(
antd.Button,
{ type: "primary", onClick: function onClick() {
return _this2.next();
} },
"Next"
),
this.state.current === steps.length - 1 && React.createElement(
antd.Button,
{ type: "primary", onClick: function onClick() {
return message.success("Processing complete!");
} },
"Done"
),
this.state.current > 0 && React.createElement(
antd.Button,
{ style: { marginLeft: 8 }, type: "ghost", onClick: function onClick() {
return _this2.prev();
} },
"Previous"
)
)
);
}
}]);
return App;
}(React.Component);
ReactDOM.render(React.createElement(App, null), document.querySelector("#stepapp"));</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment