Updated SPFx version and more (#280)
* Updated SPFx from 1.0.0 to 1.1.0 * Added closing div. Updated logo. * Include override.css using require instead of sass @import which no longer works in the build chain. * Updated README with complete building steps * Updated version to 1.0.0.1 * Updated to pull version from package.json * Include bundled Office UI Frabework CSS to ensure proper grid. * Added version info * Added safeWithCustomScriptDisabled property * Changed to static linking * Added explicit reference to office-ui-fabric v2 * Removed test CSS * Remove commented out code * Remove empty lines * Removed non-existant styles * Externalize Office UI Fabric components and CSS to CDN - keeping CSS in the project for reference.
This commit is contained in:
parent
ef0be61fd5
commit
a254b685c1
|
@ -66,6 +66,7 @@ react-script-editor | Mikael Svenson ([@mikaelsvenson](http://www.twitter.com/mi
|
||||||
Version|Date|Comments
|
Version|Date|Comments
|
||||||
-------|----|--------
|
-------|----|--------
|
||||||
1.0|March 10th, 2017|Initial release
|
1.0|March 10th, 2017|Initial release
|
||||||
|
1.0.1|August 8th, 2017|Updated SPFx version and CSS loading
|
||||||
|
|
||||||
## Disclaimer
|
## Disclaimer
|
||||||
**THIS CODE IS PROVIDED *AS IS* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING ANY IMPLIED WARRANTIES OF FITNESS FOR A PARTICULAR PURPOSE, MERCHANTABILITY, OR NON-INFRINGEMENT.**
|
**THIS CODE IS PROVIDED *AS IS* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING ANY IMPLIED WARRANTIES OF FITNESS FOR A PARTICULAR PURPOSE, MERCHANTABILITY, OR NON-INFRINGEMENT.**
|
||||||
|
@ -73,12 +74,23 @@ Version|Date|Comments
|
||||||
---
|
---
|
||||||
|
|
||||||
## Minimal Path to Awesome
|
## Minimal Path to Awesome
|
||||||
|
### Local testing
|
||||||
|
|
||||||
- Clone this repository
|
- Clone this repository
|
||||||
- In the command line run:
|
- In the command line run:
|
||||||
- `npm install`
|
- `npm install`
|
||||||
- `gulp serve`
|
- `gulp serve`
|
||||||
|
|
||||||
|
### Deploy
|
||||||
|
* Set CDN path in config\write-manifest.json to where you want to host the web part
|
||||||
|
* E.g.: https://<tenant>.sharepoint.com/sites/CDN/SiteAssets/SPFx/<partname>
|
||||||
|
* gulp --ship
|
||||||
|
* gulp package-solution --ship
|
||||||
|
* Copy contents of temp\deploy to the CDN folder
|
||||||
|
* Upload .sppkg file from sharepoint\solution to your tenant App Catalog
|
||||||
|
* E.g.: https://<tenant>.sharepoint.com/sites/AppCatalog/AppCatalog
|
||||||
|
* Add the web part to a site collection, and test it on a page
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
This web part illustrates the following concepts on top of the SharePoint Framework:
|
This web part illustrates the following concepts on top of the SharePoint Framework:
|
||||||
|
|
||||||
|
|
|
@ -6,5 +6,7 @@
|
||||||
"outputPath": "./dist/script-editor.bundle.js"
|
"outputPath": "./dist/script-editor.bundle.js"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"externals": {}
|
"externals": {
|
||||||
}
|
"office-ui-fabric-react": "https://publiccdn.sharepointonline.com/techmikael.sharepoint.com/11510075fe4212d19d3e6d07c91981263dd697bf111cb1e5f0efb15de0ec08b382cde399/2.34.2/office-ui-fabric-react.bundle.min.js"
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,7 +2,7 @@
|
||||||
"solution": {
|
"solution": {
|
||||||
"name": "Modern Script Editor web part by Puzzlepart",
|
"name": "Modern Script Editor web part by Puzzlepart",
|
||||||
"id": "1425175f-3ed8-44d2-8fc4-dd1497191294",
|
"id": "1425175f-3ed8-44d2-8fc4-dd1497191294",
|
||||||
"version": "1.0.0.0"
|
"version": "1.0.0.1"
|
||||||
},
|
},
|
||||||
"paths": {
|
"paths": {
|
||||||
"zippedPackage": "solution/pzl-script-editor.sppkg"
|
"zippedPackage": "solution/pzl-script-editor.sppkg"
|
||||||
|
|
|
@ -7,21 +7,22 @@
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@microsoft/sp-client-base": "~1.0.0",
|
"@microsoft/sp-client-base": "~1.0.0",
|
||||||
"@microsoft/sp-core-library": "~1.0.0",
|
"@microsoft/sp-core-library": "~1.1.0",
|
||||||
"@microsoft/sp-webpart-base": "~1.0.0",
|
"@microsoft/sp-webpart-base": "~1.1.0",
|
||||||
"@types/react": "0.14.46",
|
"@types/react": "0.14.46",
|
||||||
"@types/react-addons-shallow-compare": "0.14.17",
|
"@types/react-addons-shallow-compare": "0.14.17",
|
||||||
"@types/react-addons-test-utils": "0.14.15",
|
"@types/react-addons-test-utils": "0.14.15",
|
||||||
"@types/react-addons-update": "0.14.14",
|
"@types/react-addons-update": "0.14.14",
|
||||||
"@types/react-dom": "0.14.18",
|
"@types/react-dom": "0.14.18",
|
||||||
"@types/webpack-env": ">=1.12.1 <1.14.0",
|
"@types/webpack-env": ">=1.12.1 <1.14.0",
|
||||||
|
"office-ui-fabric-react": "2.34.2",
|
||||||
"react": "15.4.2",
|
"react": "15.4.2",
|
||||||
"react-dom": "15.4.2"
|
"react-dom": "15.4.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@microsoft/sp-build-web": "~1.0.0",
|
"@microsoft/sp-build-web": "~1.1.0",
|
||||||
"@microsoft/sp-module-interfaces": "~1.0.0",
|
"@microsoft/sp-module-interfaces": "~1.1.0",
|
||||||
"@microsoft/sp-webpart-workbench": "~1.0.0",
|
"@microsoft/sp-webpart-workbench": "~1.1.0",
|
||||||
"gulp": "~3.9.1",
|
"gulp": "~3.9.1",
|
||||||
"@types/chai": ">=3.4.34 <3.6.0",
|
"@types/chai": ">=3.4.34 <3.6.0",
|
||||||
"@types/mocha": ">=2.2.33 <2.6.0"
|
"@types/mocha": ">=2.2.33 <2.6.0"
|
||||||
|
@ -31,4 +32,4 @@
|
||||||
"clean": "gulp clean",
|
"clean": "gulp clean",
|
||||||
"test": "gulp test"
|
"test": "gulp test"
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,12 +1,17 @@
|
||||||
{
|
{
|
||||||
"$schema": "../../../node_modules/@microsoft/sp-module-interfaces/lib/manifestSchemas/jsonSchemas/clientSideComponentManifestSchema.json",
|
"$schema": "../../../node_modules/@microsoft/sp-module-interfaces/lib/manifestSchemas/jsonSchemas/clientSideComponentManifestSchema.json",
|
||||||
|
|
||||||
"id": "3a328f0a-99c4-4b28-95ab-fe0847f657a3",
|
"id": "3a328f0a-99c4-4b28-95ab-fe0847f657a3",
|
||||||
"alias": "ScriptEditorWebPart",
|
"alias": "ScriptEditorWebPart",
|
||||||
"componentType": "WebPart",
|
"componentType": "WebPart",
|
||||||
"version": "0.0.1",
|
"version": "*", // The "*" signifies that the version should be taken from the package.json
|
||||||
"manifestVersion": 2,
|
"manifestVersion": 2,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This property should only be set to true if it is certain that the webpart does not
|
||||||
|
* allow arbitrary scripts to be called
|
||||||
|
*/
|
||||||
|
"safeWithCustomScriptDisabled": false,
|
||||||
|
|
||||||
"preconfiguredEntries": [{
|
"preconfiguredEntries": [{
|
||||||
"groupId": "3a328f0a-99c4-4b28-95ab-fe0847f657a3",
|
"groupId": "3a328f0a-99c4-4b28-95ab-fe0847f657a3",
|
||||||
"group": { "default": "Puzzlepart" },
|
"group": { "default": "Puzzlepart" },
|
||||||
|
|
|
@ -12,14 +12,12 @@ import { IScriptEditorProps } from './components/IScriptEditorProps';
|
||||||
import { IScriptEditorWebPartProps } from './IScriptEditorWebPartProps';
|
import { IScriptEditorWebPartProps } from './IScriptEditorWebPartProps';
|
||||||
|
|
||||||
export default class ScriptEditorWebPart extends BaseClientSideWebPart<IScriptEditorWebPartProps> {
|
export default class ScriptEditorWebPart extends BaseClientSideWebPart<IScriptEditorWebPartProps> {
|
||||||
|
|
||||||
public save: (script: string) => void = (script: string) => {
|
public save: (script: string) => void = (script: string) => {
|
||||||
this.properties.script = script;
|
this.properties.script = script;
|
||||||
this.render();
|
this.render();
|
||||||
}
|
}
|
||||||
|
|
||||||
public render(): void {
|
public render(): void {
|
||||||
|
|
||||||
const element: React.ReactElement<IScriptEditorProps> = React.createElement(
|
const element: React.ReactElement<IScriptEditorProps> = React.createElement(
|
||||||
ScriptEditor,
|
ScriptEditor,
|
||||||
{
|
{
|
||||||
|
@ -41,7 +39,11 @@ export default class ScriptEditorWebPart extends BaseClientSideWebPart<IScriptEd
|
||||||
}
|
}
|
||||||
|
|
||||||
protected renderLogo(domElement: HTMLElement) {
|
protected renderLogo(domElement: HTMLElement) {
|
||||||
domElement.innerHTML = '<div style="margin-top: 30px"><img src="//www.puzzlepart.com/wp-content/uploads/2017/02/Puzzlepart-Logo-Digital-webheader200.png" onerror="this.style.display = \'none\'";" style="width:50%; float:right">';
|
domElement.innerHTML = `
|
||||||
|
<div style="margin-top: 30px">
|
||||||
|
<div style="float:right">Author: <a href="mailto:mikael.svenson@puzzlepart.com" tabindex="-1">Mikael Svenson</a></div>
|
||||||
|
<div style="float:right"><img src="//www.puzzlepart.com/wp-content/uploads/2017/08/Pzl-LogoType-200.png" onerror="this.style.display = \'none\'";"></div>
|
||||||
|
</div>`;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected getPropertyPaneConfiguration(): IPropertyPaneConfiguration {
|
protected getPropertyPaneConfiguration(): IPropertyPaneConfiguration {
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
@import url('overrides.css');
|
//@import url(https://publiccdn.sharepointonline.com/techmikael.sharepoint.com/11510075fe4212d19d3e6d07c91981263dd697bf111cb1e5f0efb15de0ec08b382cde399/5.0.1/office-ui-fabric.min.css);
|
||||||
|
|
||||||
.scriptEditor {
|
.scriptEditor {
|
||||||
.container {
|
.container {
|
||||||
|
|
|
@ -1,16 +1,35 @@
|
||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import styles from './ScriptEditor.module.scss';
|
import styles from './ScriptEditor.module.scss';
|
||||||
import { IScriptEditorProps } from './IScriptEditorProps';
|
import { IScriptEditorProps } from './IScriptEditorProps';
|
||||||
import { Dialog, DialogType, DialogFooter, Button, ButtonType, TextField } from 'office-ui-fabric-react';
|
import { Dialog, DialogType, DialogFooter } from 'office-ui-fabric-react';
|
||||||
|
import { DefaultButton, PrimaryButton } from 'office-ui-fabric-react';
|
||||||
|
import { TextField } from 'office-ui-fabric-react';
|
||||||
|
import { loadStyles } from '@microsoft/load-themed-styles';
|
||||||
|
require('./overrides.css');
|
||||||
|
|
||||||
export default class ScriptEditor extends React.Component<IScriptEditorProps, any> {
|
export default class ScriptEditor extends React.Component<IScriptEditorProps, any> {
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
|
this.loadCss();
|
||||||
this.state = {
|
this.state = {
|
||||||
showDialog: false
|
showDialog: false
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async loadCss() {
|
||||||
|
if (window["UIFabricLoaded"]) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const response = await fetch("https://publiccdn.sharepointonline.com/techmikael.sharepoint.com/11510075fe4212d19d3e6d07c91981263dd697bf111cb1e5f0efb15de0ec08b382cde399/5.0.1/office-ui-fabric.min.css");
|
||||||
|
if (response.ok) {
|
||||||
|
response.text().then((data: any) => {
|
||||||
|
loadStyles(data);
|
||||||
|
window["UIFabricLoaded"] = true;
|
||||||
|
this.forceUpdate();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public componentDidMount(): void {
|
public componentDidMount(): void {
|
||||||
this.setState({ script: this.props.script, loaded: this.props.script });
|
this.setState({ script: this.props.script, loaded: this.props.script });
|
||||||
}
|
}
|
||||||
|
@ -33,17 +52,20 @@ export default class ScriptEditor extends React.Component<IScriptEditorProps, an
|
||||||
}
|
}
|
||||||
|
|
||||||
public render(): React.ReactElement<IScriptEditorProps> {
|
public render(): React.ReactElement<IScriptEditorProps> {
|
||||||
|
if (!window["UIFabricLoaded"]) {
|
||||||
|
return <span />;
|
||||||
|
}
|
||||||
const viewMode = <span dangerouslySetInnerHTML={{ __html: this.state.script }}></span>;
|
const viewMode = <span dangerouslySetInnerHTML={{ __html: this.state.script }}></span>;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div >
|
||||||
<div className={styles.scriptEditor}>
|
<div className={styles.scriptEditor}>
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
<div className={`ms-Grid-row ms-bgColor-themeDark ms-fontColor-white ${styles.row}`}>
|
<div className={`pzl-Grid-row pzl-bgColor-themeDark pzl-fontColor-white ${styles.row}`}>
|
||||||
<div className="ms-Grid-col ms-u-lg10 ms-u-xl8 ms-u-xlPush2 ms-u-lgPush1">
|
<div className="pzl-Grid-col pzl-u-lg10 pzl-u-xl8 pzl-u-xlPush2 pzl-u-lgPush1">
|
||||||
<span className="ms-font-xl ms-fontColor-white">The Modern Script Editor web part!</span>
|
<span className="pzl-font-xl pzl-fontColor-white">The Modern Script Editor web part!</span>
|
||||||
<p className="ms-font-l ms-fontColor-white"></p>
|
<p className="pzl-font-l pzl-fontColor-white"></p>
|
||||||
<Button description='Opens the Sample Dialog' onClick={this._showDialog.bind(this)}>Edit snippet</Button>
|
<DefaultButton description='Opens the Sample Dialog' onClick={this._showDialog.bind(this)}>Edit snippet</DefaultButton>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -59,11 +81,11 @@ export default class ScriptEditor extends React.Component<IScriptEditorProps, an
|
||||||
>
|
>
|
||||||
<TextField multiline rows={15} onChanged={this._onScriptEditorTextChanged.bind(this)} value={this.state.script} />
|
<TextField multiline rows={15} onChanged={this._onScriptEditorTextChanged.bind(this)} value={this.state.script} />
|
||||||
<DialogFooter>
|
<DialogFooter>
|
||||||
<Button buttonType={ButtonType.primary} onClick={this._closeDialog.bind(this)}>Save</Button>
|
<PrimaryButton onClick={this._closeDialog.bind(this)}>Save</PrimaryButton>
|
||||||
<Button onClick={this._cancelDialog.bind(this)}>Cancel</Button>
|
<DefaultButton onClick={this._cancelDialog.bind(this)}>Cancel</DefaultButton>
|
||||||
</DialogFooter>
|
</DialogFooter>
|
||||||
{viewMode}
|
{viewMode}
|
||||||
</Dialog>
|
</Dialog>
|
||||||
</div>);
|
</div >);
|
||||||
}
|
}
|
||||||
}
|
}
|
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue