diff --git a/README.md b/README.md
index 37433ba..55c70bc 100644
--- a/README.md
+++ b/README.md
@@ -104,3 +104,27 @@ configuration like this:
]
},
```
+
+### Resetting the Telemetry System
+
+This application includes a telemetry system which is only active if the user opts in. When testing this system, it's
+sometimes helpful to reset it by deleting the `telemetry.json` file.
+
+The location of this file depends on your operating system and whether or not you're running a packaged build. Running
+from `npm start` or equivalent is a non-packaged build.
+
+In addition, macOS may store the file in one of two places depending on the OS version and a few other variables. If
+in doubt, I recommend removing both.
+
+- Windows, packaged build: `%APPDATA%\Scratch\telemetry.json`
+- Windows, non-packaged: `%APPDATA%\Electron\telemetry.json`
+- macOS, packaged build: `~/Library/Application Support/Scratch/telemetry.json` or
+ `~/Library/Containers/edu.mit.scratch.scratch-desktop/Data/Library/Application Support/Scratch/telemetry.json`
+- macOS, non-packaged build: `~/Library/Application Support/Electron/telemetry.json` or
+ `~/Library/Containers/edu.mit.scratch.scratch-desktop/Data/Library/Application Support/Electron/telemetry.json`
+
+Deleting this file will:
+
+- Remove any pending telemetry packets
+- Reset the opt in/out state: the app should display the opt in/out modal on next launch
+- Remove the random client UUID: the app will generate a new one on next launch
diff --git a/src/renderer/ScratchDesktopAppStateHOC.jsx b/src/renderer/ScratchDesktopAppStateHOC.jsx
new file mode 100644
index 0000000..13a7905
--- /dev/null
+++ b/src/renderer/ScratchDesktopAppStateHOC.jsx
@@ -0,0 +1,53 @@
+import {ipcRenderer} from 'electron';
+import bindAll from 'lodash.bindall';
+import React from 'react';
+
+/**
+ * Higher-order component to add desktop logic to AppStateHOC.
+ * @param {Component} WrappedComponent - an AppStateHOC-like component to wrap.
+ * @returns {Component} - a component similar to AppStateHOC with desktop-specific logic added.
+ */
+const ScratchDesktopAppStateHOC = function (WrappedComponent) {
+ class ScratchDesktopAppStateComponent extends React.Component {
+ constructor (props) {
+ super(props);
+ bindAll(this, [
+ 'handleTelemetryModalOptIn',
+ 'handleTelemetryModalOptOut'
+ ]);
+ this.state = {
+ // use `sendSync` because this should be set before first render
+ telemetryDidOptIn: ipcRenderer.sendSync('getTelemetryDidOptIn')
+ };
+ }
+ handleTelemetryModalOptIn () {
+ ipcRenderer.send('setTelemetryDidOptIn', true);
+ ipcRenderer.invoke('getTelemetryDidOptIn').then(telemetryDidOptIn => {
+ this.setState({telemetryDidOptIn});
+ });
+ }
+ handleTelemetryModalOptOut () {
+ ipcRenderer.send('setTelemetryDidOptIn', false);
+ ipcRenderer.invoke('getTelemetryDidOptIn').then(telemetryDidOptIn => {
+ this.setState({telemetryDidOptIn});
+ });
+ }
+ render () {
+ const shouldShowTelemetryModal = (typeof ipcRenderer.sendSync('getTelemetryDidOptIn') !== 'boolean');
+
+ return ();
+ }
+ }
+
+ return ScratchDesktopAppStateComponent;
+};
+
+export default ScratchDesktopAppStateHOC;
diff --git a/src/renderer/ScratchDesktopGUIHOC.jsx b/src/renderer/ScratchDesktopGUIHOC.jsx
new file mode 100644
index 0000000..97e5c9e
--- /dev/null
+++ b/src/renderer/ScratchDesktopGUIHOC.jsx
@@ -0,0 +1,175 @@
+import {ipcRenderer, remote} from 'electron';
+import bindAll from 'lodash.bindall';
+import omit from 'lodash.omit';
+import PropTypes from 'prop-types';
+import React from 'react';
+import {connect} from 'react-redux';
+import GUIComponent from 'scratch-gui/src/components/gui/gui.jsx';
+
+import {
+ LoadingStates,
+ onFetchedProjectData,
+ onLoadedProject,
+ defaultProjectId,
+ requestNewProject,
+ requestProjectUpload,
+ setProjectId
+} from 'scratch-gui/src/reducers/project-state';
+import {
+ openLoadingProject,
+ closeLoadingProject,
+ openTelemetryModal
+} from 'scratch-gui/src/reducers/modals';
+
+import ElectronStorageHelper from '../common/ElectronStorageHelper';
+
+import showPrivacyPolicy from './showPrivacyPolicy';
+
+/**
+ * Higher-order component to add desktop logic to the GUI.
+ * @param {Component} WrappedComponent - a GUI-like component to wrap.
+ * @returns {Component} - a component similar to GUI with desktop-specific logic added.
+ */
+const ScratchDesktopGUIHOC = function (WrappedComponent) {
+ class ScratchDesktopGUIComponent extends React.Component {
+ constructor (props) {
+ super(props);
+ bindAll(this, [
+ 'handleProjectTelemetryEvent',
+ 'handleSetTitleFromSave',
+ 'handleStorageInit',
+ 'handleUpdateProjectTitle'
+ ]);
+ this.props.onLoadingStarted();
+ ipcRenderer.invoke('get-initial-project-data').then(initialProjectData => {
+ const hasInitialProject = initialProjectData && (initialProjectData.length > 0);
+ this.props.onHasInitialProject(hasInitialProject, this.props.loadingState);
+ if (!hasInitialProject) {
+ this.props.onLoadingCompleted();
+ return;
+ }
+ this.props.vm.loadProject(initialProjectData).then(
+ () => {
+ this.props.onLoadingCompleted();
+ this.props.onLoadedProject(this.props.loadingState, true);
+ },
+ e => {
+ this.props.onLoadingCompleted();
+ this.props.onLoadedProject(this.props.loadingState, false);
+ remote.dialog.showMessageBox(remote.getCurrentWindow(), {
+ type: 'error',
+ title: 'Failed to load project',
+ message: 'Invalid or corrupt project file.',
+ detail: e.message
+ });
+
+ // this effectively sets the default project ID
+ // TODO: maybe setting the default project ID should be implicit in `requestNewProject`
+ this.props.onHasInitialProject(false, this.props.loadingState);
+
+ // restart as if we didn't have an initial project to load
+ this.props.onRequestNewProject();
+ }
+ );
+ });
+ }
+ componentDidMount () {
+ ipcRenderer.on('setTitleFromSave', this.handleSetTitleFromSave);
+ }
+ componentWillUnmount () {
+ ipcRenderer.removeListener('setTitleFromSave', this.handleSetTitleFromSave);
+ }
+ handleClickAbout () {
+ ipcRenderer.send('open-about-window');
+ }
+ handleProjectTelemetryEvent (event, metadata) {
+ ipcRenderer.send(event, metadata);
+ }
+ handleSetTitleFromSave (event, args) {
+ this.handleUpdateProjectTitle(args.title);
+ }
+ handleStorageInit (storageInstance) {
+ storageInstance.addHelper(new ElectronStorageHelper(storageInstance));
+ }
+ handleUpdateProjectTitle (newTitle) {
+ this.setState({projectTitle: newTitle});
+ }
+ render () {
+ const childProps = omit(this.props, Object.keys(ScratchDesktopGUIComponent.propTypes));
+
+ return ( this.handleClickAbout()
+ },
+ {
+ title: 'Privacy Policy',
+ onClick: () => showPrivacyPolicy()
+ },
+ {
+ title: 'Data Settings',
+ onClick: () => this.props.onTelemetrySettingsClicked()
+ }
+ ]}
+ onProjectTelemetryEvent={this.handleProjectTelemetryEvent}
+ onShowPrivacyPolicy={showPrivacyPolicy}
+ onStorageInit={this.handleStorageInit}
+ onUpdateProjectTitle={this.handleUpdateProjectTitle}
+
+ // allow passed-in props to override any of the above
+ {...childProps}
+ />);
+ }
+ }
+
+ ScratchDesktopGUIComponent.propTypes = {
+ loadingState: PropTypes.oneOf(LoadingStates),
+ onFetchedInitialProjectData: PropTypes.func,
+ onHasInitialProject: PropTypes.func,
+ onLoadedProject: PropTypes.func,
+ onLoadingCompleted: PropTypes.func,
+ onLoadingStarted: PropTypes.func,
+ onRequestNewProject: PropTypes.func,
+ onTelemetrySettingsClicked: PropTypes.func,
+ // using PropTypes.instanceOf(VM) here will cause prop type warnings due to VM mismatch
+ vm: GUIComponent.WrappedComponent.propTypes.vm
+ };
+ const mapStateToProps = state => {
+ const loadingState = state.scratchGui.projectState.loadingState;
+ return {
+ loadingState: loadingState,
+ vm: state.scratchGui.vm
+ };
+ };
+ const mapDispatchToProps = dispatch => ({
+ onLoadingStarted: () => dispatch(openLoadingProject()),
+ onLoadingCompleted: () => dispatch(closeLoadingProject()),
+ onHasInitialProject: (hasInitialProject, loadingState) => {
+ if (hasInitialProject) {
+ // emulate sb-file-uploader
+ return dispatch(requestProjectUpload(loadingState));
+ }
+
+ // `createProject()` might seem more appropriate but it's not a valid state transition here
+ // setting the default project ID is a valid transition from NOT_LOADED and acts like "create new"
+ return dispatch(setProjectId(defaultProjectId));
+ },
+ onFetchedInitialProjectData: (projectData, loadingState) =>
+ dispatch(onFetchedProjectData(projectData, loadingState)),
+ onLoadedProject: (loadingState, loadSuccess) => {
+ const canSaveToServer = false;
+ return dispatch(onLoadedProject(loadingState, canSaveToServer, loadSuccess));
+ },
+ onRequestNewProject: () => dispatch(requestNewProject(false)),
+ onTelemetrySettingsClicked: () => dispatch(openTelemetryModal())
+ });
+
+ return connect(mapStateToProps, mapDispatchToProps)(ScratchDesktopGUIComponent);
+};
+
+export default ScratchDesktopGUIHOC;
diff --git a/src/renderer/app.jsx b/src/renderer/app.jsx
index 56438c4..d437381 100644
--- a/src/renderer/app.jsx
+++ b/src/renderer/app.jsx
@@ -1,32 +1,11 @@
-import {ipcRenderer, remote} from 'electron';
-import bindAll from 'lodash.bindall';
-import omit from 'lodash.omit';
-import PropTypes from 'prop-types';
import React from 'react';
-import {connect} from 'react-redux';
import {compose} from 'redux';
import GUI from 'scratch-gui/src/index';
-import VM from 'scratch-vm';
import AppStateHOC from 'scratch-gui/src/lib/app-state-hoc.jsx';
-import {
- LoadingStates,
- onFetchedProjectData,
- onLoadedProject,
- defaultProjectId,
- requestNewProject,
- requestProjectUpload,
- setProjectId
-} from 'scratch-gui/src/reducers/project-state';
-import {
- openLoadingProject,
- closeLoadingProject,
- openTelemetryModal
-} from 'scratch-gui/src/reducers/modals';
-import ElectronStorageHelper from '../common/ElectronStorageHelper';
-
-import showPrivacyPolicy from './showPrivacyPolicy';
+import ScratchDesktopAppStateHOC from './ScratchDesktopAppStateHOC.jsx';
+import ScratchDesktopGUIHOC from './ScratchDesktopGUIHOC.jsx';
import styles from './app.css';
const appTarget = document.getElementById('app');
@@ -34,177 +13,14 @@ appTarget.className = styles.app || 'app';
GUI.setAppElement(appTarget);
-const ScratchDesktopHOC = function (WrappedComponent) {
- class ScratchDesktopComponent extends React.Component {
- constructor (props) {
- super(props);
- bindAll(this, [
- 'handleProjectTelemetryEvent',
- 'handleSetTitleFromSave',
- 'handleStorageInit',
- 'handleTelemetryModalOptIn',
- 'handleTelemetryModalOptOut',
- 'handleUpdateProjectTitle'
- ]);
- this.state = {
- // use `sendSync` because this should be set before first render
- telemetryDidOptIn: ipcRenderer.sendSync('getTelemetryDidOptIn')
- };
- this.props.onLoadingStarted();
- ipcRenderer.invoke('get-initial-project-data').then(initialProjectData => {
- const hasInitialProject = initialProjectData && (initialProjectData.length > 0);
- this.props.onHasInitialProject(hasInitialProject, this.props.loadingState);
- if (!hasInitialProject) {
- this.props.onLoadingCompleted();
- return;
- }
- this.props.vm.loadProject(initialProjectData).then(
- () => {
- this.props.onLoadingCompleted();
- this.props.onLoadedProject(this.props.loadingState, true);
- },
- e => {
- this.props.onLoadingCompleted();
- this.props.onLoadedProject(this.props.loadingState, false);
- remote.dialog.showMessageBox(remote.getCurrentWindow(), {
- type: 'error',
- title: 'Failed to load project',
- message: 'Invalid or corrupt project file.',
- detail: e.message
- });
-
- // this effectively sets the default project ID
- // TODO: maybe setting the default project ID should be implicit in `requestNewProject`
- this.props.onHasInitialProject(false, this.props.loadingState);
-
- // restart as if we didn't have an initial project to load
- this.props.onRequestNewProject();
- }
- );
- });
- }
- componentDidMount () {
- ipcRenderer.on('setTitleFromSave', this.handleSetTitleFromSave);
- }
- componentWillUnmount () {
- ipcRenderer.removeListener('setTitleFromSave', this.handleSetTitleFromSave);
- }
- handleClickAbout () {
- ipcRenderer.send('open-about-window');
- }
- handleProjectTelemetryEvent (event, metadata) {
- ipcRenderer.send(event, metadata);
- }
- handleSetTitleFromSave (event, args) {
- this.handleUpdateProjectTitle(args.title);
- }
- handleStorageInit (storageInstance) {
- storageInstance.addHelper(new ElectronStorageHelper(storageInstance));
- }
- handleTelemetryModalOptIn () {
- ipcRenderer.send('setTelemetryDidOptIn', true);
- ipcRenderer.invoke('getTelemetryDidOptIn').then(telemetryDidOptIn => {
- this.setState({telemetryDidOptIn});
- });
- }
- handleTelemetryModalOptOut () {
- ipcRenderer.send('setTelemetryDidOptIn', false);
- ipcRenderer.invoke('getTelemetryDidOptIn').then(telemetryDidOptIn => {
- this.setState({telemetryDidOptIn});
- });
- }
- handleUpdateProjectTitle (newTitle) {
- this.setState({projectTitle: newTitle});
- }
- render () {
- const shouldShowTelemetryModal = (typeof this.state.telemetryDidOptIn !== 'boolean');
-
- const childProps = omit(this.props, Object.keys(ScratchDesktopComponent.propTypes));
-
- return ( this.handleClickAbout()
- },
- {
- title: 'Privacy Policy',
- onClick: () => showPrivacyPolicy()
- },
- {
- title: 'Data Settings',
- onClick: () => this.props.onTelemetrySettingsClicked()
- }
- ]}
- onProjectTelemetryEvent={this.handleProjectTelemetryEvent}
- onShowPrivacyPolicy={showPrivacyPolicy}
- onStorageInit={this.handleStorageInit}
- onTelemetryModalOptIn={this.handleTelemetryModalOptIn}
- onTelemetryModalOptOut={this.handleTelemetryModalOptOut}
- onUpdateProjectTitle={this.handleUpdateProjectTitle}
-
- // allow passed-in props to override any of the above
- {...childProps}
- />);
- }
- }
-
- ScratchDesktopComponent.propTypes = {
- loadingState: PropTypes.oneOf(LoadingStates),
- onFetchedInitialProjectData: PropTypes.func,
- onHasInitialProject: PropTypes.func,
- onLoadedProject: PropTypes.func,
- onLoadingCompleted: PropTypes.func,
- onLoadingStarted: PropTypes.func,
- onRequestNewProject: PropTypes.func,
- onTelemetrySettingsClicked: PropTypes.func,
- vm: PropTypes.instanceOf(VM).isRequired
- };
- const mapStateToProps = state => {
- const loadingState = state.scratchGui.projectState.loadingState;
- return {
- loadingState: loadingState,
- vm: state.scratchGui.vm
- };
- };
- const mapDispatchToProps = dispatch => ({
- onLoadingStarted: () => dispatch(openLoadingProject()),
- onLoadingCompleted: () => dispatch(closeLoadingProject()),
- onHasInitialProject: (hasInitialProject, loadingState) => {
- if (hasInitialProject) {
- // emulate sb-file-uploader
- return dispatch(requestProjectUpload(loadingState));
- }
-
- // `createProject()` might seem more appropriate but it's not a valid state transition here
- // setting the default project ID is a valid transition from NOT_LOADED and acts like "create new"
- return dispatch(setProjectId(defaultProjectId));
- },
- onFetchedInitialProjectData: (projectData, loadingState) =>
- dispatch(onFetchedProjectData(projectData, loadingState)),
- onLoadedProject: (loadingState, loadSuccess) => {
- const canSaveToServer = false;
- return dispatch(onLoadedProject(loadingState, canSaveToServer, loadSuccess));
- },
- onRequestNewProject: () => dispatch(requestNewProject(false)),
- onTelemetrySettingsClicked: () => dispatch(openTelemetryModal())
- });
-
- return connect(mapStateToProps, mapDispatchToProps)(ScratchDesktopComponent);
-};
// note that redux's 'compose' function is just being used as a general utility to make
// the hierarchy of HOC constructor calls clearer here; it has nothing to do with redux's
// ability to compose reducers.
const WrappedGui = compose(
+ ScratchDesktopAppStateHOC,
AppStateHOC,
- ScratchDesktopHOC
+ ScratchDesktopGUIHOC
)(GUI);
export default ;