Integrating ReactJS into SharePoint
Linus Falck-Ytter
Boston Code Camp, April 7th, 2018
Software Developer @ MIT Sloan
@lifayt
slides.com/lifayt
github.com/lifayt
data:image/s3,"s3://crabby-images/07c12/07c122491f0cb797f7e813edd17f8dcb776e5bee" alt=""
Pop Quiz
data:image/s3,"s3://crabby-images/450fc/450fcbfc686c98cfe962294cac90e1f3443b6489" alt=""
Pop Quiz
data:image/s3,"s3://crabby-images/ff836/ff83655a2f7e3cf1e753de1bf9ba2315029bd46c" alt=""
`Most Dreaded`
data:image/s3,"s3://crabby-images/819a8/819a8dbd8a16502526f6b642eeb52ef3691aa6c7" alt=""
SharePoint Development
- Master Page/Page Layout
- Web Parts
- Bundled Solutions & Apps
- Content Editors
- Script Editors
- Iframes(?)
- File Deployment
- WSPs
- File Libraries
SharePoint Web Development
Content Editor:
- Links to an external Markup Document
- Allows manipulation of Text & Markup wysiwyg style
Script Editor:
- Write JavaScript directly onto the page
SharePoint Web Development
Content Editor:
- Links to an external Markup Document
Allows manipulation of Text & Markup wysiwyg style
Script Editor:
Write JavaScript directly onto the page
SharePoint Web Development
SharePoint Document Library:
- Deployable from anywhere
- No downtime
- Site-wide(ish)
Deployed Solution:
- Visual Studio & Package
- Client side & server side
- Downtime
- Site-wide
SharePoint Web Development
SharePoint Document
Library:
- Deployable from anywhere
- No downtime
- Site-wide
Deployed Solution:
Visual Studio & PackageClient side & server sideDowntimeSite-wide
Working with Client Side Code
SharePoint Encourages:
- Editing directly on the file system
- Injecting Javascript directly via a script editor
- Working within the SharePoint environment
Modern JavaScript development mostly happens in a local development environment, with tooling to help the build and development process
Why React?
- Fast and Fun
- Instills a helpful mindset for data flow
- Marketable skillset
- I want to fit in with the cool kids
Before
Sharepoint 2016 Feature Pack 2
I want to use React
- Add Libraries to SharePoint
- Write ES5 React Code directly in SharePoint
- Result:
data:image/s3,"s3://crabby-images/b9f03/b9f03636464d3df9eeff3ceadb2655ef28a740c2" alt=""
- Build your own ES6/React Sharepoint deployment pipeline.
- Result:
data:image/s3,"s3://crabby-images/3ec5b/3ec5be49ddeafeded43d7e130af634f5d3ffd85c" alt=""
Setting up a React/ES6 Pipeline
class Counter extends React.Component {
state = {
count: 0
}
handleIncrement = () => {
this.setState((prevState, props) => {
return ({ count: prevState.count + 1 });
})
}
handleDecrement = () => {
this.setState((prevState, props) => {
return ({ count: prevState.count - 1 });
})
}
render() {
return (
<div className='counter-app'>
<div className='counter'> {this.state.count} </div>
<button className='counter button' type='button' onClick={this.handleIncrement}>
increment
</button>
<button className='counter button' type='button' onClick={this.handleDecrement}>
decrement
</button>
</div>
);
}
}
ReactDOM.render(
<Counter />,
document.getElementById("root")
);
Setting up a React/ES6 Pipeline
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) {if (window.CP.shouldStopExecution(1)){break;} 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); }
window.CP.exitedLoop(1);
} 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 Counter = function (_React$Component) {
_inherits(Counter, _React$Component);
function Counter() {
var _ref;
var _temp, _this, _ret;
_classCallCheck(this, Counter);
for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) {if (window.CP.shouldStopExecution(2)){break;}
args[_key] = arguments[_key];
}
window.CP.exitedLoop(2);
return _ret = (_temp = (_this = _possibleConstructorReturn(this, (_ref = Counter.__proto__ || Object.getPrototypeOf(Counter)).call.apply(_ref, [this].concat(args))), _this), _this.state = {
count: 0
}, _this.handleIncrement = function () {
_this.setState(function (prevState, props) {
return { count: prevState.count + 1 };
});
}, _this.handleDecrement = function () {
_this.setState(function (prevState, props) {
return { count: prevState.count - 1 };
});
}, _temp), _possibleConstructorReturn(_this, _ret);
}
_createClass(Counter, [{
key: 'render',
value: function render() {
return React.createElement(
'div',
{ className: 'counter-app' },
React.createElement(
'div',
{ className: 'counter' },
' ',
this.state.count,
' '
),
React.createElement(
'button',
{ className: 'counter button', type: 'button', onClick: this.handleIncrement },
'increment'
),
React.createElement(
'button',
{ className: 'counter button', type: 'button', onClick: this.handleDecrement },
'decrement'
)
);
}
}]);
return Counter;
}(React.Component);
ReactDOM.render(React.createElement(Counter, null), document.getElementById("root"));
Source Code!
Transpile!
Package!
Deploy!
Writing your own Pipeline
- Familiarizing & learning the transpilation toolchain
- Setting up your own development server
- Configuring your own local development environment (linting/test-runner/etc.)
Writing your own Pipeline
Worth it if:
- You like customizing your own tooling or want to use features not directly available in a pre-built pipeline (SASS/LESS, etc)
- Your widgets need to be in SharePoint to be worked on (require access to multiple data sources in SharePoint or interact heavily with the SharePoint UI)
- You want the bare minimum: transpilation and deployment.
Using CRA as a Pipeline
data:image/s3,"s3://crabby-images/a7fa7/a7fa733897bde689051ddd61ea151b31c3f847f6" alt=""
Using CRA as a Pipeline
Issues: Very Abstracted System
- No access to the build tooling without 'ejecting'
- Not immediately obvious how to extend overall pipeline for own or custom use.
- Requires a bit of custom finagling to play nice as part of a CMS
Using CRA as a Pipeline
Providing Externals:
data:image/s3,"s3://crabby-images/f442f/f442ff150ca73e12b07cc34c604bf9544e4d23a4" alt=""
Using CRA as a Pipeline
Opinionated Build Additions:
data:image/s3,"s3://crabby-images/5b390/5b39004f2483e06581fa4029cf4e1b46b5803204" alt=""
Using CRA as a Pipeline
Opinionated Build Additions:
data:image/s3,"s3://crabby-images/a8e04/a8e04b8c8b1f3fa551866500da4bf3ccd89b161e" alt=""
Using CRA as a Pipeline
Opinionated Build Additions:
- Do you need code-splitting, or service workers?
- Do you need automatic source maps, asset manifest generation, etc?
- Might be overkill, depending on your needs. SharePoint doesn't necessarily aspire to be a PWA.
Using CRA as a Pipeline
Optimized for a single page application:
data:image/s3,"s3://crabby-images/0af3a/0af3a1b1e63da6ceff6589907f0f26fd6fbba11d" alt=""
Automating Deployment
data:image/s3,"s3://crabby-images/316a9/316a951b0a9f41c79599037765d5b0d91485251b" alt=""
Bless this guy
data:image/s3,"s3://crabby-images/1089b/1089be82129f4e20c6452bcdb79720980cb66edb" alt=""
data:image/s3,"s3://crabby-images/d6f0d/d6f0d4a7aff787b3bf89f5474ed62eae19caa6d0" alt=""
data:image/s3,"s3://crabby-images/0d4b7/0d4b7f170db605f495f81006e65dd272320cacbb" alt=""
Authenticating to SharePoint
data:image/s3,"s3://crabby-images/c4f2a/c4f2abe8fdf6c922c5a17832b4de200e72f7d123" alt=""
Do you do weird stuff like Forms Authentication? This guy has got your back!
Saving files to SharePoint
data:image/s3,"s3://crabby-images/0c769/0c769c5e4b9d4d932a77b87ac3f5c96104e5a3f2" alt=""
The Holy Grail - Automation!
data:image/s3,"s3://crabby-images/6af84/6af84b91c3abbe45575703952ef98993c2c40933" alt=""
Using Gulp
data:image/s3,"s3://crabby-images/2aadf/2aadfd816a4b82705f6272db5123407d54357811" alt=""
data:image/s3,"s3://crabby-images/a180d/a180d8b7c3d7f87b69dd645d35e169357dc9bfc9" alt=""
Using Gulp
data:image/s3,"s3://crabby-images/2aadf/2aadfd816a4b82705f6272db5123407d54357811" alt=""
Insert automated deployment into your workflow anywhere:
Using Gulp
Insert automated deployment into your workflow anywhere:
data:image/s3,"s3://crabby-images/5f12d/5f12d4265b1ea7124b0c383248f801b22ad980d0" alt=""
Demo
Caveats and Workarounds
Web Part Properties
data:image/s3,"s3://crabby-images/875cb/875cbb3a7b8fcfb7e771f9592f266869e6604629" alt=""
- Not customizable
- Requires a workaround to provide customization to a widget or web part
- Global Variables? Configuration JSON?
- It gets weird.
Sharing State Between Components
- Singleton, Global Redux Store
- Providing alternative Javascript state objects from the Master Page or Page Layout
- Exposing react lifecycle methods as a global function.
Still getting weird.
Injecting React into existing Markup
data:image/s3,"s3://crabby-images/fa40e/fa40e55ad18e508c6536672da6b9d07e9603c024" alt=""
The 'Useful' Zone
Injecting React into existing Markup
- jQuery/Javascript
- Adding the root divs for react components to the server-side markup
- Yup, still weird.
SharePoint Feature Pack 2
SPFx
SharePoint Framework ...x
- Direct Javascript embedding
- Framework agnostic
- Modern Toolchain
- End user configurable
Development Environment
- Node based
- Yeoman for scaffolding
- Provides access to various javascript frameworks currently supported: React/Angular/Knockout(?)/etc.
- Uses gulp as a task runner and for workflows
Opinionated
- TypeScript
- Config is abstracted away
- Tightly coupled to SharePoint processes
data:image/s3,"s3://crabby-images/c241e/c241e95ddca9c5543e77a6bbfac0df13006e68bb" alt=""
data:image/s3,"s3://crabby-images/0e254/0e254e95789cd637cc7592dda91eb5cf26235b21" alt=""
data:image/s3,"s3://crabby-images/e874d/e874d9a91fe3fbf2451fd035ec1f0ea5c96bc2b2" alt=""
Still nice!
- Hot Reloading Dev Server!
- Generator works nicely!
- Debugging very pleasant!
- Packages with no problems!
Building a Web Part
data:image/s3,"s3://crabby-images/648be/648be3b6d7f5c0ceb6184ed911fd1cd65625790d" alt=""
Tight Integration
- Direct access to the Properties Pane!
- Easy interaction with SharePoint REST layer
- Easy interaction with traditional C# web parts
data:image/s3,"s3://crabby-images/87c4c/87c4c396e6ffa394c4a3a2f22ef88d174110a88d" alt=""
Deploying to SharePoint
SharePoint Online
- Works out of the box, more or less.
SharePoint On Premises
- Feature Pack 2
- Configuring an environment for Apps
- Wildcard SSL certs
- Subscription Settings
- Application Domains?
- Multi Tenant Settings?
- App Catalogs?
Seriously the config takes forever
data:image/s3,"s3://crabby-images/d57e8/d57e8a722097a97a7966e5747f64620d503b5a4d" alt=""
data:image/s3,"s3://crabby-images/026c9/026c9f90c495ca22522c7e23d6ce2697c793eab8" alt=""
data:image/s3,"s3://crabby-images/b6592/b65923ae1f840f7d36f66b24c024e244c87f44c1" alt=""
Eventually you have an App!
data:image/s3,"s3://crabby-images/13cb6/13cb63d5bdc9be6c8adc18ded2118fa77e89c0ec" alt=""
(Brief) Demo
Conclusion
SPFx
Good When
- You have client side applications that will see a lot of repeated re-use. (List Displays, Carousels, Interactive List based widgets, doodads, etc.)
- You absolutely need to provide hooks into the web part properties pane.
- You want close integration with other modern SharePoint features.
- You can use it at all - (Requires SP2016 w/ FP2 and a lot of config)
DIY React
Good When
- You absolutely have to set up your own tooling, development environment and practices.
- You want to escape the boundaries of the standard SharePoint grid and provide your own React Based UI Theming and data layer.
- Industrial strength client side applications and widgets that live in SharePoint as a hosting solution but don't necessarily interact with it.
- You don't need to rely as much on a traditional C#/SharePoint base.
Thanks!
Basic Overview of My Stuff
Integrating ReactJS into SharePoint
By Linus Falck-Ytter
Integrating ReactJS into SharePoint
- 1,071