mirror of
https://github.com/upscayl/upscayl.git
synced 2024-11-23 23:21:05 +01:00
Updated project
This commit is contained in:
commit
6c18b7eeb3
12
.github/FUNDING.yml
vendored
Normal file
12
.github/FUNDING.yml
vendored
Normal file
@ -0,0 +1,12 @@
|
||||
# These are supported funding model platforms
|
||||
|
||||
github: [iamWing]
|
||||
patreon: # Replace with a single Patreon username
|
||||
open_collective: # Replace with a single Open Collective username
|
||||
ko_fi: # Replace with a single Ko-fi username
|
||||
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
|
||||
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
|
||||
liberapay: # Replace with a single Liberapay username
|
||||
issuehunt: # Replace with a single IssueHunt username
|
||||
otechie: # Replace with a single Otechie username
|
||||
custom: ['https://paypal.me/iamWing0w0']
|
31
.gitignore
vendored
Normal file
31
.gitignore
vendored
Normal file
@ -0,0 +1,31 @@
|
||||
.vscode/
|
||||
coverage/
|
||||
dist/
|
||||
node_modules/
|
||||
out/
|
||||
|
||||
# Logs
|
||||
logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
|
||||
# Yarn Integrity file
|
||||
.yarn-integrity
|
||||
|
||||
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
||||
.grunt
|
||||
|
||||
# Optional npm cache directory
|
||||
.npm
|
||||
|
||||
# Optional eslint cache
|
||||
.eslintcache
|
||||
|
||||
# Windows thumbnail cache files
|
||||
Thumbs.db
|
||||
|
||||
# OS X files
|
||||
.DS_Store
|
||||
._*
|
165
CHANGELOG_PRE_V4.md
Normal file
165
CHANGELOG_PRE_V4.md
Normal file
@ -0,0 +1,165 @@
|
||||
# Changelog (pre `v4`)
|
||||
All notable changes up to `v3.0.0` are documented in this file.
|
||||
|
||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||
|
||||
## [Unreleased]
|
||||
|
||||
## [3.0.0] - 2020-12-26
|
||||
### Added
|
||||
- NPM packages `react-router` & `react-router-dom` to package dependencies.
|
||||
- NPM package `eslint-plugin-react-hooks` to `devDependencies` as required by
|
||||
the updated version of `eslint-config-airbnb`.
|
||||
- NPM package `copy-webpack-plugin` to `devDependencies` as replacement of
|
||||
`copy-pkg-json-webpack-plugin`.
|
||||
- NPM packages `@typescript-eslint/eslint-plugin` & `@typescript-eslint/parser`
|
||||
to `devDependencies` for using ESLint to lint TypeScript.
|
||||
|
||||
### Changed
|
||||
- __Project is now being developed based on Node.js `v14 (LTS)`.__
|
||||
- Minor version upgrades on package dependencies:
|
||||
- `eslint-import-resolver-webpack` - `0.11.1` -> `0.13.0`
|
||||
- `eslint-plugin-import` - `2.18.2` -> `2.22.1`
|
||||
- `eslint-plugin-jsx-a11y` - `6.2.3` -> `6.4.1`
|
||||
- `eslint-plugin-react` - `7.17.0` -> ``7.21.5`
|
||||
- `lodash` - `4.17.15` -> `4.17.20`
|
||||
- Major version upgrades on package dependencies:
|
||||
- `react` & `react-dom` - `16.12.0` -> `17.0.1`
|
||||
- `cross-env` - `5.2.1` -> `7.0.3`
|
||||
- `css-loader` - `1.0.1` -> `5.0.1`
|
||||
- `electron` - `3.1.13` -> `11.1.1`
|
||||
- `electron-builder` - `20.44.4` -> `22.9.1`
|
||||
- `eslint` - `5.16.0` -> `7.16.0`
|
||||
- `eslint-config-airbnb` - `17.1.1` -> `18.2.1`
|
||||
- `file-loader` - `2.0.0` -> `6.2.0`
|
||||
- `html-webpack-plugin` - `3.2.0` -> `4.5.0`
|
||||
- `mocha` - `5.2.0` -> `8.2.1`
|
||||
- `rimraf` - `2.7.1` -> `3.0.2`
|
||||
- `source-map-loader` - `0.2.4` -> `2.0.0`
|
||||
- `spectron` - `5.0.0` -> `13.0.0`
|
||||
- `style-loader` - `0.23.1` -> `2.0.0`
|
||||
- `ts-loader` - `5.4.5` -> `8.0.12`
|
||||
- `ts-node` - `7.0.1` -> `9.1.1`
|
||||
- `typescript` - `3.7.2` - > `4.1.3`
|
||||
- `webpack` - `4.41.2` -> `5.11.0`
|
||||
- `webpack-cli` - `3.3.10` -> `4.3.0`
|
||||
- Moved `@types` packages from `dependencies` to `devDependencies` as those
|
||||
have no need to be included in production builds.
|
||||
- Commands of NPM scripts `dev` & `prod` to make them work with Webpack 5.
|
||||
- Migrated to ESLint from TSLint.
|
||||
- Updated prefix of internal paths from `@` to `_` to avoid confusions with
|
||||
scoped NPM packages.
|
||||
- Indentation for `.ts` files is now set as `2` spaces instead of `4`.
|
||||
- `electron-builder` settings:
|
||||
- `win`:
|
||||
- Enabled `asar`. _(turn it off if the executable built doesn't work)_
|
||||
- Disabled one click installer.
|
||||
- Allowed custom installation directory in the installer built.
|
||||
- `mac`:
|
||||
- Changed build target back to `dmg`.
|
||||
- `buildVersion` is now being used as build number instead of just another
|
||||
parameter for semantic version number.
|
||||
- Updated section `Known issues` in `README`.
|
||||
|
||||
### Fixed
|
||||
- ESLint errors/warnings on `main.ts` & `renderer.tsx`.
|
||||
- `electron-builder` fails to build `dmg` on `macOS` (issue
|
||||
[electron-builder #3990])by upgrading the package version to `> 21.2.0`.
|
||||
|
||||
### Removed
|
||||
- NPM package `copy-pkg-json-webpack-plugin` as it doesn't work with Webpack 5
|
||||
and seems not very well maintained.
|
||||
- NPM packages `acorn` & `ajv` from `devDependencies` as they're not being used
|
||||
in this boilerplate.
|
||||
- NPM packages `tslint` & `tslint-microsoft-contrib` as TSLint is now
|
||||
deprecated.
|
||||
|
||||
## [2.0.2] - 2019-12-02
|
||||
`v2.0.2` is a minor hotfix release fixed the documentation error and build
|
||||
error on `macOS Catalina(10.15+)`.
|
||||
|
||||
### Added
|
||||
- Extended README section `Getting started` with `npm start` command
|
||||
description.
|
||||
- README section `Known issue`.
|
||||
|
||||
### Changed
|
||||
- Minor version upgrades on package dependencies.
|
||||
- `macOS` build target to `pkg` from default `dmg` due to no 32-bit apps
|
||||
support from `macOS Catalina` that caused `electron-builder` fails to build
|
||||
`dmg` image on `macOS` prior to `electron-builder@21.2.0`.
|
||||
_`pkg` build is unaffected and is used as a workaround for the current version
|
||||
prior to the major version upgrades on dependencies in next release._
|
||||
_Related issue: [electron-builder #3990](https://github.com/electron-userland/electron-builder/issues/3990)_
|
||||
|
||||
### Fixed
|
||||
- [Issue #2] - incorrect command `npm run install` to `npm install` in `README`.
|
||||
|
||||
## [2.0.1] - 2018-02-05
|
||||
`v2.0.1` is a minor hotfix release patched the `NODE_ENV` not set on Windows
|
||||
issue.
|
||||
|
||||
### Added
|
||||
- Package `cross-env` as `devDependencies`.
|
||||
- README section "Author".
|
||||
|
||||
### Fixed
|
||||
- NPM scripts won't set environment variables on Windows issue.
|
||||
|
||||
## 2.0.0 - 2018-02-04
|
||||
`v2.0.0` is a major release that most part of the boilerplate has been rewritten.
|
||||
|
||||
### Added
|
||||
- ESLint for code checking on Javascript files.
|
||||
- Airbnb's extensible `.eslintrc` package & its' peer dependencies.
|
||||
- `.eslintrc` that extends Airbnb's config and has customised rules configured.
|
||||
- _Rule `no-default-export` is set for JavaScript files to align with
|
||||
TypeScript._
|
||||
- ESLint plugin `eslint-import-resolver-webpack` for ESLint to resolve path
|
||||
aliases set in Webpack config.
|
||||
- Webpack plugin `copy-pkg-json-webpack-plugin` to generate a `package.json`
|
||||
file and pack into Webpack's bundled package for production.
|
||||
- Build commands `build:mac` & `build:win` to package & build the installer of
|
||||
your Electron app for macOS & Windows using `electron-builder`.
|
||||
- README section "Building the installer for your Electron app" & sub-section
|
||||
"Extra options".
|
||||
|
||||
### Changed
|
||||
- Refactored Webpack config file to have `mainConfig` & `rendererConfig`
|
||||
cleaned up, and set mode by environment variable.
|
||||
- `.gitignore` to ignore folder `out/` which will be auto-generated during the
|
||||
build process.
|
||||
- README section "How does it work?" is now renamed to "Getting started" &
|
||||
completed the documentation of this section.
|
||||
- README section "Folder structure" to reflect the changes in `v2.0.0`.
|
||||
|
||||
### Fixed
|
||||
- CSS files fail to inject into views issue by setting Webpack to use
|
||||
`style-loader` alongside with `css-loader` to bundle the files in Webpack
|
||||
config file.
|
||||
- `baseUrl` in `tsconfig.json` points to root directory incorrectly issue.
|
||||
Corrected to current directory so VSCode can resolves the path aliases
|
||||
correctly.
|
||||
|
||||
### Removed
|
||||
- Redux & React-Redux related settings, including packages listed on
|
||||
`devDependencies`, path aliases & folders listed in folder structure.
|
||||
- Since Electron's built-in IPC & basic React states should be enough to get
|
||||
the works done, and most Electron apps which have their application logic
|
||||
runs on Electron's `main` process rather then `renderer` process actually
|
||||
don't need React-Redux, `redux` & `react-redux` are no longer included in
|
||||
this boilerplate.
|
||||
- Redux & React-Redux can still be used on this boilerplate by installing the
|
||||
package yourself. For details, please refer to the corresponding library's
|
||||
documents, there's no different than working on any other project which
|
||||
isn't based on this boilerplate.
|
||||
- Separated Webpack config files for `development` & `production` mode.
|
||||
|
||||
[Unreleased]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/compare/v3.0.0...HEAD
|
||||
[2.0.1]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/compare/v2.0.0...v2.0.1
|
||||
[2.0.2]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/compare/v2.0.1...v2.0.2
|
||||
[3.0.0]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/compare/v2.0.2...v3.0.0
|
||||
|
||||
[Issue #2]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/issues/2
|
||||
[electron-builder #3990]: https://github.com/electron-userland/electron-builder/issues/3990
|
91
CHANGELOG_V4+.md
Normal file
91
CHANGELOG_V4+.md
Normal file
@ -0,0 +1,91 @@
|
||||
# Changelog `v4+`
|
||||
All notable changes to this project on `v4+` will be documented in this file.
|
||||
|
||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||
|
||||
## [Unreleased]
|
||||
|
||||
## [v4.0.0] - 2022-07-22
|
||||
### Added
|
||||
- Jest as default unit testing framework, with sample test suite for `main`.
|
||||
- Integrated Electron preload pattern.
|
||||
- NPM scripts:
|
||||
- `watch-test` to run Jest in watch mode.
|
||||
- `next-rc`, `next-patch`, `next-minor` & `next-major` for quick
|
||||
package version number advance.
|
||||
|
||||
### Changed
|
||||
- Webpack will now take the module path alias from `tsconfig.json` and set it
|
||||
for you thanks to `tsconfig-paths-webpack-plugin`. Manually set up in Webpack
|
||||
config is no longer needed.
|
||||
- `tsconfig` now configured to use `ES2020` features, with module resolution
|
||||
set to `Node16` to match the NodeJS version used by Electron.
|
||||
- Migrated to the new `createRoot` API introduced in React `v18`.
|
||||
- Some APIs changed in Electron `main` entry script:
|
||||
- `mainWindow` now use `loadFile` API instead of `loadURL`.
|
||||
- Replaced `app.on('ready')` with `app.whenReady()` to align with syntax from
|
||||
[Electron official quick start guide](https://www.electronjs.org/docs/latest/tutorial/quick-start).
|
||||
- `electron-builder` now configured to build `universal` `dmg` for mac, 32 &
|
||||
64 bit `exe` for Windows.
|
||||
- Moved `electron-builder`, Webpack & Webpack-related packages to
|
||||
`optionalDependencies`.
|
||||
- Revamped `README`.
|
||||
- __*Starting from this version, the maintenance schedule will be on a monthly
|
||||
update basis to keep the package dependencies up to date and keep the
|
||||
development going.*__
|
||||
|
||||
### Fixed
|
||||
- Allow to use `import default` statement on non ES modules (e.g. React,
|
||||
lodash) by enabling `esModuleInterop` in `tsconfig`.
|
||||
[#14](https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/issues/14)
|
||||
|
||||
### Removed
|
||||
- Mocha in favour of Jest.
|
||||
- Spectron as it ahs been deprecated.
|
||||
*See - [Spectron Deprecation Notice](https://www.electronjs.org/blog/spectron-deprecation-notice)*
|
||||
|
||||
### Updates on package dependencies
|
||||
### Added
|
||||
- `@types/jest`, `jest`, `ts-jest`, `eslint-plugin-jest` *- Jest support*
|
||||
- `eslint-config-airbnb-typescript` *- Enhance Airbnb's rules in TypeScript*
|
||||
- `ts-config-paths-webpack-plugin` *- Load `tsconfig` path alias into Webpack
|
||||
config*
|
||||
|
||||
### Updated
|
||||
- Major version updates:
|
||||
- `react` & `react-dom` - `17.0.1` -> `18.2.0`
|
||||
- `@types/react` - `17.0.0` -> `18.0.15`
|
||||
- `@types/react-dom` - `17.0.0` -> `18.0.6`
|
||||
- `@typescript-eslint/eslint-plugin` & `@typescript-eslint/parser` -
|
||||
`4.11.0` -> `5.30.7`
|
||||
- `electron` - `11.1.1` -> `19.0.9`
|
||||
- `eslint-config-airbnb` - `18.2.1` -> `19.0.4`
|
||||
- `copy-webpack-plugin` - `7.0.0` -> `11.0.0`
|
||||
- `css-loader` - `5.0.1` -> `6.7.1`
|
||||
- `electron-builder` - `22.9.1` -> `23.1.0`
|
||||
- `html-webpack-plugin` - `4.5.0` -> `5.5.0`
|
||||
- `style-loader` - `2.0.0` -> `3.3.1`
|
||||
- `ts-loader` - `8.0.12` -> `9.3.1`
|
||||
- `ts-config-paths` - `3.9.0` -> `4.0.0`
|
||||
- Minor & patch version updates:
|
||||
- `eslint-import-resolver-webpack` - `0.13.0` -> `0.13.2`
|
||||
- `eslint-plugin-import` - `2.22.1` -> `2.26.0`
|
||||
- `eslint-plugin-jsx-a11y` - `6.4.1` -> `6.6.1`
|
||||
- `eslint-plugin-react` - `7.21.5` -> `7.30.1`
|
||||
- `eslint-plugin-react-hoots` - `4.2.0` -> `4.6.0`
|
||||
- `lodash` - `4.17.20` -> `4.17.21`
|
||||
- `webpack` - `5.11.0` -> `5.73.0`
|
||||
- `webpack-cli` - `4.3.0` -> `4.10.0`
|
||||
|
||||
### Removed
|
||||
- `react-router`, `react-router-dom`, `@types/react-router`,
|
||||
`@types/react-router-dom`
|
||||
|
||||
*\- Not being used in any part of the boilerplate*
|
||||
- `@types/mocha`, `mocha`, `ts-node` *- Replaced by `@types/jest`, `jest` &
|
||||
`ts-jest`*
|
||||
- `spectron` *- Deprecated package; No replacement*
|
||||
|
||||
[Unreleased]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/compare/v4.0.0...HEAD
|
||||
[v4.0.0]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/compare/v3.0.0...v4.0.0
|
21
LICENSE
Normal file
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2018 Devtography
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
275
README.md
Normal file
275
README.md
Normal file
@ -0,0 +1,275 @@
|
||||
# Electron-React-TypeScript-Webpack-Boilerplate
|
||||
A boilerplate that let you instantly start working on your next [Electron] app
|
||||
in [TypeScript] with no time wasted messing with the config files.
|
||||
|
||||
- Ready to use [Electron] project template with [React], [Webpack] and
|
||||
[TypeScript] seamlessly integrated.
|
||||
- [ESLint] set up with TypeScript, Airbnb's rules, and [Jest] support.
|
||||
- [Jest] integrated and configured.
|
||||
- [`electron-builder`] for app packaging, with basic build config for Windows
|
||||
macOS included.
|
||||
- Clean, easy to read and alter config files. No config file is hidden behind
|
||||
yet another script!
|
||||
- Monthly maintenance to keep things up to date!
|
||||
|
||||
*This boilerplate is tested on the latest macOS and Windows. If anything
|
||||
doesn't work, please [file an issue].*
|
||||
|
||||
### Maintenance schedule
|
||||
Starting from `v4.0.0`, the project maintenance will become much more regular.
|
||||
A new release will be published on a monthly basis to keep the package
|
||||
dependencies, package configurations and APIs / syntax up to date.
|
||||
|
||||
Maintenance work will begin on 1st of each month, and expect the new version to
|
||||
be released within the first week of the month. New features from different
|
||||
tools integrated in this boilerplate might not always be implemented at once,
|
||||
especially on experimental features. If you want any particular feature to be
|
||||
implemented, please [file an issue], or consider make a [new pull request].
|
||||
|
||||
### Development plan
|
||||
- [ ] Create a `create-react-app`-like package initialiser __!!!__
|
||||
- [ ] Integrate another end-to-end testing framework to replace [Spectron]
|
||||
- [ ] Migrate to Webpack 5 `Asset Modules`
|
||||
|
||||
---
|
||||
|
||||
## 🚨 🚧 CAUTION 🚧 🚨
|
||||
- [Spectron] has officially been deprecated by the [Electron] team on
|
||||
February 1, 2022, thus, its' integration has also been dropped from this
|
||||
boilerplate on `v4+`.
|
||||
|
||||
A replacement will be integrated in future version (pending for `v5`).
|
||||
Currently evaluating different options including [Playwright] and
|
||||
[WebdriverIO].
|
||||
|
||||
*See - [Spectron Deprecation Notice]*
|
||||
|
||||
- `mocha` has been dropped and replaced by [Jest] on `v4+`. If you're using
|
||||
`mocha` as your unit testing framework, please reference to `package.json`
|
||||
from [`v3.0.0`].
|
||||
|
||||
---
|
||||
|
||||
## Getting started
|
||||
1. Clone this repository, or if you're hosting your Electron project on GitHub,
|
||||
click [`Use this template`] to create a new project.
|
||||
|
||||
2. Edit the following fields in `package.json` for your own project:
|
||||
|
||||
```json
|
||||
{
|
||||
"name": "your-project-name",
|
||||
"version": "whatever-you-like",
|
||||
"description": "your-own-description",
|
||||
"build": {
|
||||
"appId": "your-app-id",
|
||||
"productName": "your-product-name",
|
||||
"buildVersion": "your-build-number"
|
||||
},
|
||||
"author": "who's-the-author?",
|
||||
"license": "if-you-don't-want-to-use-MIT",
|
||||
"repository": "type-and-link-of-your-repo",
|
||||
"bugs": "issue-page-of-your-repo",
|
||||
"homepage": "homepage-of-your-repo"
|
||||
}
|
||||
```
|
||||
|
||||
3. `npm install` to install the dependencies.
|
||||
|
||||
*Please note that `optionalDependencies` should only be omitted on your
|
||||
CI/CD pipeline for unit testing. It's meant to save some bandwidth. You'll
|
||||
need all the packages listed for development.*
|
||||
|
||||
Done! Now run `npm run dev` to start the Webpack in development and watch mode.
|
||||
It's time to start working on your project.
|
||||
|
||||
Be aware that starting Webpack will only compile your files to `dist` folder
|
||||
but won't start the Electron app. Use `npm start` command to start your
|
||||
Electron app once the files are compiled.
|
||||
|
||||
__*Starting from `v4.0.0`, you no longer need to manually config your module
|
||||
path alias in `webpack.config.js`. All module path alias set in `tsconfig.json`
|
||||
will be configured in Webpack automatically thanks to [`tsconfig-paths`] and
|
||||
[`tsconfig-paths-webpack-plugin`].*__
|
||||
|
||||
## Build your Electron app package
|
||||
Different from the official [Electron quick start guide], this boilerplate uses
|
||||
[`electron-builder`] instead of [Electron Forge] to package your Electron app.
|
||||
|
||||
By default, the build configuration in `package.json` is configured to build the
|
||||
mac universal package (for Apple Silicon & Intel based machines) and Windows
|
||||
`exe` installer (both 32 & 64 bit). You should not need to change anything in
|
||||
the build script other than supplying the app icon unless you need to sign your
|
||||
code/package or build for Linux.
|
||||
|
||||
For code signing and notarization, or to build for Linux, please read
|
||||
[`electron-builder`'s document] for configuring the build script.
|
||||
|
||||
To package your Electron app, run `npm run prod` to get your code compiled in
|
||||
`production` mode, then use `npm run build:(win|mac)` to build the package.
|
||||
|
||||
## Known issues
|
||||
- [`electron-builder`] packages the file into Electron's `asar` archive format
|
||||
by default. Based on past experiences with old Electron & `electron-builder`
|
||||
versions, this might lead to runtime error on Windows while launching the
|
||||
installed Electron app.
|
||||
|
||||
One way to verify this issue is to build the mac package and see if your app
|
||||
runs fine on mac. If it's the case, you can override the `asar` archive
|
||||
option in the build configuration in `package.json` by adding `asar: false`
|
||||
in `win` section.
|
||||
|
||||
This solution isn't ideal but since `asar` archiving is
|
||||
meant to improve performance of reading files if bundler like Webpack is not
|
||||
being used. The app packaging workflow defined in this boilerplate already
|
||||
uses Webpack to minify your code in `production` builds, so there shouldn't
|
||||
be any significant performance different with `asar` archiving disabled.
|
||||
|
||||
## Project folders & files
|
||||
- `dist/` - [Webpack] output location
|
||||
|
||||
__Contents will be flushed automatically on execution of `npm run <dev|prod>`
|
||||
script.__
|
||||
|
||||
- `out/` - [`electron-builder`] output location
|
||||
|
||||
- `public/` - Global static assets.
|
||||
- `index.html` - Template for `HTML Webpack Plugin`
|
||||
|
||||
Update the value of `<title>` tag to change the default window title.
|
||||
|
||||
- `style.css` - `CSS` file location sample
|
||||
|
||||
Not much defined in this file. You can either put your `CSS` settings here
|
||||
or use any other tools you prefer.
|
||||
|
||||
- `src/` - Folder for all your source code
|
||||
- `main/` - For modules which run on the `main` process.
|
||||
- `main.ts` - [Electron] `main` process entry point
|
||||
|
||||
- `preload` - Preload scripts go here
|
||||
- `ipc-api.ts` - APIs for IPC between `main` & `renderer`
|
||||
|
||||
Consider convert this module into a collection of submodules if you have
|
||||
many APIs for IPC. See example as below:
|
||||
```ts
|
||||
// ipc-api/index.ts
|
||||
import submoduleA from './submodule-a';
|
||||
import submoduleB from './submodule-b';
|
||||
|
||||
export default { ...submoduleA, ...submoduleB };
|
||||
|
||||
// ipc-api/submodule-a.ts
|
||||
import { ipcRenderer } from 'electron';
|
||||
|
||||
function a { ipcRenderer.send('a'); }
|
||||
|
||||
export default { a };
|
||||
|
||||
// ipc-api/submodule-b.ts
|
||||
import { ipcRenderer } from 'electron';
|
||||
|
||||
function b { ipcRenderer.send('b'); }
|
||||
|
||||
export default { b };
|
||||
```
|
||||
|
||||
- `preload.ts` - [Electron] preload script entry point
|
||||
|
||||
There should be no need to modify this file unless you want to use other
|
||||
key(s) for your IPC APIs. By default, all APIs defined in `ipc-api`
|
||||
module are exposed under key `ipcApi` in `contextBridge`.
|
||||
|
||||
- `renderer/` - Where the frontend scripts stay
|
||||
- `App.tsx` - Root [React] component
|
||||
- `renderer.tsx` - [Electron] `renderer` process entry point
|
||||
|
||||
*`public/style.css` imported here. Change it if you want.*
|
||||
|
||||
- `types/` - Home for self-defined `.d.ts` files
|
||||
- `global.d.ts` - Extends global scope interfaces
|
||||
|
||||
This file includes ambient declaration for calling the IPC APIs defined in
|
||||
`preload/ipc-api` from the `renderer`. Remember __NOT__ to remove this
|
||||
part, otherwise TypeScript will tell you `type not exist`. However, if
|
||||
you've opted to use a different key other than `ipcAPI` in the preload
|
||||
script, __DO__ remember to update this file to match your own settings.
|
||||
|
||||
- `utils/` - Place to store the helper scripts
|
||||
- `node-env.ts` - Shortcut to determine `NODE` environment
|
||||
|
||||
- `tests/` - Unit testing files location
|
||||
|
||||
To avoid test files mixing up with the source code, [Jest] is configured to
|
||||
look for test file(s) within this folder only.
|
||||
|
||||
File name of the test files can either be `[filename].test.tsx` or
|
||||
`[filename].spec.ts(x)`. `js(x)` can also be used for test files, but I assume
|
||||
you'd use TypeScript if you're using this boilerplate.
|
||||
|
||||
- `main/main.spec.ts` - Sample test file for `src/main/main`
|
||||
- `tsconfig.json` - TypeScript config file for `tests` module
|
||||
- `.eslintignore` - [ESLint] ignore file
|
||||
- `.eslintrc` - [ESLint] config file
|
||||
|
||||
Configured to use Airbnb's rules with [TypeScript] supported, and rules for
|
||||
[Jest] applied.
|
||||
|
||||
- `.gitignore` - Git ignore file
|
||||
- `CHANGELOG_PRE_V4.md` - Changelog of this boilerplate prior to `v4.0.0`
|
||||
- `CHANGELOG_V4+.md` - Changelog of this boilerplate from `v4.0.0` onwards
|
||||
- `jest.config.js` - [Jest] config file
|
||||
- `LICENSE` - MIT license
|
||||
- `package-lock.json`
|
||||
- `package.json`
|
||||
|
||||
Includes basic build config for `electron-builder`. It's likely that you'll
|
||||
have to personalise the build config when it comes to the time you're about
|
||||
to release your app. Please read [`electron-builder`'s document] for the
|
||||
build config setup guides.
|
||||
|
||||
- `README.md`
|
||||
- `tsconfig.json` - [TypeScript] config file
|
||||
|
||||
Module path aliases are configured here. [Jest] & [Webpack] will pick up the
|
||||
alias settings here to config their own. No need to manually config in Jest
|
||||
& Webpack again.
|
||||
|
||||
- `webpack.config.json` - [Webpack] config file
|
||||
|
||||
Includes configurations targetting `electron-main`, `electron-preload`, and
|
||||
`electron-renderer` respectively.
|
||||
|
||||
## Author
|
||||
[Wing Chau](https://github.com/iamWing) [@Devtography](https://github.com/Devtography)
|
||||
|
||||
## Donation
|
||||
Maintaining this project takes time, lots of cups of coffee, and I do it for
|
||||
free. Consider buy me coffee via [donations]. 100% of donation will fund my
|
||||
coffee buying budget for quality coffee beans from great roasters I know 😉 ☕️️
|
||||
|
||||
## License
|
||||
Electron React TypeScript Webpack Boilerplate is open source software
|
||||
[licensed as MIT](LICENSE).
|
||||
|
||||
[Electron]: https://www.electronjs.org
|
||||
[React]: https://reactjs.org
|
||||
[Webpack]: https://webpack.js.org
|
||||
[TypeScript]: https://www.typescriptlang.org
|
||||
[ESLint]: http://eslint.org
|
||||
[Jest]: https://jestjs.io
|
||||
[`electron-builder`]: https://github.com/electron-userland/electron-builder
|
||||
[file an issue]: https://www.electronjs.org
|
||||
[new pull request]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/compare
|
||||
[Spectron]: https://github.com/electron-userland/spectron
|
||||
[Playwright]: https://playwright.dev
|
||||
[WebdriverIO]: https://webdriver.io
|
||||
[Spectron Deprecation Notice]: https://www.electronjs.org/blog/spectron-deprecation-notice
|
||||
[`v3.0.0`]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/releases/tag/v3.0.0
|
||||
[`Use this template`]: https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/generate
|
||||
[`tsconfig-paths`]: https://github.com/dividab/tsconfig-paths
|
||||
[`tsconfig-paths-webpack-plugin`]: https://github.com/dividab/tsconfig-paths-webpack-plugin
|
||||
[Electron quick start guide]: https://www.electronjs.org/docs/latest/tutorial/quick-start
|
||||
[Electron Forge]: https://github.com/electron-userland/electron-forge
|
||||
[`electron-builder`'s document]: https://www.electron.build
|
||||
[donations]: https://github.com/sponsors/iamWing
|
30
jest.config.js
Normal file
30
jest.config.js
Normal file
@ -0,0 +1,30 @@
|
||||
const { pathsToModuleNameMapper } = require('ts-jest')
|
||||
const { compilerOptions } = require('./tsconfig.json')
|
||||
|
||||
/** @type {import('ts-jest/dist/types').InitialOptionsTsJest} */
|
||||
module.exports = config = {
|
||||
testEnvironment: 'node',
|
||||
globals: {
|
||||
'ts-jest': {
|
||||
tsconfig: 'tsconfig.json',
|
||||
},
|
||||
},
|
||||
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json'],
|
||||
moduleNameMapper: pathsToModuleNameMapper(
|
||||
compilerOptions.paths,
|
||||
{ prefix: '<rootDir>/' },
|
||||
),
|
||||
modulePathIgnorePatterns: [
|
||||
'<rootDir>/dist',
|
||||
'<rootDir>/node_modules',
|
||||
'<rootDir>/out',
|
||||
],
|
||||
transform: {
|
||||
'^.+\\.(ts|tsx)$': 'ts-jest',
|
||||
},
|
||||
testMatch: [
|
||||
'**/tests/**/*.(spec|test).(ts?(x)|js?(x))',
|
||||
],
|
||||
collectCoverage: true,
|
||||
verbose: true,
|
||||
};
|
89
package.json
Normal file
89
package.json
Normal file
@ -0,0 +1,89 @@
|
||||
{
|
||||
"name": "electron-react-typescript-webpack-boilerplate",
|
||||
"version": "4.0.0",
|
||||
"description": "Pre-configured boilerplate for Electron + React + TypeScript",
|
||||
"main": "./dist/main.bundle.js",
|
||||
"scripts": {
|
||||
"start": "electron ./dist/main.bundle.js",
|
||||
"webpack:dev": "rimraf dist && cross-env NODE_ENV=development webpack --watch --progress --color",
|
||||
"prod": "rimraf dist && cross-env NODE_ENV=production webpack --progress --color",
|
||||
"test": "cross-env NODE_ENV=test jest",
|
||||
"watch-test": "npm run test -- --watchAll",
|
||||
"next-rc": "npm --no-git-tag-version version prerelease --preid=rc",
|
||||
"next-patch": "npm --no-git-tag-version version patch",
|
||||
"next-minor": "npm --no-git-tag-version version minor",
|
||||
"next-major": "npm --no-git-tag-version version major",
|
||||
"build:win": "electron-builder build --win",
|
||||
"build:mac": "electron-builder build --mac"
|
||||
},
|
||||
"build": {
|
||||
"appId": "com.devtography.electron_boilerplate",
|
||||
"productName": "Electron+React+TypeScript Boilerplate",
|
||||
"directories": {
|
||||
"app": "./dist/",
|
||||
"output": "./out/"
|
||||
},
|
||||
"mac": {
|
||||
"target": {
|
||||
"target": "dmg",
|
||||
"arch": "universal"
|
||||
}
|
||||
},
|
||||
"win": {
|
||||
"target": {
|
||||
"target": "nsis",
|
||||
"arch": [
|
||||
"x64",
|
||||
"ia32"
|
||||
]
|
||||
}
|
||||
},
|
||||
"nsis": {
|
||||
"oneClick": false,
|
||||
"allowToChangeInstallationDirectory": true
|
||||
},
|
||||
"buildVersion": "1"
|
||||
},
|
||||
"author": "Wing Chau @Devtography",
|
||||
"license": "MIT",
|
||||
"dependencies": {
|
||||
"@tensorflow/tfjs": "^3.19.0",
|
||||
"react": "^18.2.0",
|
||||
"react-dom": "^18.2.0",
|
||||
"react-dropzone": "^14.2.2",
|
||||
"upscaler": "^1.0.0-beta.5"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/jest": "^28.1.6",
|
||||
"@types/react": "^18.0.15",
|
||||
"@types/react-dom": "^18.0.6",
|
||||
"cross-env": "^7.0.3",
|
||||
"electron": "^19.0.9",
|
||||
"jest": "^28.1.3",
|
||||
"ts-jest": "^28.0.7",
|
||||
"typescript": "^4.7.4"
|
||||
},
|
||||
"optionalDependencies": {
|
||||
"copy-webpack-plugin": "^11.0.0",
|
||||
"css-loader": "^6.7.1",
|
||||
"electron-builder": "^23.1.0",
|
||||
"file-loader": "^6.2.0",
|
||||
"html-webpack-plugin": "^5.5.0",
|
||||
"lodash": "^4.17.21",
|
||||
"rimraf": "^3.0.2",
|
||||
"style-loader": "^3.3.1",
|
||||
"ts-loader": "^9.3.1",
|
||||
"tsconfig-paths": "^4.0.0",
|
||||
"tsconfig-paths-webpack-plugin": "^3.5.2",
|
||||
"webpack": "^5.73.0",
|
||||
"webpack-cli": "^4.10.0"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://github.com/Devtography/electron-react-typescript-webpack-boilerplate"
|
||||
},
|
||||
"bugs": {
|
||||
"url": "https://github.com/Devtography/electron-react-typescript-webpack-boilerplate/issues"
|
||||
},
|
||||
"homepage": "https://github.com/Devtography/electron-react-typescript-webpack-boilerplate#readme"
|
||||
}
|
13
public/index.html
Normal file
13
public/index.html
Normal file
@ -0,0 +1,13 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title>Hello World!</title>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div id="app" class="app"></div>
|
||||
</body>
|
||||
|
||||
</html>
|
3
public/style.css
Normal file
3
public/style.css
Normal file
@ -0,0 +1,3 @@
|
||||
.app {
|
||||
color: grey;
|
||||
}
|
68
src/main/main.ts
Normal file
68
src/main/main.ts
Normal file
@ -0,0 +1,68 @@
|
||||
/**
|
||||
* Entry point of the Election app.
|
||||
*/
|
||||
import * as path from "path";
|
||||
import * as nodeEnv from "_utils/node-env";
|
||||
// eslint-disable-next-line import/no-extraneous-dependencies
|
||||
import { BrowserWindow, app, ipcMain } from "electron";
|
||||
|
||||
let mainWindow: Electron.BrowserWindow | undefined;
|
||||
|
||||
function createWindow() {
|
||||
// Create the browser window.
|
||||
mainWindow = new BrowserWindow({
|
||||
height: 600,
|
||||
width: 800,
|
||||
webPreferences: {
|
||||
devTools: nodeEnv.dev,
|
||||
preload: path.join(__dirname, "./preload.bundle.js"),
|
||||
webSecurity: nodeEnv.prod,
|
||||
nodeIntegration: true,
|
||||
},
|
||||
});
|
||||
|
||||
// and load the index.html of the app.
|
||||
mainWindow.loadFile("index.html").finally(() => {
|
||||
/* no action */
|
||||
});
|
||||
|
||||
// Emitted when the window is closed.
|
||||
mainWindow.on("closed", () => {
|
||||
mainWindow = undefined;
|
||||
});
|
||||
}
|
||||
app.commandLine.appendSwitch("ignore-gpu-blacklist");
|
||||
app
|
||||
.whenReady()
|
||||
.then(() => {
|
||||
if (nodeEnv.dev || nodeEnv.prod) createWindow();
|
||||
|
||||
app.on("activate", () => {
|
||||
if (BrowserWindow.getAllWindows.length === 0) createWindow();
|
||||
});
|
||||
})
|
||||
.finally(() => {
|
||||
/* no action */
|
||||
});
|
||||
|
||||
app.on("window-all-closed", () => {
|
||||
if (process.platform !== "darwin") app.quit();
|
||||
});
|
||||
|
||||
// const getUpscaledImage = async () => {
|
||||
// const file = fs.readFileSync(path.resolve(__dirname, "./image.png"));
|
||||
// const image = tf.node.decodeImage(file, 3);
|
||||
// const tensor = await upscaler.upscale(image, {
|
||||
// output: "tensor",
|
||||
// patchSize: 64,
|
||||
// padding: 6,
|
||||
// });
|
||||
// image.dispose();
|
||||
// const upscaledTensor = await tf.node.encodePng(tensor);
|
||||
// tensor.dispose();
|
||||
// return upscaledTensor;
|
||||
// };
|
||||
|
||||
ipcMain.on("hello", async () => {});
|
||||
|
||||
export const exportedForTests = nodeEnv.test ? { createWindow } : undefined;
|
4
src/preload/preload.ts
Normal file
4
src/preload/preload.ts
Normal file
@ -0,0 +1,4 @@
|
||||
import { contextBridge } from "electron";
|
||||
import { ipcRenderer } from "electron";
|
||||
|
||||
contextBridge.exposeInMainWorld("ipcRenderer", ipcRenderer);
|
121
src/renderer/App.css
Normal file
121
src/renderer/App.css
Normal file
@ -0,0 +1,121 @@
|
||||
body {
|
||||
padding: 40px;
|
||||
font-family: -apple-system, BlinkMacSystemFont, sans-serif;
|
||||
display: flex;
|
||||
}
|
||||
* {
|
||||
box-sizing: border-box;
|
||||
}
|
||||
#root {
|
||||
display: flex;
|
||||
}
|
||||
html,
|
||||
body,
|
||||
#root {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
.dropzone {
|
||||
cursor: move; /* fallback if grab cursor is unsupported */
|
||||
cursor: grab;
|
||||
cursor: -moz-grab;
|
||||
cursor: -webkit-grab;
|
||||
width: 100%;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
flex: 1;
|
||||
padding: 40px;
|
||||
border: 5px dotted #ddd;
|
||||
margin: 0 auto;
|
||||
border-radius: 20px;
|
||||
}
|
||||
|
||||
.app {
|
||||
width: 100%;
|
||||
text-align: center;
|
||||
display: flex;
|
||||
flex: 1;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.original-image {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
flex: 1;
|
||||
flex-direction: column;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.original img {
|
||||
transition-duration: 1.2s;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
button {
|
||||
background: #ddd;
|
||||
padding: 5px 10px;
|
||||
border: 1px solid #aaa;
|
||||
}
|
||||
button:first-child {
|
||||
border-radius: 5px 0 0 5px;
|
||||
border-right: none;
|
||||
}
|
||||
button:last-child {
|
||||
border-radius: 0 5px 5px 0;
|
||||
}
|
||||
|
||||
button.active {
|
||||
background: #c4c4ff;
|
||||
}
|
||||
|
||||
.display {
|
||||
position: relative;
|
||||
margin-top: 5px;
|
||||
}
|
||||
.display .image-container {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.display .scaled-up img {
|
||||
display: block;
|
||||
margin: 0;
|
||||
position: absolute;
|
||||
left: 0%;
|
||||
}
|
||||
|
||||
.dragOverlay {
|
||||
position: absolute;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
z-index: 1;
|
||||
}
|
||||
|
||||
.dragger {
|
||||
cursor: pointer;
|
||||
height: 100%;
|
||||
width: 4px;
|
||||
background: rgba(255, 255, 255, 0.8);
|
||||
position: absolute;
|
||||
transform: translate(50px 0);
|
||||
}
|
||||
|
||||
.header {
|
||||
display: flex;
|
||||
width: 100%;
|
||||
min-height: 40px;
|
||||
}
|
||||
.interpolation {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
}
|
182
src/renderer/App.tsx
Normal file
182
src/renderer/App.tsx
Normal file
@ -0,0 +1,182 @@
|
||||
import Upscaler from "upscaler";
|
||||
import React, { useCallback, useState, useEffect, useRef } from "react";
|
||||
import { useDropzone } from "react-dropzone";
|
||||
import "./App.css";
|
||||
|
||||
const upscaler = new Upscaler();
|
||||
function App() {
|
||||
const [src, setSrc] = useState<string>();
|
||||
const [originalSize, setOriginalSize] = useState({ width: 0, height: 0 });
|
||||
const [scale, setScale] = useState(1);
|
||||
const [interpolation, setInterpolation] = useState("bicubic");
|
||||
const [upscaledImageSrc, setUpscaledImageSrc] = useState<string>();
|
||||
const [displayUpscaledImageSrc, setDisplayUpscaledImageSrc] = useState(false);
|
||||
const [dragX, setDragX] = useState(0.5);
|
||||
const [dragging, setDragging] = useState(false);
|
||||
const container = useRef<HTMLDivElement>();
|
||||
const onDrop = useCallback((acceptedFiles) => {
|
||||
const file = acceptedFiles[0];
|
||||
const fr = new FileReader();
|
||||
fr.onload = async () => {
|
||||
setSrc(fr.result as string);
|
||||
};
|
||||
fr.readAsDataURL(file);
|
||||
}, []);
|
||||
const { getRootProps, getInputProps, isDragActive } = useDropzone({ onDrop });
|
||||
useEffect(() => {
|
||||
if (src) {
|
||||
const img = new Image();
|
||||
img.crossOrigin = "anonymous";
|
||||
img.src = src;
|
||||
img.onload = () => {
|
||||
upscaler.upscale(img).then(setUpscaledImageSrc);
|
||||
const width = img.width;
|
||||
const height = img.height;
|
||||
setOriginalSize({
|
||||
width,
|
||||
height,
|
||||
});
|
||||
};
|
||||
}
|
||||
}, [src]);
|
||||
|
||||
useEffect(() => {
|
||||
if (originalSize) {
|
||||
let upscaledImageSrcTimer;
|
||||
const timer = setTimeout(() => {
|
||||
setScale(3);
|
||||
upscaledImageSrcTimer = setTimeout(() => {
|
||||
setDisplayUpscaledImageSrc(true);
|
||||
}, 1200);
|
||||
}, 300);
|
||||
return () => {
|
||||
clearTimeout(timer);
|
||||
clearTimeout(upscaledImageSrcTimer);
|
||||
};
|
||||
}
|
||||
}, [originalSize]);
|
||||
|
||||
const startDragging = () => {
|
||||
setDragging(true);
|
||||
};
|
||||
|
||||
const drag = (e) => {
|
||||
if (dragging) {
|
||||
const offsetWidth = container.current.offsetWidth;
|
||||
const x = e.clientX - (window.innerWidth - offsetWidth) / 2 - 10;
|
||||
setDragX(x / offsetWidth);
|
||||
}
|
||||
};
|
||||
|
||||
const stopDragging = () => {
|
||||
console.log("stop");
|
||||
setDragging(false);
|
||||
};
|
||||
|
||||
if (src) {
|
||||
const left = dragX * 100;
|
||||
return (
|
||||
<div
|
||||
className="original-image"
|
||||
style={{
|
||||
width: originalSize ? originalSize.width * scale : null,
|
||||
}}
|
||||
>
|
||||
<div className="header">
|
||||
{displayUpscaledImageSrc && (
|
||||
<>
|
||||
<div className="interpolation">
|
||||
<button
|
||||
className={interpolation === "none" ? "active" : null}
|
||||
onClick={() => setInterpolation("none")}
|
||||
>
|
||||
None
|
||||
</button>
|
||||
<button
|
||||
className={interpolation === "bicubic" ? "active" : null}
|
||||
onClick={() => setInterpolation("bicubic")}
|
||||
>
|
||||
Bicubic interpolation
|
||||
</button>
|
||||
</div>
|
||||
<div>Upscaled image</div>
|
||||
</>
|
||||
)}
|
||||
</div>
|
||||
<div
|
||||
className="display"
|
||||
style={{
|
||||
width: originalSize ? originalSize.width * scale : null,
|
||||
height: originalSize ? originalSize.height * scale : null,
|
||||
}}
|
||||
>
|
||||
{displayUpscaledImageSrc && (
|
||||
<div
|
||||
className="dragOverlay"
|
||||
ref={container}
|
||||
onMouseMove={drag}
|
||||
onMouseUp={stopDragging}
|
||||
>
|
||||
<div
|
||||
className="dragger"
|
||||
onMouseDown={startDragging}
|
||||
style={{
|
||||
left: `calc(${left}%)`,
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
<div className="image-container original">
|
||||
<img
|
||||
src={src}
|
||||
alt="Original"
|
||||
width={originalSize ? originalSize.width * scale : null}
|
||||
style={{
|
||||
imageRendering: interpolation === "none" ? "pixelated" : null,
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
{displayUpscaledImageSrc && (
|
||||
<div
|
||||
className="image-container scaled-up"
|
||||
style={{
|
||||
width: `${100 - left}%`,
|
||||
left: `${left}%`,
|
||||
}}
|
||||
>
|
||||
<img
|
||||
style={{
|
||||
left: ((originalSize.width * scale * left) / 100) * -1,
|
||||
}}
|
||||
alt="Upscaled"
|
||||
src={upscaledImageSrc}
|
||||
width={originalSize ? originalSize.width * scale : null}
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
{displayUpscaledImageSrc && <p>Resized to 3x</p>}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="dropzone" {...getRootProps()}>
|
||||
<input {...getInputProps()} />
|
||||
{isDragActive ? (
|
||||
<p>Drop the files here ...</p>
|
||||
) : (
|
||||
<p>Drag 'n' drop some files here, or click to select files</p>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default () => {
|
||||
return (
|
||||
<div className="app">
|
||||
<h1>React Demo Integration</h1>
|
||||
<App />
|
||||
</div>
|
||||
);
|
||||
};
|
BIN
src/renderer/image.jpg
Normal file
BIN
src/renderer/image.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 160 KiB |
13
src/renderer/renderer.tsx
Normal file
13
src/renderer/renderer.tsx
Normal file
@ -0,0 +1,13 @@
|
||||
/**
|
||||
* React renderer.
|
||||
*/
|
||||
// Import the styles here to process them with webpack
|
||||
import "_public/style.css";
|
||||
|
||||
import App from "_renderer/App";
|
||||
import * as React from "react";
|
||||
import { createRoot } from "react-dom/client";
|
||||
|
||||
const container = document.getElementById("app");
|
||||
const root = createRoot(container!);
|
||||
root.render(<App />);
|
10
src/types/global.d.ts
vendored
Normal file
10
src/types/global.d.ts
vendored
Normal file
@ -0,0 +1,10 @@
|
||||
import { ipcRenderer } from "electron";
|
||||
declare global {
|
||||
interface Window {
|
||||
/** APIs for Electron IPC */
|
||||
ipcRenderer?: typeof ipcRenderer;
|
||||
}
|
||||
}
|
||||
|
||||
// Makes TS sees this as an external modules so we can extend the global scope.
|
||||
export {};
|
12
src/utils/node-env.ts
Normal file
12
src/utils/node-env.ts
Normal file
@ -0,0 +1,12 @@
|
||||
/**
|
||||
* Predefined NODE_ENV values
|
||||
*/
|
||||
|
||||
/** `NODE_ENV=production` */
|
||||
export const prod = process.env.NODE_ENV === 'production';
|
||||
|
||||
/** `NODE_ENV=development` */
|
||||
export const dev = process.env.NODE_ENV === 'development';
|
||||
|
||||
/** `NODE_ENV=test` */
|
||||
export const test = process.env.NODE_ENV === 'test';
|
25
tests/main/main.spec.ts
Normal file
25
tests/main/main.spec.ts
Normal file
@ -0,0 +1,25 @@
|
||||
import { exportedForTests } from '_main/main';
|
||||
import { BrowserWindow } from 'electron';
|
||||
|
||||
jest.mock('electron', () => ({
|
||||
app: {
|
||||
on: jest.fn(),
|
||||
whenReady: jest.fn(() => Promise.resolve()),
|
||||
},
|
||||
ipcMain: { on: jest.fn() },
|
||||
BrowserWindow: jest.fn().mockImplementation(() => ({
|
||||
loadFile: jest.fn(() => Promise.resolve()),
|
||||
on: jest.fn(),
|
||||
})),
|
||||
}));
|
||||
|
||||
test('Private props exported for unit tests', () => {
|
||||
expect(exportedForTests).toBeDefined();
|
||||
});
|
||||
|
||||
test('func createWindow()', () => {
|
||||
const { createWindow } = exportedForTests!;
|
||||
|
||||
createWindow();
|
||||
expect(BrowserWindow).toHaveBeenCalledTimes(1);
|
||||
});
|
7
tests/tsconfig.json
Normal file
7
tests/tsconfig.json
Normal file
@ -0,0 +1,7 @@
|
||||
{
|
||||
"extends": "../tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"noEmit": true
|
||||
},
|
||||
"include": ["./**/*"]
|
||||
}
|
28
tsconfig.json
Normal file
28
tsconfig.json
Normal file
@ -0,0 +1,28 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"strict": false,
|
||||
"allowJs": true,
|
||||
"baseUrl": ".",
|
||||
"module": "ES2020",
|
||||
"moduleResolution": "Node16",
|
||||
|
||||
"noImplicitAny": false,
|
||||
"paths": {
|
||||
"_/*": ["src/*"],
|
||||
"_public/*": ["public/*"],
|
||||
"_main/*": ["src/main/*"],
|
||||
"_preload/*": ["src/preload/*"],
|
||||
"_renderer/*": ["src/renderer/*"],
|
||||
"_types/*": ["src/types/*"],
|
||||
"_utils/*": ["src/utils/*"],
|
||||
"_tests/*": ["tests/*"]
|
||||
},
|
||||
"typeRoots": ["./node_modules/@types", "./src/types"],
|
||||
"sourceMap": true,
|
||||
"allowSyntheticDefaultImports": true,
|
||||
"esModuleInterop": true,
|
||||
"jsx": "react",
|
||||
"target": "ES2020"
|
||||
},
|
||||
"include": ["src/**/*", "tests/**/*"]
|
||||
}
|
95
webpack.config.js
Normal file
95
webpack.config.js
Normal file
@ -0,0 +1,95 @@
|
||||
const lodash = require('lodash');
|
||||
const CopyPlugin = require('copy-webpack-plugin');
|
||||
const HtmlWebpackPlugin = require('html-webpack-plugin');
|
||||
const TsconfigPathsPlugin = require('tsconfig-paths-webpack-plugin');
|
||||
const path = require('path');
|
||||
|
||||
function srcPaths(src) {
|
||||
return path.join(__dirname, src);
|
||||
}
|
||||
|
||||
const isEnvProduction = process.env.NODE_ENV === 'production';
|
||||
const isEnvDevelopment = process.env.NODE_ENV === 'development';
|
||||
|
||||
// #region Common settings
|
||||
const commonConfig = {
|
||||
devtool: isEnvDevelopment ? 'source-map' : false,
|
||||
mode: isEnvProduction ? 'production' : 'development',
|
||||
output: { path: srcPaths('dist') },
|
||||
node: { __dirname: false, __filename: false },
|
||||
resolve: {
|
||||
extensions: ['.js', '.json', '.ts', '.tsx'],
|
||||
plugins: [new TsconfigPathsPlugin({
|
||||
configFile: './tsconfig.json',
|
||||
extensions: ['.js', '.json', '.ts', '.tsx'],
|
||||
})],
|
||||
},
|
||||
module: {
|
||||
rules: [
|
||||
{
|
||||
test: /\.(ts|tsx)$/,
|
||||
exclude: /node_modules/,
|
||||
loader: 'ts-loader',
|
||||
},
|
||||
{
|
||||
test: /\.(scss|css)$/,
|
||||
use: ['style-loader', 'css-loader'],
|
||||
},
|
||||
{
|
||||
test: /\.(jpg|png|svg|ico|icns)$/,
|
||||
loader: 'file-loader',
|
||||
options: {
|
||||
name: '[path][name].[ext]',
|
||||
},
|
||||
},
|
||||
],
|
||||
},
|
||||
};
|
||||
// #endregion
|
||||
|
||||
const mainConfig = lodash.cloneDeep(commonConfig);
|
||||
mainConfig.entry = './src/main/main.ts';
|
||||
mainConfig.target = 'electron-main';
|
||||
mainConfig.output.filename = 'main.bundle.js';
|
||||
mainConfig.plugins = [
|
||||
new CopyPlugin({
|
||||
patterns: [
|
||||
{
|
||||
from: 'package.json',
|
||||
to: 'package.json',
|
||||
transform: (content, _path) => {
|
||||
const jsonContent = JSON.parse(content);
|
||||
const electronVersion = jsonContent.devDependencies.electron;
|
||||
|
||||
delete jsonContent.devDependencies;
|
||||
delete jsonContent.optionalDependencies;
|
||||
delete jsonContent.scripts;
|
||||
delete jsonContent.build;
|
||||
|
||||
jsonContent.main = './main.bundle.js';
|
||||
jsonContent.scripts = { start: 'electron ./main.bundle.js' };
|
||||
jsonContent.devDependencies = { electron: electronVersion }
|
||||
|
||||
return JSON.stringify(jsonContent, undefined, 2);
|
||||
},
|
||||
},
|
||||
],
|
||||
}),
|
||||
];
|
||||
|
||||
const preloadConfig = lodash.cloneDeep(commonConfig);
|
||||
preloadConfig.entry = './src/preload/preload.ts';
|
||||
preloadConfig.target = 'electron-preload';
|
||||
preloadConfig.output.filename = 'preload.bundle.js';
|
||||
|
||||
const rendererConfig = lodash.cloneDeep(commonConfig);
|
||||
rendererConfig.entry = './src/renderer/renderer.tsx';
|
||||
rendererConfig.target = 'electron-renderer';
|
||||
rendererConfig.output.filename = 'renderer.bundle.js';
|
||||
rendererConfig.plugins = [
|
||||
new HtmlWebpackPlugin({
|
||||
template: path.resolve(__dirname, './public/index.html'),
|
||||
}),
|
||||
];
|
||||
|
||||
module.exports = [mainConfig, preloadConfig, rendererConfig];
|
Loading…
Reference in New Issue
Block a user