Compare commits

...

86 Commits
2.2.1 ... 2.3.3

Author SHA1 Message Date
Dave Simpson
77136687d3 Use macos-latest runner for macOS ARM build (#2513) 2024-09-24 18:30:58 +02:00
github-actions[bot]
16bc1a4610 Updated translation files (#2392)
Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
2024-09-24 17:05:43 +02:00
Dave Simpson
2921979678 Use macos-13 for Intel build (#2508) 2024-09-24 15:59:16 +02:00
Giacomo Cusinato
a5bf56ffa6 feat: upload using programmer by default if board requires it 2024-09-19 11:57:42 +02:00
Giacomo Cusinato
2de8bd1717 feat: decode grpc status objects and map them to protocol types
Status object thrown by grpc commands contains metadata that needs to be serialized in order to map it to custom errors generated through proto files https://github.com/grpc/grpc-web/issues/399
2024-09-19 11:57:42 +02:00
Giacomo Cusinato
1ec0a8cc77 feat: use Arduino CLI 1.0.4 (#2457)
* fix: use `@pingghost/protoc` to compile proto files

The npm package previously used (`protoc`) is still lacking apple arm32 support, see https://github.com/YePpHa/node-protoc/pull/10

* feat: use Arduino CLI 1.0.4

* fix: allow use of node16 in github actions

* chore: update `arduino-language-server` version for cli-1.0.0

* fix: deprecated platform order test

Arduino deprecated platforms should have more priority then other deprecated ones
2024-09-06 11:38:55 +02:00
Giacomo Cusinato
c3adde5460 feat: add shared space support (#2486) 2024-09-06 10:29:31 +02:00
Dave Simpson
2e78e96b75 [chore] Update Windows signing Cert to eToken (#2452) 2024-07-03 09:42:10 +02:00
Akos Kitta
aa9b10d68e fix: copy example with .pde main sketch file
Closes arduino/arduino-ide#2377

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-22 16:19:10 +01:00
Akos Kitta
4217c0001d fix: add missing installed version to the platform
Closes arduino/arduino-ide#2378

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-21 17:01:28 +01:00
Akos Kitta
a088ba99f5 fix: invalid custom board option handling in FQBN
Closes arduino/arduino-ide#1588

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-20 13:38:52 +01:00
Akos Kitta
2a325a5b74 feat: cancelable verify+upload
Closes arduino/arduino-ide#1199

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-20 13:38:52 +01:00
Akos Kitta
347e3d8118 fix: can unset network#proxy in the CLI config
An empty object (`{}`) must be used to correctly unset the CLI config
value to its default.

Closes arduino/arduino-ide#2184

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-20 13:38:52 +01:00
Akos Kitta
8e09971078 feat: use Arduino CLI 0.36.0-rc.1 APIs
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-20 13:38:52 +01:00
Akos Kitta
48e7bf6b5d chore: switch to version 2.3.3 after the release
To produce a correctly versioned nightly build.
See the [docs](1b9c7e93e0/docs/internal/release-procedure.md (7-%EF%B8%8F-bump-version-metadata-of-packages)) for more details.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-20 12:55:39 +01:00
Akos Kitta
95c4399c07 fix(ci): use go 1.21 for the on the fly bin builds
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-19 17:37:27 +01:00
Akos Kitta
4a807ab538 fix: no required programmer for debug --info
Ref: arduino/arduino-cli#2540
Closes: arduino/arduino-ide#2368

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-19 17:37:27 +01:00
Akos Kitta
1a98485b02 fix(security): use ip@2.0.1 for CVE-2023-42282
Refs:
 - https://github.com/advisories/GHSA-78xj-cgh5-2h22
  - 32f468f124
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-19 17:02:23 +01:00
github-actions[bot]
8fe6a81230 Updated translation files 2024-02-19 11:40:19 +01:00
Akos Kitta
547a630598 chore: switch to version 2.3.2 after the release
To produce a correctly versioned nightly build.
See the [docs](1b9c7e93e0/docs/internal/release-procedure.md (7-%EF%B8%8F-bump-version-metadata-of-packages)) for more details.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-15 10:27:29 +01:00
github-actions[bot]
ff8c646cfa Updated translation files 2024-02-14 13:35:47 +01:00
Akos Kitta
316e0fd8be fix(security): update all vulnerable dependencies
Resolutions:
 - `nano@^10.1.3`,
 - `msgpackr@^1.10.1`,
 - `axios@^1.6.7`

Fixes:
 - [GHSA-wf5p-g6vw-rhxx](https://github.com/advisories/GHSA-wf5p-g6vw-rhxx) (`CVE-2023-45857`)
 - [GHSA-jchw-25xp-jwwc](https://github.com/advisories/GHSA-jchw-25xp-jwwc) (`CVE-2023-26159`)
 - [GHSA-7hpj-7hhx-2fgx](https://github.com/advisories/GHSA-7hpj-7hhx-2fgx) (`CVE-2023-52079`)

Ref: nrwl/nx#20493
Ref: eclipse-theia/theia#13365

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-14 09:31:44 +01:00
Akos Kitta
ca779e5cf2 fix: debug widget layout updates
Use customized `PanelLayout#removeWidget` and
`PanelLayout#insertWidget` logic for the layout
updates. The customized functions ensure no side
effect if adding/removing the widget to/from the
layout but it's already present/absent.

Unlike the default [`PanelLayout#removeWidget`](9f5e11025b/packages/widgets/src/panellayout.ts (L154-L156))
behavior, do not try to remove the widget if it's
not present (has a negative index). Otherwise,
required widgets might be removed based on the
default [`ArrayExt#removeAt`](9f5e11025b/packages/algorithm/src/array.ts (L1075-L1077))
behavior.

Closes: arduino/arduino-ide#2354

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-08 14:17:10 +01:00
per1234
74c580175b Use arduino/setup-task@v1 action for Linux build job
Unfortunately the latest v2 version of the arduino/setup-task action used to install the Task task runner tool in the
runner machine has a dependency on a higher version of glibc than is provided by the Linux container. For this reason,
the workflow is configured to use arduino/setup-task@v1 for the Linux build job.

We will receive pull requests from Dependabot offering to update this outdated action dependency at each subsequent
major version release of the action (which are not terribly frequent). We must decline the bump of the action in that
specific step, but we can accept the bumps of all other usages of the action in the workflows. Dependabot remembers when
you decline a bump so this should not be too bothersome.
2024-02-07 20:40:49 -08:00
dependabot[bot]
71bd189eb1 build(deps): Bump arduino/setup-task from 1 to 2
Bumps [arduino/setup-task](https://github.com/arduino/setup-task) from 1 to 2.
- [Release notes](https://github.com/arduino/setup-task/releases)
- [Commits](https://github.com/arduino/setup-task/compare/v1...v2)

---
updated-dependencies:
- dependency-name: arduino/setup-task
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-02-07 20:40:49 -08:00
Akos Kitta
0822ed28da chore: switch to version 2.3.1 after the release
To produce a correctly versioned nightly build.
See the [docs](1b9c7e93e0/docs/internal/release-procedure.md (7-%EF%B8%8F-bump-version-metadata-of-packages)) for more details.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-07 19:05:11 +01:00
Akos Kitta
1b9c7e93e0 chore: use version 2.3.0 for the next release
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-07 13:09:34 +01:00
Akos Kitta
d419a6c6f0 feat: preference to limit thread count of the LS
Added a new preference (`arduino.language.asyncWorkers`) to control the
number of async workers used by `clangd`.
Users can fine tune the `clangd` thread count to overcome the excessive
CPU usage.

Use 0.1.2 Arduino Tools VSIX in IDE2.

Ref: arduino/arduino-language-server#177
Ref: arduino/vscode-arduino-tools#46

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-02-07 12:28:57 +01:00
github-actions[bot]
dda7770105 Updated translation files 2024-02-01 11:29:45 +01:00
Akos Kitta
763fde036c feat: disable debug widget if unsupported by board
Remove the 'Add configuration...' select option from the debug widget.

Closes #14

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-01-31 12:04:29 +01:00
per1234
0e7b0c9486 Bundle native Arduino Firmware Uploader with Apple Silicon build
A separate build of Arduino IDE is produced for each of the macOS host architectures:

* Apple Silicon (ARM)
* Intel (x86)

The Arduino IDE distribution includes several bundled helper tools. The build of those tools should also be selected according to the target host architecture.

At the time the infrastructure was set up for bundling the "Arduino Firmware Uploader" tool with the Arduino IDE distribution, that tool was only produced in a macOS x86 variant. So this was used even in the Apple Silicon Arduino IDE distribution, relying on Rosetta 2 to provide compatibility when used on Apple Silicon machines. Since that time, the Arduino Firmware Uploader build infrastructure has been updated to produce an Apple Silicon build of the tool and such a build is available for the IDE's current version dependency of Arduino Firmware Updater.

The "Arduino Firmware Uploader" tool bundling script is hereby updated to use the most appropriate variant of the dependency for the target host architecture. This provides the following benefits

* Compatibility with systems that don't have Rosetta 2 installed
* Improved performance
2024-01-22 02:32:45 -08:00
Akos Kitta
b8dd39c729 chore: pinned Arduino CLI 0.35.1
Closes arduino/arduino-ide#2318

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-01-17 17:30:25 +01:00
Akos Kitta
d6de53780d chore(dev): use latest settings for auto-format
From VS Code `>=1.85`

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-01-17 17:30:25 +01:00
per1234
2f2b19f613 Bump built-in example sketches version to 1.10.1
The Arduino IDE installation includes a collection of example sketches demonstrating fundamental concepts.

These examples are hosted in a dedicated repository, which is a dependency of this project. A new release has been made
in that `arduino/arduino-examples` repository.

The infrastructure for downloading the examples during the Arduino IDE build is hereby updated to use the latest release
of the `arduino/arduino-examples` repository.
2024-01-15 00:40:18 -08:00
Akos Kitta
0ca1a31747 fix: board <select> update on board detach
When the previously selected board is not detected, unset the `<select>`
option.

Closes #2222

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-01-15 09:25:10 +01:00
Akos Kitta
d01f95647e fix: sketch Save As errors on name collision
The Save As operation is halted and the problem clearly communicated to
the user when there is a collision between the target primary source
file name and existing secondary source files of the sketch.

Closes #827

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-01-15 09:24:50 +01:00
Akos Kitta
074f654457 fix: sketch Save As preserves the folder structure
This commit rewrites how IDE copies sketches as part of the _Save As_
operation. Instead of copying to the destination, IDE copies the sketch
into a temporary location, then to the desired destination.

This commit drops [`cpy`](https://www.npmjs.com/package/cpy).
Ref: 47b89a70b5

Closes #2077

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-01-15 09:24:50 +01:00
Akos Kitta
3eef857b48 fix: can open IDE from an ino file
Use a fallback frontend app config when starting IDE2 from an ino file
(from Explorer, Finder, etc.) and the app config is not yet set.

Closes #2209

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2024-01-15 09:16:11 +01:00
Akos Kitta
73b6dc4774 feat: use new debug -I -P CLI output
- Can pick a programmer if missing,
 - Can auto-select a programmer on app start,
 - Can edit the `launch.json`,
 - Adjust board discovery to new gRPC API. From now on, it's a client
 read stream, not a duplex.
 - Allow `.cxx` and `.cc` file extensions. (Closes #2265)
 - Drop `debuggingSupported` from `BoardDetails`.
 - Dedicated service endpoint for checking the debugger.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
Akos Kitta
42bf1a0e99 test: test Arduino state update for extensions
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
Akos Kitta
5abdc18fcc fix: make hosted plugin support testable
Hide the concrete implementation behind an interface so that tests can
`require` it.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
Akos Kitta
633346a3b0 feat: new window inherits the custom board options
A new startup task ensures setting any custom board menu selection in a
new sketch window.

Closes #2271

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
Akos Kitta
0f83a48649 chore(deps): update to electron@27.0.3
- Related change: 153e34f11b
 - Reported at: https://github.com/arduino/arduino-ide/pull/2267#issuecomment-1795180432
 - External: https://forum.arduino.cc/t/ide-2-2-1-main-window-randomly-goes-blank/1166219

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
Akos Kitta
a0bd5d022f chore: use 0.7.5 Arduino LS
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
Akos Kitta
101ba650f3 feat: handle v prefix in CLI GH release name
Ref: arduino/arduino-cli#2374
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
Akos Kitta
64ce35edbb feat: show in tooltip if core is from sketchbook
Closes #2270

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-12-13 17:32:07 +01:00
per1234
2dae4c8258 Remove version pin of Git dependency in Linux build container Dockerfile
A Docker container is used to produce the Linux build of Arduino IDE.

The dependencies of the build are pinned to a specific version in the Dockerfile in order to ensure a stable environment
in the images.

One such dependency is Git. The version of Git available from the package repository of the Ubuntu 18.04 distro used for
the image is too outdated to be used for this purpose. For this reason, a PPA is used as the source for the Git package.
Unfortunately the maintainers of the PPA remove the previous package every time they add a package for a newer version
of Git. This breaks the version pinned installation of the Git package:

5.515 E: Version '1:2.42.0-0ppa1~ubuntu18.04.1' for 'git' was not found

For this reason it is necessary to unpin Git in the Dockerfile. This will cause whichever version is available from the
PPA to be installed each time the image is built. Although not ideal for this application, that shouldn't cause any
problems in practice since Git has quite a stable and carefully maintained interface.
2023-12-10 23:36:50 -08:00
per1234
e7754b7c3b Use actions/setup-go@v4 for Linux build job
Unfortunately the latest v5 version of the actions/setup-go action used to set up the Go programming language in the
runner machine has a dependency on a higher version of glibc than is provided by the Linux container. For this reason,
the workflow is configured to use actions/setup-go@v4 for the Linux build job. We will receive pull requests from
Dependabot offering to update this outdated action dependency for at each subsequent major version release of the action
(which are not terribly frequent). We must decline the bump of the action in that specific step, but accept the bumps of
all other usages of the action in the workflows. Dependabot remembers when you decline a bump so this should not be too
bothersome.
2023-12-06 21:49:28 -08:00
dependabot[bot]
3d2511194a build(deps): Bump actions/setup-go from 4 to 5
Bumps [actions/setup-go](https://github.com/actions/setup-go) from 4 to 5.
- [Release notes](https://github.com/actions/setup-go/releases)
- [Commits](https://github.com/actions/setup-go/compare/v4...v5)

---
updated-dependencies:
- dependency-name: actions/setup-go
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2023-12-06 21:49:28 -08:00
dependabot[bot]
59a3c4faf0 build(deps): Bump actions/setup-python from 4 to 5
Bumps [actions/setup-python](https://github.com/actions/setup-python) from 4 to 5.
- [Release notes](https://github.com/actions/setup-python/releases)
- [Commits](https://github.com/actions/setup-python/compare/v4...v5)

---
updated-dependencies:
- dependency-name: actions/setup-python
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2023-12-06 20:35:14 -08:00
Akos Kitta
22a69f7488 chore(deps): update vulnerable dependencies
- Forced the resolution of `@babel/traverse@7.23.2` brought in by
`@theia/cli`. (eclipse-theia/theia#13024)
- Updated to `auth0-js@9.21.3` to transitively pull `crypto-js@4.2.0` in
with the security fixes.

GitHub Advisory Database refs:
 - https://github.com/advisories/GHSA-67hx-6x53-jw92
 - https://github.com/advisories/GHSA-xwcq-pm8m-c4vf

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-11-09 11:32:37 +01:00
dependabot[bot]
503533d712 build(deps): Bump actions/setup-node from 3 to 4
Bumps [actions/setup-node](https://github.com/actions/setup-node) from 3 to 4.
- [Release notes](https://github.com/actions/setup-node/releases)
- [Commits](https://github.com/actions/setup-node/compare/v3...v4)

---
updated-dependencies:
- dependency-name: actions/setup-node
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2023-10-23 14:09:12 -07:00
Akos Kitta
8687e2e044 chore(doc): update main screenshot in the readme
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-10-23 07:42:29 +02:00
per1234
69b73657b6 Simplify expression in container conditional steps of build workflow
The "Arduino IDE" GitHub Actions workflow uses a Docker container for the build job of certain target, while running others directly in the
runner environment.

Due to differences between these two environments, some steps must run only when a container is used by the job, and others only when the job is running in the runner environment. This is done by a conditional on the job matrix data regarding the container. The container value is set to null for the jobs that run in the runner environment, while containing a full container configuration mapping for the jobs that run in a container.

Previously the conditional unnecessarily used the value of the image key of the container object specifically. The comparison
can be done against the container value itself. Doing this will make it a little easier to understand the workflow code, since the conditional more clearly reflects the matrix (where `container` is set to `null` instead of `container.image`).
2023-10-19 06:40:46 -07:00
per1234
7e8f723df3 Pin Python version at 3.11 in build workflow
Python 3.12.x is incompatible with the current version of node-gyp (9.4.0). For this reason, it is necessary to
configure the "GitHub Actions" workflow to use the compatible Python 3.11.x until the next release of node-gyp (which
should contain a fix for the breakage) is made.
2023-10-19 06:40:46 -07:00
per1234
d19778d0fb Remove obviated build workflow step for removing Linux channel file
In addition to the builds, when a nightly or production release is published, a "channel update info file" is also
uploaded to Amazon S3 by the "Arduino IDE" GitHub Actions workflow. The IDE checks the channel file on the server to get
information about available updates.

Previously the Linux production release builds were being remade manually due to the ones produced by GitHub Actions not
being compatible with older distro versions due to a dynamically linked dependency. For this reason, a step was
temporarily added to the workflow to cause it to not upload the Linux channel file in order to avoid Linux users from
receiving an update offer before the limited compatibility automated build had been replaced with the manually produced
build.

The automated build system has been adjusted to produce Linux builds with the intended range of compatibility, but the
step that deleted the channel file was not removed at that time. The obviated step is hereby removed in order to allow
complete releases to be published by the workflow.
2023-10-19 06:40:46 -07:00
per1234
e5ef564817 Correct conditional logic for publishing steps of build workflow
The "Arduino IDE" GitHub Actions workflow is used to generate several distinct types of builds:

- Tester builds of commits
- Nightly builds
- Release builds

Different actions must be performed depending on which type of build is being produced. The workflow uses dedicated jobs
for publishing the nightly builds and for publishing the release builds. Those jobs are configured to run only when
certain criteria are met.

One such criteria is that the merge-channel-files job ran as expected. There are four possible result types of a job,
which should be handled as follows:

| Result   | Run dependent job? |
| -------- | ------------------ |
| success  | Yes                |
| failure  | No                 |
| canceled | No                 |
| skipped  | Yes                |

GitHub Actions automatically takes the desired action regarding whether the dependent job should run for the first three
result types, but that is not the case for the "skipped" result. The merge-channel-files job dependency is skipped when
a channel file merge is not needed and so this is not cause to cancel the build publishing.

The only way to make a dependent job run when the dependency was skipped is to add the `always()` expression to the job
conditional. This goes too far in the other direction by causing the job to run even when the dependency failed or was
canceled. So it is necessary to also add logic for each of the dependency job result types to the conditional, which
makes it quite complex when combined with the logic for the other criteria of the job.

In order to reduce the amount of complexity of the conditionals of the dependent jobs, a job was interposed in the
dependency chain, which was intended to act simply as a container for the logic about the merge-channel-files job
result. Unfortunately it turns out that even if the direct dependency job's result was success, if any ancestor in the
dependency chain was skipped, GitHub Actions still skips all dependent jobs without an `always()` expression in their
conditional, meaning the intermediate job was pointless. This caused the build publishing jobs to be skipped under the
conditions where they should have ran.

The pointless intermediate job is hereby removed, an `always()` expression added to the conditionals of the dependent
jobs, and the full logic for how to handle each dependent job result type added there as well.
2023-10-19 06:40:46 -07:00
Lukas
ce01351bfe Update README.md
Co-authored-by: per1234 <accounts@perglass.com>
2023-10-16 01:44:26 -07:00
Lukas
a110c2b1f2 adding a hyperlink to the README 2023-10-16 01:44:26 -07:00
Akos Kitta
153e34f11b chore(deps): update dependencies
To fix all security vulnerabilities detected by `Dependabot`.

 - remove `shelljs`. replace with `fs` and `console`.
 - remove `uuid`. replace with `@phosphor/coreutils`.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-10-13 08:50:39 +02:00
per1234
ed1cb6bcf9 Make Linux build in a container for compatibility with older distros
Background
----------

The Linux build of Arduino IDE is dynamically linked against the libstdc++ and glibc shared libraries. This results in
it having a dependency on the version of the libraries that happens to be present in the environment it is built in.

Although newer versions of the shared libraries are compatible with executables linked against an older version, the
reverse is not true. This means that building Arduino IDE on a Linux machine with a recent distro version installed
causes the IDE to error on startup for users who have a distro with older versions of the dependencies. For example:

```
Error: /usr/lib/x86_64-linux-gnu/libstdc++.so.6: version `GLIBCXX_3.4.26' not found (required by /home/per/Downloads/arduino-ide_nightly-20231006_Linux_64bit/resources/app/lib/backend/native/nsfw.node)
```

or:

```
Error: /lib/x86_64-linux-gnu/libc.so.6: version `GLIBC_2.33' not found (required by /home/per/Downloads/arduino-ide_2.0.5-snapshot-90b1f67_Linux_64bit/resources/app/node_modules/nsfw/build/Release/nsfw.node)
```

We were originally able to achieve our targeted range of Linux distro compatibility by running the Linux build in the
ubuntu-18.04 GitHub Actions hosted runner machine. Unfortunately GitHub stopped offering that runner machine. This meant
we were forced to update to using the ubuntu-20.04 runner machine instead, which caused the loss of compatibility of
the automatically generated builds with previously supported distro versions (e.g., Ubuntu 18.04). Since that time, the
release builds have been produced manually, which is inefficient and prone to human error.

Update
------

The identified solution to restoring fully automated builds with the target range of Linux distro compatibility is to
run the build in a Docker container that provides the suitable environment. This means a combination of an older distro
version with the modern versions of the development tool dependencies of the build.

Such a combination was achieved by creating a bespoke image based on the ubuntu:18.04 image. The Dockerfile is hosted in
the Arduino IDE repository in order to allow it to be maintained in parallel with the code and infrastructure.

Image Publishing
----------------

A "Push Container Images" GitHub Actions continuous delivery workflow is added to push updated images to the GitHub
Container registry when a commit that modifies relevant files is pushed to the main branch.

This means the image does not have formally versioned tags and the IDE build uses the container that results from the
configuration at the tip of the main branch at the time of the build. I think that is a reasonable approach in this use
case where the image is targeted to a single application rather than intended to be used by multiple projects.

Container Validation
--------------------

The build workflow is configured to trigger on completion of that push workflow in order to provide validation of the IDE build using the
updated container as well as the resulting tester builds of the IDE.

A "Check Containers" GitHub Actions continuous integration workflow is added to provide basic validation for changes to
the Dockerfile. It will automatically build the image and run the container on any push or pull request that modifies
relevant files.

Container Workflow Design
-------------------------

With the goal of reusability, the image data is contained in a job matrix in the workflow to allow them to accommodate
any number of arbitrary images.

Build Workflow Configuration
----------------------------

A container property is added to the build job matrix data. If the container.image property is set to null, GitHub
Actions will run the job directly in the runner environment instead of in a container. This allows us to produce the
builds using either a container or a bare runner machine as is appropriate for each target.

Unfortunately the latest v4 version of the actions/checkout action used to checkout the repository into the job
environment has a dependency on a higher version of glibc than is provided by the Linux container. For this reason, the
workflow is configured to use actions/checkout@v3 for the Linux build job. We will likely receive pull requests from
Dependabot offering to update this outdated action dependency for the v4 and at each subsequent major version release of
the action (which are not terribly frequent). We must decline the bump of the action in that specific step, but accept
the bumps of all other usages of the action in the workflows. Dependabot remembers when you decline a bump so this
should not be too bothersome.
2023-10-12 11:24:29 -07:00
Akos Kitta
a8e63c8c90 test: relax accessible sketch path test condition
- Do not expect `EACCESS` on Linux.
 - Run the test only if `EACCESS` occurs via the `stat` syscall.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-10-09 19:24:51 +02:00
Akos Kitta
0b2410d49a test: run cloud sketches tests on the CI (#2092)
- fix(test): integration tests are more resilient.
   - run the Create integration tests with other slow tests,
   - queued `PUT`/`DELETE` requests to make the test timeout happy,
   - reduced the `/sketches/search` offset to 1/5th and
   - remove Create API logging.
 - fix(linter): ignore `lib` folder. Remove obsolete `.node_modules`
pattern.
 - feat(ci): enable Create API integration tests.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
Co-authored-by: per1234 <accounts@perglass.com>
2023-10-07 10:38:54 +02:00
per1234
3f4d2745a8 Use a generalized criterion for S3 publishing determination in build workflow
The "Arduino IDE" GitHub Actions workflow uploads the nightly and release builds to Amazon S3, from which they are
downloaded by the auto-update as well as directly by users via the links on the "Software" page of arduino.cc.

The workflow can also be useful in forks. Either by those who want to test contributions staged in their fork prior to
submitting a PR to the parent repo, or by those maintaining a hard fork of the project. Even though these forks wouldn't
(and couldn't due to lack of access to the encrypted credential secrets only available to the workflow when ran in a
trusted context in Arduino's repo)credentials stored in Arduino's repo) use the S3 upload component of the workflow,
they may still find it valuable for continuous integration as well as continuous deployment via the tester builds and
release builds the workflow also publishes to the GitHub repository it runs in. For this reason, the workflow contains
code to determine whether it should attempt the S3 uploads.

Previously the repository name was used as the criteria in that code. The project specificity of that approach makes the
workflow less easily reusable. A more generally applicable criterion is whether the encrypted credential certificate is
defined.

The new criterion allows the workflow to be used in any repository where the administrator has created an encrypted
secret containing their AWS credentials. That might be other projects owned by Arduino, or even 3rd party projects where
the owners want to take a similar build publishing approach using their own AWS account.
2023-10-06 09:50:52 -07:00
per1234
136545491d Deduplicate S3 publishing determination code in build workflow
The "Arduino IDE" GitHub Actions workflow uploads the nightly and release builds to Amazon S3, from which they are
downloaded by the auto-update as well as directly by users via the links on the "Software" page of arduino.cc.

The workflow can also be useful in forks. Either by those who want to test contributions staged in their fork prior to
submitting a PR to the parent repo, or by those maintaining a hard fork of the project. Even though these forks wouldn't
(and couldn't due to lack of access to the encrypted credential secrets only available to the workflow when ran in a
trusted context in Arduino's repo)credentials stored in Arduino's repo) use the S3 upload component of the workflow,
they may still find it valuable for continuous integration as well as continuous deployment via the tester builds and
release builds the workflow also publishes to the GitHub repository it runs in. For this reason, the workflow contains
code to determine whether it should attempt the S3 uploads. Previously that code was duplicated in both the nightly and
release publishing jobs of the workflow. Since the workflow already contains a job specifically for the purpose of
determining the characteristics of the build being performed and making that information available from single source
for use throughout the rest of the workflow, it makes sense to also move the S3 upload determination code to that job.
2023-10-06 09:50:52 -07:00
per1234
7e1d441e6a Add native Apple Silicon target to build workflow
On every release tag, and manual trigger when the "Include builds on non-free runners" checkbox is checked, make the
Arduino IDE build for native Apple Silicon host in addition to the builds that are always generated by the "Arduino IDE"
GitHub Actions workflow.

Previously, the build workflow only produced a build for x86-64 (AKA "Intel") macOS hosts. Although it is possible to
use those builds on Apple Silicon machines via the Rosetta 2 translation software, the performance is significantly
inferior to a native build so we must also provide Apple Silicon native builds.

Previously the Apple Silicon builds were produced manually. The reason for using that inefficient and error-prone
approach instead of the automated continuous deployment system used for other builds was that GitHub did not provide the
necessary Apple Silicon runner machines and Arduino was not capable of setting up such self-hosted machines in a manner
that would make them feasible for the project maintainers to use. GitHub hosted Apple Silicon runner machines are now
available so we can add the target to the build workflow.

GitHub gives unlimited use of the basic runner machines for workflow runs in public repositories. However, the macOS ARM
architecture is only provided in runner machines which are classified as "larger runner". Use of these runners is
charged on a per-minute basis, without any of the free allowances GitHub provides for the normal runners. In order to
avoid unnecessary expenditures, native Apple Silicon builds must be generated only when there is compelling reason to do
so. Such a build is needed for every release, so the workflow is configured to always generate the builds when triggered
by a tag. In addition to releases, Apple Silicon tester builds for pull requests that might have special implications
for this target. For this reason, the workflow is configured to allow Apple Silicon builds to be triggered manually by a
repository maintainer.

The workflow uses a job matrix to run the build for each target on the appropriate runner machine in parallel, using the
universally applicable workflow code for all jobs. It uses another job matrix to generate individual workflow artifacts
for the tester builds of each target. Previously it was possible to always use the same matrix configurations for all
workflow runs. With the addition of the selectively run macOS ARM job, it is now necessary to generate these matrixes on
the fly.

The electron-updater package used by Arduino IDE's auto-update capability uses a data file (known as the "channel update
info file") to check for the availability of updates. A single "channel update info file" is used for the data of the
macOS x86 and ARM builds. Since a separate job is used to produce each of those builds, this means the "channel update
info file" produced by each of the macOS build jobs must be merged into a single file.
2023-10-06 09:50:52 -07:00
per1234
f0706e1849 Deduplicate type determination code in build workflow
The "Arduino IDE" GitHub Actions workflow is used to generate several distinct types of builds:

- Tester builds of commits
- Nightly builds
- Release builds

Different actions must be performed depending on which type of build is being produced. The workflow contains code that
uses various criteria to determine the build type.

Previously that code was duplicated in multiple places:

- The packaging job
- The changelog generation job
- The nightly build publishing job
- The release publishing job

This duplication is avoided by moving the code to a dedicated job that makes the build type information available to all
subsequent jobs via outputs.
2023-10-06 09:50:52 -07:00
per1234
4708bae9ab Use separate attributes for human identifier and runner identifier in build job matrix
The "Arduino IDE" GitHub Actions workflow uses a job matrix to make the builds for each host target in parallel.

The same steps are used for each job, but some configuration adjustments must be made on a per-target basis. This is
done through various attributes in the matrix configuration.

Previously the `os` attribute was used for two distinct things:

- The machine identifier of the GitHub Actions runner machine of the job.
- The differentiator in the human-targeted job name.

The attribute name "os" (for "operating system") was misleading because runners are differentiated by more than only the
operating system on the machine.

The use of a machine identifier as a differentiator in the human-targeted job name was a bad idea because these
identifiers would only effectively communicate the nature of a build to humans who are quite knowledgeable about the
GitHub Actions workflow syntax.

The impact of these poor decisions has not been too severe previously due to there only being a single job for each
operating system. However, there is a need for multiple jobs per operating system in order to support multiple host
architectures (e.g., macOS x86 and ARM).

The solution is to:

- Use an appropriate name for the runner identifier attribute.
- Use a dedicated attribute for the human friendly job name differentiator.
2023-10-06 09:50:52 -07:00
Akos Kitta
57975f8d91 fix: use board+port at startup if it's restored (#2242)
- update status bar if board+port is restored,
 - refresh the debug toolbar if board+port is restored,
 - init `Include Library` if board+port is ready, and
 - init library examples if board+port is ready

Closes #2237
Closes #2239

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-10-05 14:41:14 +02:00
Akos Kitta
8f4bcc83ec chore(deps): update to theia@1.41.0 (#2211)
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-29 18:40:13 +02:00
Akos Kitta
ce02e263ec fix: storage service injection
Store the board config data per sketch and not per IDE installation.

The (Theia) default `StorageService` implementation is workspace-scoped
when `@theia/workspace` is part of the final application.

Closes #2240

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-27 17:34:24 +02:00
Akos Kitta
ed2d8ad13c chore(deps): update electron@25.5.0
- Update to `electron-builder@24.6.3`.
 - Fix obsolete electron security section in the development docs.

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-26 10:06:47 +02:00
Akos Kitta
bb4b1450e3 chore: use Node.js >=18.17.0
remove `msvs_version` npm config on win32
Ref: nodejs/node-gyp#2822

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-26 10:06:47 +02:00
Akos Kitta
8a5dee9307 chore: format resources 💄
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-26 10:06:47 +02:00
Akos Kitta
5939b65511 chore: update prettier config
narrow the `resources` folder constraint in
the .gitignore

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-26 10:06:47 +02:00
Akos Kitta
7f660d76a8 fix: refresh the user-fields at app startup
Ref: arduino/arduino-ide#2165
Closes arduino/arduino-ide#2230

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-26 09:57:02 +02:00
Akos Kitta
9f48296d4d fix: defer board+port state update for extensions
If it is set before the board+port settings are restored from the
`localStorage`, extensions will see no board+port.

Ref: arduino/arduino-ide#2165
Ref: dankeboy36/esp-exception-decoder#10

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-26 09:56:34 +02:00
Akos Kitta
ec28623a97 fix: forward backend logging to electron (#2236)
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-26 09:45:03 +02:00
dependabot[bot]
73ddbefc3e build(deps): Bump actions/checkout from 3 to 4
Bumps [actions/checkout](https://github.com/actions/checkout) from 3 to 4.
- [Release notes](https://github.com/actions/checkout/releases)
- [Changelog](https://github.com/actions/checkout/blob/main/CHANGELOG.md)
- [Commits](https://github.com/actions/checkout/compare/v3...v4)

---
updated-dependencies:
- dependency-name: actions/checkout
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
2023-09-05 00:27:27 -07:00
Akos Kitta
fe53b8e0d0 chore: update version to 2.2.2
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-09-01 08:53:21 +02:00
Akos Kitta
90b886ea92 chore: update to arduino-fwuploader@2.4.1
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-08-31 12:50:11 +02:00
Akos Kitta
97e26a9584 fix(ci): fix the changelog generation
Pinned `@octokit/rest` to `19.0.13`, so that it works with Node.js 16+.

Closes arduino/arduino-ide#2200

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-08-31 11:42:43 +02:00
Akos Kitta
f0704b678c fix(i18n): corrected the module for tr and ru
Closes arduino/arduino-ide#2201

Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-08-31 08:48:05 +02:00
Akos Kitta
95bd2cf2e7 chore: update version to 2.2.1
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-08-29 17:12:17 +02:00
Akos Kitta
8d2808893c fix: name of translation (UA) module
Signed-off-by: Akos Kitta <a.kitta@arduino.cc>
2023-08-29 08:21:11 +02:00
github-actions[bot]
e5b5b2a4be Updated translation files 2023-08-28 11:21:11 +02:00
255 changed files with 32356 additions and 20218 deletions

View File

@@ -1,66 +1,66 @@
module.exports = { module.exports = {
parser: '@typescript-eslint/parser', // Specifies the ESLint parser parser: '@typescript-eslint/parser', // Specifies the ESLint parser
parserOptions: { parserOptions: {
ecmaVersion: 2020, // Allows for the parsing of modern ECMAScript features ecmaVersion: 2020, // Allows for the parsing of modern ECMAScript features
sourceType: 'module', // Allows for the use of imports sourceType: 'module', // Allows for the use of imports
ecmaFeatures: { ecmaFeatures: {
jsx: true, // Allows for the parsing of JSX jsx: true, // Allows for the parsing of JSX
},
}, },
ignorePatterns: [ },
'node_modules/*', ignorePatterns: [
'**/node_modules/*', 'node_modules/*',
'.node_modules/*', '**/node_modules/*',
'.github/*', '.github/*',
'.browser_modules/*', '.browser_modules/*',
'docs/*', 'docs/*',
'scripts/*', 'scripts/*',
'electron-app/lib/*', 'electron-app/lib/*',
'electron-app/src-gen/*', 'electron-app/src-gen/*',
'electron-app/gen-webpack*.js', 'electron-app/gen-webpack*.js',
'!electron-app/webpack.config.js', '!electron-app/webpack.config.js',
'plugins/*', 'electron-app/plugins/*',
'arduino-ide-extension/src/node/cli-protocol', 'arduino-ide-extension/src/node/cli-protocol',
'**/lib/*',
],
settings: {
react: {
version: 'detect', // Tells eslint-plugin-react to automatically detect the version of React to use
},
},
extends: [
'plugin:@typescript-eslint/recommended', // Uses the recommended rules from the @typescript-eslint/eslint-plugin
'plugin:react/recommended', // Uses the recommended rules from @eslint-plugin-react
'plugin:react-hooks/recommended', // Uses recommended rules from react hooks
'plugin:prettier/recommended',
'prettier', // Uses eslint-config-prettier to disable ESLint rules from @typescript-eslint/eslint-plugin that would conflict with prettier
],
plugins: ['prettier', 'unused-imports'],
rules: {
'@typescript-eslint/no-unused-expressions': 'off',
'@typescript-eslint/no-namespace': 'off',
'@typescript-eslint/no-var-requires': 'off',
'@typescript-eslint/no-empty-function': 'warn',
'@typescript-eslint/no-empty-interface': 'warn',
'no-unused-vars': 'off',
'unused-imports/no-unused-imports': 'error',
'unused-imports/no-unused-vars': [
'warn',
{
vars: 'all',
varsIgnorePattern: '^_',
args: 'after-used',
argsIgnorePattern: '^_',
},
], ],
settings: { 'react/display-name': 'warn',
react: { eqeqeq: ['error', 'smart'],
version: 'detect', // Tells eslint-plugin-react to automatically detect the version of React to use 'guard-for-in': 'off',
}, 'id-blacklist': 'off',
}, 'id-match': 'off',
extends: [ 'no-underscore-dangle': 'off',
'plugin:@typescript-eslint/recommended', // Uses the recommended rules from the @typescript-eslint/eslint-plugin 'no-unused-expressions': 'off',
'plugin:react/recommended', // Uses the recommended rules from @eslint-plugin-react 'no-var': 'error',
'plugin:react-hooks/recommended', // Uses recommended rules from react hooks radix: 'error',
'plugin:prettier/recommended', 'prettier/prettier': 'warn',
'prettier', // Uses eslint-config-prettier to disable ESLint rules from @typescript-eslint/eslint-plugin that would conflict with prettier },
],
plugins: ['prettier', 'unused-imports'],
rules: {
'@typescript-eslint/no-unused-expressions': 'off',
'@typescript-eslint/no-namespace': 'off',
'@typescript-eslint/no-var-requires': 'off',
'@typescript-eslint/no-empty-function': 'warn',
'@typescript-eslint/no-empty-interface': 'warn',
'no-unused-vars': 'off',
'unused-imports/no-unused-imports': 'error',
'unused-imports/no-unused-vars': [
'warn',
{
vars: 'all',
varsIgnorePattern: '^_',
args: 'after-used',
argsIgnorePattern: '^_',
},
],
'react/display-name': 'warn',
eqeqeq: ['error', 'smart'],
'guard-for-in': 'off',
'id-blacklist': 'off',
'id-match': 'off',
'no-underscore-dangle': 'off',
'no-unused-expressions': 'off',
'no-var': 'error',
radix: 'error',
'prettier/prettier': 'warn',
},
}; };

View File

@@ -1,7 +1,7 @@
name: Bug report name: Bug report
description: Report a problem with the code or documentation in this repository. description: Report a problem with the code or documentation in this repository.
labels: labels:
- "type: imperfection" - 'type: imperfection'
body: body:
- type: textarea - type: textarea
id: description id: description

View File

@@ -1,7 +1,7 @@
name: Feature request name: Feature request
description: Suggest an enhancement to this project. description: Suggest an enhancement to this project.
labels: labels:
- "type: enhancement" - 'type: enhancement'
body: body:
- type: textarea - type: textarea
id: description id: description

View File

@@ -1,15 +1,18 @@
### Motivation ### Motivation
<!-- Why this pull request? --> <!-- Why this pull request? -->
### Change description ### Change description
<!-- What does your code do? --> <!-- What does your code do? -->
### Other information ### Other information
<!-- Any additional information that could help the review process --> <!-- Any additional information that could help the review process -->
### Reviewer checklist ### Reviewer checklist
* [ ] PR addresses a single concern. - [ ] PR addresses a single concern.
* [ ] The PR has no duplicates (please search among the [Pull Requests](https://github.com/arduino/arduino-ide/pulls) before creating one) - [ ] The PR has no duplicates (please search among the [Pull Requests](https://github.com/arduino/arduino-ide/pulls) before creating one)
* [ ] PR title and description are properly filled. - [ ] PR title and description are properly filled.
* [ ] Docs have been added / updated (for bug fixes / features) - [ ] Docs have been added / updated (for bug fixes / features)

View File

@@ -12,4 +12,4 @@ updates:
schedule: schedule:
interval: daily interval: daily
labels: labels:
- "topic: infrastructure" - 'topic: infrastructure'

View File

@@ -1,27 +1,27 @@
# Used by the "Sync Labels" workflow # Used by the "Sync Labels" workflow
# See: https://github.com/Financial-Times/github-label-sync#label-config-file # See: https://github.com/Financial-Times/github-label-sync#label-config-file
- name: "topic: accessibility" - name: 'topic: accessibility'
color: "00ffff" color: '00ffff'
description: Enabling the use of the software by everyone description: Enabling the use of the software by everyone
- name: "topic: CLI" - name: 'topic: CLI'
color: "00ffff" color: '00ffff'
description: Related to Arduino CLI description: Related to Arduino CLI
- name: "topic: cloud" - name: 'topic: cloud'
color: "00ffff" color: '00ffff'
description: Related to Arduino Cloud and cloud sketches description: Related to Arduino Cloud and cloud sketches
- name: "topic: debugger" - name: 'topic: debugger'
color: "00ffff" color: '00ffff'
description: Related to the integrated debugger description: Related to the integrated debugger
- name: "topic: language server" - name: 'topic: language server'
color: "00ffff" color: '00ffff'
description: Related to the Arduino Language Server description: Related to the Arduino Language Server
- name: "topic: serial monitor" - name: 'topic: serial monitor'
color: "00ffff" color: '00ffff'
description: Related to the Serial Monitor description: Related to the Serial Monitor
- name: "topic: theia" - name: 'topic: theia'
color: "00ffff" color: '00ffff'
description: Related to the Theia IDE framework description: Related to the Theia IDE framework
- name: "topic: theme" - name: 'topic: theme'
color: "00ffff" color: '00ffff'
description: Related to GUI theming description: Related to GUI theming

View File

@@ -0,0 +1,112 @@
# The Arduino IDE Linux build workflow job runs in this container.
# syntax=docker/dockerfile:1
FROM ubuntu:18.04
# See: https://unofficial-builds.nodejs.org/download/release/
ARG node_version="18.17.1"
RUN \
apt-get \
--yes \
update
# This is required to get add-apt-repository
RUN \
apt-get \
--yes \
install \
"software-properties-common=0.96.24.32.22"
# Install Git
# The PPA is required to get a modern version of Git. The version in the Ubuntu 18.04 package repository is 2.17.1,
# while action/checkout@v3 requires 2.18 or higher.
RUN \
add-apt-repository \
--yes \
"ppa:git-core/ppa" && \
apt-get \
--yes \
update && \
\
apt-get \
--yes \
install \
"git" && \
\
apt-get \
--yes \
purge \
"software-properties-common"
# The repository path must be added to safe.directory, otherwise any Git operations on it would fail with a
# "dubious ownership" error. actions/checkout configures this, but it is not applied to containers.
RUN \
git config \
--add \
--global \
"safe.directory" "/__w/arduino-ide/arduino-ide"
ENV \
GIT_CONFIG_GLOBAL="/root/.gitconfig"
# Install Python
# The Python installed by actions/setup-python has dependency on a higher version of glibc than available in the
# ubuntu:18.04 container.
RUN \
apt-get \
--yes \
install \
"python3.8-minimal=3.8.0-3ubuntu1~18.04.2" && \
\
ln \
--symbolic \
--force \
"$(which python3.8)" \
"/usr/bin/python3"
# Install Theia's package dependencies
# These are pre-installed in the GitHub Actions hosted runner machines.
RUN \
apt-get \
--yes \
install \
"libsecret-1-dev=0.18.6-1" \
"libx11-dev=2:1.6.4-3ubuntu0.4" \
"libxkbfile-dev=1:1.0.9-2"
# Install Node.js
# It is necessary to use the "unofficial" linux-x64-glibc-217 build because the official Node.js 18.x is dynamically
# linked against glibc 2.28, while Ubuntu 18.04 has glibc 2.27.
ARG node_installation_path="/tmp/node-installation"
ARG artifact_name="node-v${node_version}-linux-x64-glibc-217"
RUN \
mkdir "$node_installation_path" && \
cd "$node_installation_path" && \
\
apt-get \
--yes \
install \
"wget=1.19.4-1ubuntu2.2" && \
\
archive_name="${artifact_name}.tar.xz" && \
wget \
"https://unofficial-builds.nodejs.org/download/release/v${node_version}/${archive_name}" && \
\
apt-get \
--yes \
purge \
"wget" && \
\
tar \
--file="$archive_name" \
--extract && \
rm "$archive_name"
ENV PATH="${PATH}:${node_installation_path}/${artifact_name}/bin"
# Install Yarn
# Yarn is pre-installed in the GitHub Actions hosted runner machines.
RUN \
npm \
install \
--global \
"yarn@1.22.19"

View File

@@ -12,11 +12,17 @@ on:
- '.vscode/**' - '.vscode/**'
- 'docs/**' - 'docs/**'
- 'scripts/**' - 'scripts/**'
- '!scripts/merge-channel-files.js'
- 'static/**' - 'static/**'
- '*.md' - '*.md'
tags: tags:
- '[0-9]+.[0-9]+.[0-9]+*' - '[0-9]+.[0-9]+.[0-9]+*'
workflow_dispatch: workflow_dispatch:
inputs:
paid-runners:
description: Include builds on non-free runners
type: boolean
default: false
pull_request: pull_request:
paths-ignore: paths-ignore:
- '.github/**' - '.github/**'
@@ -24,16 +30,103 @@ on:
- '.vscode/**' - '.vscode/**'
- 'docs/**' - 'docs/**'
- 'scripts/**' - 'scripts/**'
- '!scripts/merge-channel-files.js'
- 'static/**' - 'static/**'
- '*.md' - '*.md'
schedule: schedule:
- cron: '0 3 * * *' # run every day at 3AM (https://docs.github.com/en/actions/reference/events-that-trigger-workflows#scheduled-events-schedule) - cron: '0 3 * * *' # run every day at 3AM (https://docs.github.com/en/actions/reference/events-that-trigger-workflows#scheduled-events-schedule)
workflow_run:
workflows:
- Push Container Images
branches:
- main
types:
- completed
env: env:
# See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml # See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml
GO_VERSION: "1.19" GO_VERSION: '1.21'
# See: https://github.com/actions/setup-node/#readme
NODE_VERSION: '18.17'
JOB_TRANSFER_ARTIFACT: build-artifacts JOB_TRANSFER_ARTIFACT: build-artifacts
CHANGELOG_ARTIFACTS: changelog CHANGELOG_ARTIFACTS: changelog
STAGED_CHANNEL_FILES_ARTIFACT: staged-channel-files
BASE_BUILD_DATA: |
- config:
# Human identifier for the job.
name: Windows
runs-on: [self-hosted, windows-sign-pc]
# The value is a string representing a JSON document.
# Setting this to null causes the job to run directly in the runner machine instead of in a container.
container: |
null
# Name of the secret that contains the certificate.
certificate-secret: INSTALLER_CERT_WINDOWS_CER
# Name of the secret that contains the certificate password.
certificate-password-secret: INSTALLER_CERT_WINDOWS_PASSWORD
# File extension for the certificate.
certificate-extension: pfx
# Container for windows cert signing
certificate-container: INSTALLER_CERT_WINDOWS_CONTAINER
# Quoting on the value is required here to allow the same comparison expression syntax to be used for this
# and the companion needs.select-targets.outputs.merge-channel-files property (output values always have string
# type).
mergeable-channel-file: 'false'
# as this runs on a self hosted runner, we need to avoid building with the default working directory path,
# otherwise paths in the build job will be too long for `light.exe`
# we use the below as a Symbolic link (just changing the wd will break the checkout action)
# this is a work around (see: https://github.com/actions/checkout/issues/197).
working-directory: 'C:\a'
artifacts:
- path: '*Windows_64bit.exe'
name: Windows_X86-64_interactive_installer
- path: '*Windows_64bit.msi'
name: Windows_X86-64_MSI
- path: '*Windows_64bit.zip'
name: Windows_X86-64_zip
- config:
name: Linux
runs-on: ubuntu-latest
container: |
{
\"image\": \"ghcr.io/arduino/arduino-ide/linux:main\"
}
mergeable-channel-file: 'false'
artifacts:
- path: '*Linux_64bit.zip'
name: Linux_X86-64_zip
- path: '*Linux_64bit.AppImage'
name: Linux_X86-64_app_image
- config:
name: macOS x86
runs-on: macos-13
container: |
null
# APPLE_SIGNING_CERTIFICATE_P12 secret was produced by following the procedure from:
# https://www.kencochrane.com/2020/08/01/build-and-sign-golang-binaries-for-macos-with-github-actions/#exporting-the-developer-certificate
certificate-secret: APPLE_SIGNING_CERTIFICATE_P12
certificate-password-secret: KEYCHAIN_PASSWORD
certificate-extension: p12
mergeable-channel-file: 'true'
artifacts:
- path: '*macOS_64bit.dmg'
name: macOS_X86-64_dmg
- path: '*macOS_64bit.zip'
name: macOS_X86-64_zip
- config:
name: macOS ARM
runs-on: macos-latest
container: |
null
certificate-secret: APPLE_SIGNING_CERTIFICATE_P12
certificate-password-secret: KEYCHAIN_PASSWORD
certificate-extension: p12
mergeable-channel-file: 'true'
artifacts:
- path: '*macOS_arm64.dmg'
name: macOS_arm64_dmg
- path: '*macOS_arm64.zip'
name: macOS_arm64_zip
jobs: jobs:
run-determination: run-determination:
@@ -60,56 +153,210 @@ jobs:
echo "result=$RESULT" >> $GITHUB_OUTPUT echo "result=$RESULT" >> $GITHUB_OUTPUT
build: build-type-determination:
name: build (${{ matrix.config.os }})
needs: run-determination needs: run-determination
if: needs.run-determination.outputs.result == 'true' if: needs.run-determination.outputs.result == 'true'
runs-on: ubuntu-latest
outputs:
is-release: ${{ steps.determination.outputs.is-release }}
is-nightly: ${{ steps.determination.outputs.is-nightly }}
channel-name: ${{ steps.determination.outputs.channel-name }}
publish-to-s3: ${{ steps.determination.outputs.publish-to-s3 }}
permissions: {}
steps:
- name: Determine the type of build
id: determination
run: |
if [[
"${{ startsWith(github.ref, 'refs/tags/') }}" == "true"
]]; then
is_release="true"
is_nightly="false"
channel_name="stable"
elif [[
"${{ github.event_name }}" == "schedule" ||
(
"${{ github.event_name }}" == "workflow_dispatch" &&
"${{ github.ref }}" == "refs/heads/main"
)
]]; then
is_release="false"
is_nightly="true"
channel_name="nightly"
else
is_release="false"
is_nightly="false"
channel_name="nightly"
fi
echo "is-release=$is_release" >> $GITHUB_OUTPUT
echo "is-nightly=$is_nightly" >> $GITHUB_OUTPUT
echo "channel-name=$channel_name" >> $GITHUB_OUTPUT
# Only attempt upload to Amazon S3 if the credentials are available.
echo "publish-to-s3=${{ secrets.AWS_SECRET_ACCESS_KEY != '' }}" >> $GITHUB_OUTPUT
select-targets:
needs: build-type-determination
runs-on: ubuntu-latest
outputs:
artifact-matrix: ${{ steps.assemble.outputs.artifact-matrix }}
build-matrix: ${{ steps.assemble.outputs.build-matrix }}
merge-channel-files: ${{ steps.assemble.outputs.merge-channel-files }}
permissions: {}
steps:
- name: Assemble target data
id: assemble
run: |
# Only run the builds that incur runner charges on release or select manually triggered runs.
if [[
"${{ needs.build-type-determination.outputs.is-release }}" == "true" ||
"${{ github.event.inputs.paid-runners }}" == "true"
]]; then
build_matrix="$(
(
echo "${{ env.BASE_BUILD_DATA }}";
echo "${{ env.PAID_RUNNER_BUILD_DATA }}"
) | \
yq \
--output-format json \
'[.[].config]'
)"
artifact_matrix="$(
(
echo "${{ env.BASE_BUILD_DATA }}";
echo "${{ env.PAID_RUNNER_BUILD_DATA }}"
) | \
yq \
--output-format json \
'[.[].artifacts.[]]'
)"
# The build matrix produces two macOS jobs (x86 and ARM) so the "channel update info files"
# generated by each must be merged.
merge_channel_files="true"
else
build_matrix="$(
echo "${{ env.BASE_BUILD_DATA }}" | \
yq \
--output-format json \
'[.[].config]'
)"
artifact_matrix="$(
echo "${{ env.BASE_BUILD_DATA }}" | \
yq \
--output-format json \
'[.[].artifacts.[]]'
)"
merge_channel_files="false"
fi
# Set workflow step outputs.
# See: https://docs.github.com/actions/using-workflows/workflow-commands-for-github-actions#multiline-strings
delimiter="$RANDOM"
echo "build-matrix<<$delimiter" >> $GITHUB_OUTPUT
echo "$build_matrix" >> $GITHUB_OUTPUT
echo "$delimiter" >> $GITHUB_OUTPUT
delimiter="$RANDOM"
echo "artifact-matrix<<$delimiter" >> $GITHUB_OUTPUT
echo "$artifact_matrix" >> $GITHUB_OUTPUT
echo "$delimiter" >> $GITHUB_OUTPUT
echo "merge-channel-files=$merge_channel_files" >> $GITHUB_OUTPUT
build:
name: build (${{ matrix.config.name }})
needs:
- build-type-determination
- select-targets
env:
# https://github.blog/changelog/2024-03-07-github-actions-all-actions-will-run-on-node20-instead-of-node16-by-default/
ACTIONS_ALLOW_USE_UNSECURE_NODE_VERSION: true
# Location of artifacts generated by build.
BUILD_ARTIFACTS_PATH: electron-app/dist/build-artifacts
# to skip passing signing credentials to electron-builder
IS_WINDOWS_CONFIG: ${{ matrix.config.name == 'Windows' }}
INSTALLER_CERT_WINDOWS_CER: "/tmp/cert.cer"
# We are hardcoding the path for signtool because is not present on the windows PATH env var by default.
# Keep in mind that this path could change when upgrading to a new runner version
SIGNTOOL_PATH: "C:/Program Files (x86)/Windows Kits/10/bin/10.0.19041.0/x86/signtool.exe"
WIN_CERT_PASSWORD: ${{ secrets.INSTALLER_CERT_WINDOWS_PASSWORD }}
WIN_CERT_CONTAINER_NAME: ${{ secrets.INSTALLER_CERT_WINDOWS_CONTAINER }}
strategy: strategy:
matrix: matrix:
config: config: ${{ fromJson(needs.select-targets.outputs.build-matrix) }}
- os: windows-2019 runs-on: ${{ matrix.config.runs-on }}
certificate-secret: WINDOWS_SIGNING_CERTIFICATE_PFX # Name of the secret that contains the certificate. container: ${{ fromJSON(matrix.config.container) }}
certificate-password-secret: WINDOWS_SIGNING_CERTIFICATE_PASSWORD # Name of the secret that contains the certificate password. defaults:
certificate-extension: pfx # File extension for the certificate. run:
- os: ubuntu-20.04 # Avoid problems caused by different default shell for container jobs (sh) vs non-container jobs (bash).
- os: macos-latest shell: bash
# APPLE_SIGNING_CERTIFICATE_P12 secret was produced by following the procedure from:
# https://www.kencochrane.com/2020/08/01/build-and-sign-golang-binaries-for-macos-with-github-actions/#exporting-the-developer-certificate
certificate-secret: APPLE_SIGNING_CERTIFICATE_P12
certificate-password-secret: KEYCHAIN_PASSWORD
certificate-extension: p12
runs-on: ${{ matrix.config.os }}
timeout-minutes: 90 timeout-minutes: 90
steps: steps:
- name: Symlink custom working directory
shell: cmd
if: runner.os == 'Windows' && matrix.config.working-directory
run: |
if not exist "${{ matrix.config.working-directory }}" mklink /d "${{ matrix.config.working-directory }}" "C:\actions-runner\_work\arduino-ide\arduino-ide"
- name: Checkout - name: Checkout
if: fromJSON(matrix.config.container) == null
uses: actions/checkout@v4
- name: Checkout
# actions/checkout@v4 has dependency on a higher version of glibc than available in the Linux container.
if: fromJSON(matrix.config.container) != null
uses: actions/checkout@v3 uses: actions/checkout@v3
- name: Install Node.js 16.14 - name: Install Node.js
uses: actions/setup-node@v3 if: fromJSON(matrix.config.container) == null && runner.os != 'Windows'
uses: actions/setup-node@v4
with: with:
node-version: '16.14' node-version: ${{ env.NODE_VERSION }}
registry-url: 'https://registry.npmjs.org' registry-url: 'https://registry.npmjs.org'
cache: 'yarn' cache: 'yarn'
- name: Install Python 3.x - name: Install Python 3.x
uses: actions/setup-python@v4 if: fromJSON(matrix.config.container) == null && runner.os != 'Windows'
uses: actions/setup-python@v5
with: with:
python-version: '3.x' python-version: '3.11.x'
- name: Install Go - name: Install Go
if: fromJSON(matrix.config.container) == null && runner.os != 'Windows'
uses: actions/setup-go@v5
with:
go-version: ${{ env.GO_VERSION }}
- name: Install Go
# actions/setup-go@v5 has dependency on a higher version of glibc than available in the Linux container.
if: fromJSON(matrix.config.container) != null && runner.os != 'Windows'
uses: actions/setup-go@v4 uses: actions/setup-go@v4
with: with:
go-version: ${{ env.GO_VERSION }} go-version: ${{ env.GO_VERSION }}
- name: Install Taskfile - name: Install Taskfile
if: fromJSON(matrix.config.container) == null && runner.os != 'Windows'
uses: arduino/setup-task@v2
with:
repo-token: ${{ secrets.GITHUB_TOKEN }}
version: 3.x
- name: Install Taskfile
# actions/setup-task@v2 has dependency on a higher version of glibc than available in the Linux container.
if: fromJSON(matrix.config.container) != null && runner.os != 'Windows'
uses: arduino/setup-task@v1 uses: arduino/setup-task@v1
with: with:
repo-token: ${{ secrets.GITHUB_TOKEN }} repo-token: ${{ secrets.GITHUB_TOKEN }}
version: 3.x version: 3.x
- name: Package - name: Package
shell: bash
env: env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
AC_USERNAME: ${{ secrets.AC_USERNAME }} AC_USERNAME: ${{ secrets.AC_USERNAME }}
@@ -117,12 +364,18 @@ jobs:
AC_TEAM_ID: ${{ secrets.AC_TEAM_ID }} AC_TEAM_ID: ${{ secrets.AC_TEAM_ID }}
AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }}
AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
IS_NIGHTLY: ${{ github.event_name == 'schedule' || (github.event_name == 'workflow_dispatch' && github.ref == 'refs/heads/main') }} IS_NIGHTLY: ${{ needs.build-type-determination.outputs.is-nightly }}
IS_RELEASE: ${{ startsWith(github.ref, 'refs/tags/') }} IS_RELEASE: ${{ needs.build-type-determination.outputs.is-release }}
CAN_SIGN: ${{ secrets[matrix.config.certificate-secret] != '' }} CAN_SIGN: ${{ secrets[matrix.config.certificate-secret] != '' }}
# The CREATE_* environment vars are only used to run tests. These secrets are optional. Dependent tests will
# be skipped if not available.
CREATE_USERNAME: ${{ secrets.CREATE_USERNAME }}
CREATE_PASSWORD: ${{ secrets.CREATE_PASSWORD }}
CREATE_CLIENT_SECRET: ${{ secrets.CREATE_CLIENT_SECRET }}
working-directory: ${{ runner.os == 'Windows' && matrix.config.working-directory || './' }}
run: | run: |
# See: https://www.electron.build/code-signing # See: https://www.electron.build/code-signing
if [ $CAN_SIGN = false ]; then if [ $CAN_SIGN = false ] || [ $IS_WINDOWS_CONFIG = true ]; then
echo "Skipping the app signing: certificate not provided." echo "Skipping the app signing: certificate not provided."
else else
export CSC_LINK="${{ runner.temp }}/signing_certificate.${{ matrix.config.certificate-extension }}" export CSC_LINK="${{ runner.temp }}/signing_certificate.${{ matrix.config.certificate-extension }}"
@@ -131,10 +384,6 @@ jobs:
export CSC_FOR_PULL_REQUEST=true export CSC_FOR_PULL_REQUEST=true
fi fi
if [ "${{ runner.OS }}" = "Windows" ]; then
npm config set msvs_version 2017 --global
fi
npx node-gyp install npx node-gyp install
yarn install --immutable yarn install --immutable
@@ -147,35 +396,121 @@ jobs:
yarn --cwd electron-app build yarn --cwd electron-app build
yarn --cwd electron-app package yarn --cwd electron-app package
# Both macOS jobs generate a "channel update info file" with same path and name. The second job to complete would
# overwrite the file generated by the first in the workflow artifact.
- name: Stage channel file for merge
if: >
needs.select-targets.outputs.merge-channel-files == 'true' &&
matrix.config.mergeable-channel-file == 'true'
working-directory: ${{ runner.os == 'Windows' && matrix.config.working-directory || './' }}
run: |
staged_channel_files_path="${{ runner.temp }}/staged-channel-files"
mkdir "$staged_channel_files_path"
mv \
"${{ env.BUILD_ARTIFACTS_PATH }}/${{ needs.build-type-determination.outputs.channel-name }}-mac.yml" \
"${staged_channel_files_path}/${{ needs.build-type-determination.outputs.channel-name }}-mac-${{ runner.arch }}.yml"
# Set workflow environment variable for use in other steps.
# See: https://docs.github.com/actions/using-workflows/workflow-commands-for-github-actions#setting-an-environment-variable
echo "STAGED_CHANNEL_FILES_PATH=$staged_channel_files_path" >> "$GITHUB_ENV"
- name: Upload staged-for-merge channel file artifact
uses: actions/upload-artifact@v3
if: >
needs.select-targets.outputs.merge-channel-files == 'true' &&
matrix.config.mergeable-channel-file == 'true'
with:
if-no-files-found: error
name: ${{ env.STAGED_CHANNEL_FILES_ARTIFACT }}
path: ${{ runner.os == 'Windows' && matrix.config.working-directory && format('{0}/{1}', matrix.config.working-directory, env.STAGED_CHANNEL_FILES_PATH) || env.STAGED_CHANNEL_FILES_PATH }}
- name: Upload [GitHub Actions] - name: Upload [GitHub Actions]
uses: actions/upload-artifact@v3 uses: actions/upload-artifact@v3
with: with:
name: ${{ env.JOB_TRANSFER_ARTIFACT }} name: ${{ env.JOB_TRANSFER_ARTIFACT }}
path: electron-app/dist/build-artifacts path: ${{ runner.os == 'Windows' && matrix.config.working-directory && format('{0}/{1}', matrix.config.working-directory, env.BUILD_ARTIFACTS_PATH) || env.BUILD_ARTIFACTS_PATH }}
- name: Manual Clean up for self-hosted runners
if: runner.os == 'Windows' && matrix.config.working-directory
shell: cmd
run: |
rmdir /s /q "${{ matrix.config.working-directory }}\${{ env.BUILD_ARTIFACTS_PATH }}"
merge-channel-files:
needs:
- build-type-determination
- select-targets
- build
if: needs.select-targets.outputs.merge-channel-files == 'true'
runs-on: ubuntu-latest
permissions: {}
steps:
- name: Set environment variables
run: |
# See: https://docs.github.com/actions/using-workflows/workflow-commands-for-github-actions#setting-an-environment-variable
echo "CHANNEL_FILES_PATH=${{ runner.temp }}/channel-files" >> "$GITHUB_ENV"
- name: Checkout
uses: actions/checkout@v4
- name: Download staged-for-merge channel files artifact
uses: actions/download-artifact@v3
with:
name: ${{ env.STAGED_CHANNEL_FILES_ARTIFACT }}
path: ${{ env.CHANNEL_FILES_PATH }}
- name: Remove no longer needed artifact
uses: geekyeggo/delete-artifact@v2
with:
name: ${{ env.STAGED_CHANNEL_FILES_ARTIFACT }}
- name: Install Node.js
uses: actions/setup-node@v4
with:
node-version: ${{ env.NODE_VERSION }}
registry-url: 'https://registry.npmjs.org'
cache: 'yarn'
- name: Install Go
uses: actions/setup-go@v5
with:
go-version: ${{ env.GO_VERSION }}
- name: Install Task
uses: arduino/setup-task@v2
with:
repo-token: ${{ secrets.GITHUB_TOKEN }}
version: 3.x
- name: Install dependencies
run: yarn
- name: Merge "channel update info files"
run: |
node \
./scripts/merge-channel-files.js \
--channel "${{ needs.build-type-determination.outputs.channel-name }}" \
--input "${{ env.CHANNEL_FILES_PATH }}"
- name: Upload merged channel files to job transfer artifact
uses: actions/upload-artifact@v3
with:
if-no-files-found: error
name: ${{ env.JOB_TRANSFER_ARTIFACT }}
path: ${{ env.CHANNEL_FILES_PATH }}
artifacts: artifacts:
name: ${{ matrix.artifact.name }} artifact name: ${{ matrix.artifact.name }} artifact
needs: build needs:
- select-targets
- build
if: always() && needs.build.result != 'skipped' if: always() && needs.build.result != 'skipped'
runs-on: ubuntu-latest runs-on: ubuntu-latest
strategy: strategy:
matrix: matrix:
artifact: artifact: ${{ fromJson(needs.select-targets.outputs.artifact-matrix) }}
- path: '*Linux_64bit.zip'
name: Linux_X86-64_zip
- path: '*Linux_64bit.AppImage'
name: Linux_X86-64_app_image
- path: '*macOS_64bit.dmg'
name: macOS_dmg
- path: '*macOS_64bit.zip'
name: macOS_zip
- path: '*Windows_64bit.exe'
name: Windows_X86-64_interactive_installer
- path: '*Windows_64bit.msi'
name: Windows_X86-64_MSI
- path: '*Windows_64bit.zip'
name: Windows_X86-64_zip
steps: steps:
- name: Download job transfer artifact - name: Download job transfer artifact
@@ -191,20 +526,22 @@ jobs:
path: ${{ env.JOB_TRANSFER_ARTIFACT }}/${{ matrix.artifact.path }} path: ${{ env.JOB_TRANSFER_ARTIFACT }}/${{ matrix.artifact.path }}
changelog: changelog:
needs: build needs:
- build-type-determination
- build
runs-on: ubuntu-latest runs-on: ubuntu-latest
outputs: outputs:
BODY: ${{ steps.changelog.outputs.BODY }} BODY: ${{ steps.changelog.outputs.BODY }}
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v3 uses: actions/checkout@v4
with: with:
fetch-depth: 0 # To fetch all history for all branches and tags. fetch-depth: 0 # To fetch all history for all branches and tags.
- name: Generate Changelog - name: Generate Changelog
id: changelog id: changelog
env: env:
IS_RELEASE: ${{ startsWith(github.ref, 'refs/tags/') }} IS_RELEASE: ${{ needs.build-type-determination.outputs.is-release }}
run: | run: |
export LATEST_TAG=$(git describe --abbrev=0) export LATEST_TAG=$(git describe --abbrev=0)
export GIT_LOG=$(git log --pretty=" - %s [%h]" $LATEST_TAG..HEAD | sed 's/ *$//g') export GIT_LOG=$(git log --pretty=" - %s [%h]" $LATEST_TAG..HEAD | sed 's/ *$//g')
@@ -230,15 +567,27 @@ jobs:
echo "$BODY" > CHANGELOG.txt echo "$BODY" > CHANGELOG.txt
- name: Upload Changelog [GitHub Actions] - name: Upload Changelog [GitHub Actions]
if: github.event_name == 'schedule' || (github.event_name == 'workflow_dispatch' && github.ref == 'refs/heads/main') if: needs.build-type-determination.outputs.is-nightly == 'true'
uses: actions/upload-artifact@v3 uses: actions/upload-artifact@v3
with: with:
name: ${{ env.JOB_TRANSFER_ARTIFACT }} name: ${{ env.JOB_TRANSFER_ARTIFACT }}
path: CHANGELOG.txt path: CHANGELOG.txt
publish: publish:
needs: changelog needs:
if: github.repository == 'arduino/arduino-ide' && (github.event_name == 'schedule' || (github.event_name == 'workflow_dispatch' && github.ref == 'refs/heads/main')) - build-type-determination
- merge-channel-files
- changelog
if: >
always() &&
needs.build-type-determination.result == 'success' &&
(
needs.merge-channel-files.result == 'skipped' ||
needs.merge-channel-files.result == 'success'
) &&
needs.changelog.result == 'success' &&
needs.build-type-determination.outputs.publish-to-s3 == 'true' &&
needs.build-type-determination.outputs.is-nightly == 'true'
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Download [GitHub Actions] - name: Download [GitHub Actions]
@@ -258,8 +607,19 @@ jobs:
AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
release: release:
needs: changelog needs:
if: startsWith(github.ref, 'refs/tags/') - build-type-determination
- merge-channel-files
- changelog
if: >
always() &&
needs.build-type-determination.result == 'success' &&
(
needs.merge-channel-files.result == 'skipped' ||
needs.merge-channel-files.result == 'success'
) &&
needs.changelog.result == 'success' &&
needs.build-type-determination.outputs.is-release == 'true'
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Download [GitHub Actions] - name: Download [GitHub Actions]
@@ -283,17 +643,8 @@ jobs:
file_glob: true file_glob: true
body: ${{ needs.changelog.outputs.BODY }} body: ${{ needs.changelog.outputs.BODY }}
# Temporary measure to prevent release update offers before the manually produced builds are uploaded.
# The step must be removed once fully automated builds are regained.
- name: Remove "channel update info files" related to manual builds
run: |
# See: https://github.com/arduino/arduino-ide/issues/2018
rm "${{ env.JOB_TRANSFER_ARTIFACT }}/stable-linux.yml"
# See: https://github.com/arduino/arduino-ide/issues/408
rm "${{ env.JOB_TRANSFER_ARTIFACT }}/stable-mac.yml"
- name: Publish Release [S3] - name: Publish Release [S3]
if: github.repository == 'arduino/arduino-ide' if: needs.build-type-determination.outputs.publish-to-s3 == 'true'
uses: docker://plugins/s3 uses: docker://plugins/s3
env: env:
PLUGIN_SOURCE: '${{ env.JOB_TRANSFER_ARTIFACT }}/*' PLUGIN_SOURCE: '${{ env.JOB_TRANSFER_ARTIFACT }}/*'
@@ -307,6 +658,7 @@ jobs:
# This job must run after all jobs that use the transfer artifact. # This job must run after all jobs that use the transfer artifact.
needs: needs:
- build - build
- merge-channel-files
- publish - publish
- release - release
- artifacts - artifacts

View File

@@ -74,9 +74,11 @@ jobs:
- identifier: macOS signing certificate # Text used to identify certificate in notifications. - identifier: macOS signing certificate # Text used to identify certificate in notifications.
certificate-secret: APPLE_SIGNING_CERTIFICATE_P12 # Name of the secret that contains the certificate. certificate-secret: APPLE_SIGNING_CERTIFICATE_P12 # Name of the secret that contains the certificate.
password-secret: KEYCHAIN_PASSWORD # Name of the secret that contains the certificate password. password-secret: KEYCHAIN_PASSWORD # Name of the secret that contains the certificate password.
type: pkcs12
- identifier: Windows signing certificate - identifier: Windows signing certificate
certificate-secret: WINDOWS_SIGNING_CERTIFICATE_PFX certificate-secret: INSTALLER_CERT_WINDOWS_CER
password-secret: WINDOWS_SIGNING_CERTIFICATE_PASSWORD # The password for the Windows certificate is not needed, because its not a container, but a single certificate.
type: x509
steps: steps:
- name: Set certificate path environment variable - name: Set certificate path environment variable
@@ -95,7 +97,7 @@ jobs:
CERTIFICATE_PASSWORD: ${{ secrets[matrix.certificate.password-secret] }} CERTIFICATE_PASSWORD: ${{ secrets[matrix.certificate.password-secret] }}
run: | run: |
( (
openssl pkcs12 \ openssl ${{ matrix.certificate.type }} \
-in "${{ env.CERTIFICATE_PATH }}" \ -in "${{ env.CERTIFICATE_PATH }}" \
-legacy \ -legacy \
-noout \ -noout \
@@ -122,26 +124,43 @@ jobs:
CERTIFICATE_PASSWORD: ${{ secrets[matrix.certificate.password-secret] }} CERTIFICATE_PASSWORD: ${{ secrets[matrix.certificate.password-secret] }}
id: get-days-before-expiration id: get-days-before-expiration
run: | run: |
EXPIRATION_DATE="$( if [[ ${{ matrix.certificate.type }} == "pkcs12" ]]; then
( EXPIRATION_DATE="$(
openssl pkcs12 \ (
-in "${{ env.CERTIFICATE_PATH }}" \ openssl pkcs12 \
-clcerts \ -in "${{ env.CERTIFICATE_PATH }}" \
-legacy \ -clcerts \
-nodes \ -legacy \
-passin env:CERTIFICATE_PASSWORD -nodes \
) | ( -passin env:CERTIFICATE_PASSWORD
openssl x509 \ ) | (
-noout \ openssl x509 \
-enddate -noout \
) | ( -enddate
grep \ ) | (
--max-count=1 \ grep \
--only-matching \ --max-count=1 \
--perl-regexp \ --only-matching \
'notAfter=(\K.*)' --perl-regexp \
) 'notAfter=(\K.*)'
)" )
)"
elif [[ ${{ matrix.certificate.type }} == "x509" ]]; then
EXPIRATION_DATE="$(
(
openssl x509 \
-in ${{ env.CERTIFICATE_PATH }} \
-noout \
-enddate
) | (
grep \
--max-count=1 \
--only-matching \
--perl-regexp \
'notAfter=(\K.*)'
)
)"
fi
DAYS_BEFORE_EXPIRATION="$((($(date --utc --date="$EXPIRATION_DATE" +%s) - $(date --utc +%s)) / 60 / 60 / 24))" DAYS_BEFORE_EXPIRATION="$((($(date --utc --date="$EXPIRATION_DATE" +%s) - $(date --utc +%s)) / 60 / 60 / 24))"

58
.github/workflows/check-containers.yml vendored Normal file
View File

@@ -0,0 +1,58 @@
name: Check Containers
on:
pull_request:
paths:
- ".github/workflows/check-containers.ya?ml"
- "**.Dockerfile"
- "**/Dockerfile"
push:
paths:
- ".github/workflows/check-containers.ya?ml"
- "**.Dockerfile"
- "**/Dockerfile"
repository_dispatch:
schedule:
# Run periodically to catch breakage caused by external changes.
- cron: "0 7 * * MON"
workflow_dispatch:
jobs:
run:
name: Run (${{ matrix.image.path }})
runs-on: ubuntu-latest
permissions: {}
services:
registry:
image: registry:2
ports:
- 5000:5000
env:
IMAGE_NAME: name/app:latest
REGISTRY: localhost:5000
strategy:
fail-fast: false
matrix:
image:
- path: .github/workflows/assets/linux.Dockerfile
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Build and push to local registry
uses: docker/build-push-action@v5
with:
context: .
file: ${{ matrix.image.path }}
push: true
tags: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
- name: Run container
run: |
docker \
run \
--rm \
${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}

View File

@@ -2,7 +2,7 @@ name: Check Internationalization
env: env:
# See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml # See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml
GO_VERSION: "1.19" GO_VERSION: '1.21'
# See: https://docs.github.com/en/actions/reference/events-that-trigger-workflows # See: https://docs.github.com/en/actions/reference/events-that-trigger-workflows
on: on:
@@ -56,22 +56,22 @@ jobs:
steps: steps:
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v3 uses: actions/checkout@v4
- name: Install Node.js 16.14 - name: Install Node.js 18.17
uses: actions/setup-node@v3 uses: actions/setup-node@v4
with: with:
node-version: '16.14' node-version: '18.17'
registry-url: 'https://registry.npmjs.org' registry-url: 'https://registry.npmjs.org'
cache: 'yarn' cache: 'yarn'
- name: Install Go - name: Install Go
uses: actions/setup-go@v4 uses: actions/setup-go@v5
with: with:
go-version: ${{ env.GO_VERSION }} go-version: ${{ env.GO_VERSION }}
- name: Install Taskfile - name: Install Taskfile
uses: arduino/setup-task@v1 uses: arduino/setup-task@v2
with: with:
repo-token: ${{ secrets.GITHUB_TOKEN }} repo-token: ${{ secrets.GITHUB_TOKEN }}
version: 3.x version: 3.x

View File

@@ -8,7 +8,7 @@ on:
env: env:
CHANGELOG_ARTIFACTS: changelog CHANGELOG_ARTIFACTS: changelog
# See: https://github.com/actions/setup-node/#readme # See: https://github.com/actions/setup-node/#readme
NODE_VERSION: 16.x NODE_VERSION: '18.17'
jobs: jobs:
create-changelog: create-changelog:
@@ -16,10 +16,10 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v3 uses: actions/checkout@v4
- name: Install Node.js - name: Install Node.js
uses: actions/setup-node@v3 uses: actions/setup-node@v4
with: with:
node-version: ${{ env.NODE_VERSION }} node-version: ${{ env.NODE_VERSION }}
registry-url: 'https://registry.npmjs.org' registry-url: 'https://registry.npmjs.org'
@@ -32,7 +32,7 @@ jobs:
- name: Create full changelog - name: Create full changelog
id: full-changelog id: full-changelog
run: | run: |
yarn add @octokit/rest --ignore-workspace-root-check yarn add @octokit/rest@19.0.13 --ignore-workspace-root-check
mkdir "${{ github.workspace }}/${{ env.CHANGELOG_ARTIFACTS }}" mkdir "${{ github.workspace }}/${{ env.CHANGELOG_ARTIFACTS }}"
# Get the changelog file name to build # Get the changelog file name to build

View File

@@ -2,7 +2,7 @@ name: i18n-nightly-push
env: env:
# See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml # See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml
GO_VERSION: "1.19" GO_VERSION: '1.21'
on: on:
schedule: schedule:
@@ -14,22 +14,22 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v3 uses: actions/checkout@v4
- name: Install Node.js 16.14 - name: Install Node.js 18.17
uses: actions/setup-node@v3 uses: actions/setup-node@v4
with: with:
node-version: '16.14' node-version: '18.17'
registry-url: 'https://registry.npmjs.org' registry-url: 'https://registry.npmjs.org'
cache: 'yarn' cache: 'yarn'
- name: Install Go - name: Install Go
uses: actions/setup-go@v4 uses: actions/setup-go@v5
with: with:
go-version: ${{ env.GO_VERSION }} go-version: ${{ env.GO_VERSION }}
- name: Install Task - name: Install Task
uses: arduino/setup-task@v1 uses: arduino/setup-task@v2
with: with:
repo-token: ${{ secrets.GITHUB_TOKEN }} repo-token: ${{ secrets.GITHUB_TOKEN }}
version: 3.x version: 3.x

View File

@@ -2,7 +2,7 @@ name: i18n-weekly-pull
env: env:
# See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml # See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml
GO_VERSION: "1.19" GO_VERSION: '1.21'
on: on:
schedule: schedule:
@@ -14,22 +14,22 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v3 uses: actions/checkout@v4
- name: Install Node.js 16.14 - name: Install Node.js 18.17
uses: actions/setup-node@v3 uses: actions/setup-node@v4
with: with:
node-version: '16.14' node-version: '18.17'
registry-url: 'https://registry.npmjs.org' registry-url: 'https://registry.npmjs.org'
cache: 'yarn' cache: 'yarn'
- name: Install Go - name: Install Go
uses: actions/setup-go@v4 uses: actions/setup-go@v5
with: with:
go-version: ${{ env.GO_VERSION }} go-version: ${{ env.GO_VERSION }}
- name: Install Task - name: Install Task
uses: arduino/setup-task@v1 uses: arduino/setup-task@v2
with: with:
repo-token: ${{ secrets.GITHUB_TOKEN }} repo-token: ${{ secrets.GITHUB_TOKEN }}
version: 3.x version: 3.x

View File

@@ -0,0 +1,70 @@
name: Push Container Images
on:
pull_request:
paths:
- ".github/workflows/push-container-images.ya?ml"
push:
paths:
- ".github/workflows/push-container-images.ya?ml"
- "**.Dockerfile"
- "**/Dockerfile"
repository_dispatch:
schedule:
# Run periodically to catch breakage caused by external changes.
- cron: "0 8 * * MON"
workflow_dispatch:
jobs:
push:
name: Push (${{ matrix.image.name }})
# Only run the job when GITHUB_TOKEN has the privileges required for Container registry login.
if: >
(
github.event_name != 'pull_request' &&
github.repository == 'arduino/arduino-ide'
) ||
(
github.event_name == 'pull_request' &&
github.event.pull_request.head.repo.full_name == 'arduino/arduino-ide'
)
runs-on: ubuntu-latest
permissions:
contents: read
packages: write
strategy:
fail-fast: false
matrix:
image:
- path: .github/workflows/assets/linux.Dockerfile
name: ${{ github.repository }}/linux
registry: ghcr.io
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Log in to the Container registry
uses: docker/login-action@v3
with:
password: ${{ secrets.GITHUB_TOKEN }}
registry: ${{ matrix.image.registry }}
username: ${{ github.repository_owner }}
- name: Extract metadata for image
id: metadata
uses: docker/metadata-action@v5
with:
images: ${{ matrix.image.registry }}/${{ matrix.image.name }}
- name: Build and push image
uses: docker/build-push-action@v5
with:
context: .
file: ${{ matrix.image.path }}
labels: ${{ steps.metadata.outputs.labels }}
# Workflow is triggered on relevant events for the sake of a "dry run" validation but image is only pushed to
# registry on commit to the main branch.
push: ${{ github.ref == 'refs/heads/main' }}
tags: ${{ steps.metadata.outputs.tags }}

View File

@@ -5,15 +5,15 @@ name: Sync Labels
on: on:
push: push:
paths: paths:
- ".github/workflows/sync-labels.ya?ml" - '.github/workflows/sync-labels.ya?ml'
- ".github/label-configuration-files/*.ya?ml" - '.github/label-configuration-files/*.ya?ml'
pull_request: pull_request:
paths: paths:
- ".github/workflows/sync-labels.ya?ml" - '.github/workflows/sync-labels.ya?ml'
- ".github/label-configuration-files/*.ya?ml" - '.github/label-configuration-files/*.ya?ml'
schedule: schedule:
# Run daily at 8 AM UTC to sync with changes to shared label configurations. # Run daily at 8 AM UTC to sync with changes to shared label configurations.
- cron: "0 8 * * *" - cron: '0 8 * * *'
workflow_dispatch: workflow_dispatch:
repository_dispatch: repository_dispatch:
@@ -27,7 +27,7 @@ jobs:
steps: steps:
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v3 uses: actions/checkout@v4
- name: Download JSON schema for labels configuration file - name: Download JSON schema for labels configuration file
id: download-schema id: download-schema
@@ -106,7 +106,7 @@ jobs:
echo "flag=--dry-run" >> $GITHUB_OUTPUT echo "flag=--dry-run" >> $GITHUB_OUTPUT
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v3 uses: actions/checkout@v4
- name: Download configuration files artifact - name: Download configuration files artifact
uses: actions/download-artifact@v3 uses: actions/download-artifact@v3

View File

@@ -8,30 +8,30 @@ on:
env: env:
# See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml # See vars.GO_VERSION field of https://github.com/arduino/arduino-cli/blob/master/DistTasks.yml
GO_VERSION: "1.19" GO_VERSION: '1.21'
NODE_VERSION: 16.x NODE_VERSION: '18.17'
jobs: jobs:
pull-from-jsonbin: pull-from-jsonbin:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v3 uses: actions/checkout@v4
- name: Install Node.js - name: Install Node.js
uses: actions/setup-node@v3 uses: actions/setup-node@v4
with: with:
node-version: ${{ env.NODE_VERSION }} node-version: ${{ env.NODE_VERSION }}
registry-url: 'https://registry.npmjs.org' registry-url: 'https://registry.npmjs.org'
cache: 'yarn' cache: 'yarn'
- name: Install Go - name: Install Go
uses: actions/setup-go@v4 uses: actions/setup-go@v5
with: with:
go-version: ${{ env.GO_VERSION }} go-version: ${{ env.GO_VERSION }}
- name: Install Task - name: Install Task
uses: arduino/setup-task@v1 uses: arduino/setup-task@v2
with: with:
repo-token: ${{ secrets.GITHUB_TOKEN }} repo-token: ${{ secrets.GITHUB_TOKEN }}
version: 3.x version: 3.x

2
.gitignore vendored
View File

@@ -1,7 +1,7 @@
node_modules/ node_modules/
lib/ lib/
downloads/ downloads/
resources/ arduino-ide-extension/src/node/resources
arduino-ide-extension/Examples/ arduino-ide-extension/Examples/
src-gen/ src-gen/
gen-webpack.config.js gen-webpack.config.js

11
.prettierignore Normal file
View File

@@ -0,0 +1,11 @@
lib
dist
plugins
src-gen
i18n
gen-webpack*
.browser_modules
arduino-ide-extension/src/node/resources
cli-protocol
*color-theme.json
arduino-icons.json

View File

@@ -1,7 +0,0 @@
{
"singleQuote": true,
"tabWidth": 2,
"useTabs": false,
"printWidth": 80,
"endOfLine": "auto"
}

28
.prettierrc.json Normal file
View File

@@ -0,0 +1,28 @@
{
"singleQuote": true,
"tabWidth": 2,
"useTabs": false,
"printWidth": 80,
"endOfLine": "auto",
"overrides": [
{
"files": "*.json",
"options": {
"tabWidth": 2
}
},
{
"files": "*.css",
"options": {
"tabWidth": 4,
"singleQuote": false
}
},
{
"files": "*.html",
"options": {
"tabWidth": 4
}
}
]
}

15
.vscode/launch.json vendored
View File

@@ -7,7 +7,7 @@
"name": "App", "name": "App",
"runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron", "runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron",
"windows": { "windows": {
"runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron.cmd", "runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron.cmd"
}, },
"cwd": "${workspaceFolder}/electron-app", "cwd": "${workspaceFolder}/electron-app",
"args": [ "args": [
@@ -19,7 +19,7 @@
"--no-app-auto-install", "--no-app-auto-install",
"--plugins=local-dir:./plugins", "--plugins=local-dir:./plugins",
"--hosted-plugin-inspect=9339", "--hosted-plugin-inspect=9339",
"--no-ping-timeout", "--no-ping-timeout"
], ],
"env": { "env": {
"NODE_ENV": "development" "NODE_ENV": "development"
@@ -42,7 +42,7 @@
"name": "App [Dev]", "name": "App [Dev]",
"runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron", "runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron",
"windows": { "windows": {
"runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron.cmd", "runtimeExecutable": "${workspaceFolder}/node_modules/.bin/electron.cmd"
}, },
"cwd": "${workspaceFolder}/electron-app", "cwd": "${workspaceFolder}/electron-app",
"args": [ "args": [
@@ -56,7 +56,7 @@
"--hosted-plugin-inspect=9339", "--hosted-plugin-inspect=9339",
"--content-trace", "--content-trace",
"--open-devtools", "--open-devtools",
"--no-ping-timeout", "--no-ping-timeout"
], ],
"env": { "env": {
"NODE_ENV": "development" "NODE_ENV": "development"
@@ -115,15 +115,12 @@
"request": "attach", "request": "attach",
"name": "Attach by Process ID", "name": "Attach by Process ID",
"processId": "${command:PickProcess}" "processId": "${command:PickProcess}"
}, }
], ],
"compounds": [ "compounds": [
{ {
"name": "Launch Electron Backend & Frontend", "name": "Launch Electron Backend & Frontend",
"configurations": [ "configurations": ["App", "Attach to Electron Frontend"]
"App",
"Attach to Electron Frontend"
]
} }
] ]
} }

View File

@@ -7,6 +7,6 @@
}, },
"typescript.tsdk": "node_modules/typescript/lib", "typescript.tsdk": "node_modules/typescript/lib",
"editor.codeActionsOnSave": { "editor.codeActionsOnSave": {
"source.fixAll.eslint": true "source.fixAll.eslint": "explicit"
}, }
} }

10
.vscode/tasks.json vendored
View File

@@ -4,8 +4,11 @@
{ {
"label": "Rebuild App", "label": "Rebuild App",
"type": "shell", "type": "shell",
"command": "yarn rebuild:browser && yarn rebuild:electron", "command": "yarn rebuild",
"group": "build", "group": "build",
"options": {
"cwd": "${workspaceFolder}/electron-app"
},
"presentation": { "presentation": {
"reveal": "always", "reveal": "always",
"panel": "new", "panel": "new",
@@ -37,10 +40,7 @@
{ {
"label": "Watch All", "label": "Watch All",
"type": "shell", "type": "shell",
"dependsOn": [ "dependsOn": ["Watch Extension", "Watch App"]
"Watch Extension",
"Watch App"
]
} }
] ]
} }

View File

@@ -4,7 +4,7 @@
[![Arduino IDE](https://github.com/arduino/arduino-ide/workflows/Arduino%20IDE/badge.svg)](https://github.com/arduino/arduino-ide/actions?query=workflow%3A%22Arduino+IDE%22) [![Arduino IDE](https://github.com/arduino/arduino-ide/workflows/Arduino%20IDE/badge.svg)](https://github.com/arduino/arduino-ide/actions?query=workflow%3A%22Arduino+IDE%22)
This repository contains the source code of the Arduino IDE 2.x. If you're looking for the old IDE, go to the repository of the 1.x version at https://github.com/arduino/Arduino. This repository contains the source code of the Arduino IDE 2.x. If you're looking for the old IDE, go to the [repository of the 1.x version](https://github.com/arduino/Arduino).
The Arduino IDE 2.x is a major rewrite, sharing no code with the IDE 1.x. It is based on the [Theia IDE](https://theia-ide.org/) framework and built with [Electron](https://www.electronjs.org/). The backend operations such as compilation and uploading are offloaded to an [arduino-cli](https://github.com/arduino/arduino-cli) instance running in daemon mode. This new IDE was developed with the goal of preserving the same interface and user experience of the previous major version in order to provide a frictionless upgrade. The Arduino IDE 2.x is a major rewrite, sharing no code with the IDE 1.x. It is based on the [Theia IDE](https://theia-ide.org/) framework and built with [Electron](https://www.electronjs.org/). The backend operations such as compilation and uploading are offloaded to an [arduino-cli](https://github.com/arduino/arduino-cli) instance running in daemon mode. This new IDE was developed with the goal of preserving the same interface and user experience of the previous major version in order to provide a frictionless upgrade.

View File

@@ -55,12 +55,14 @@ The Config Service knows about your system, like for example the default sketch
- checking whether a file is in a data or sketch directory - checking whether a file is in a data or sketch directory
### `"arduino"` configuration in the `package.json`: ### `"arduino"` configuration in the `package.json`:
- `"cli"`:
- `"version"` type `string` | `{ owner: string, repo: string, commitish?: string }`: if the type is a `string` and is a valid semver, it will get the corresponding [released](https://github.com/arduino/arduino-cli/releases) CLI. If the type is `string` and is a [date in `YYYYMMDD`](https://arduino.github.io/arduino-cli/latest/installation/#nightly-builds) format, it will get a nightly CLI. If the type is an object, a CLI, build from the sources in the `owner/repo` will be used. If `commitish` is not defined, the HEAD of the default branch will be used. In any other cases an error is thrown. - `"cli"`:
- `"version"` type `string` | `{ owner: string, repo: string, commitish?: string }`: if the type is a `string` and is a valid semver, it will get the corresponding [released](https://github.com/arduino/arduino-cli/releases) CLI. If the type is `string` and is a [date in `YYYYMMDD`](https://arduino.github.io/arduino-cli/latest/installation/#nightly-builds) format, it will get a nightly CLI. If the type is an object, a CLI, build from the sources in the `owner/repo` will be used. If `commitish` is not defined, the HEAD of the default branch will be used. In any other cases an error is thrown.
#### Rebuild gRPC protocol interfaces #### Rebuild gRPC protocol interfaces
- Some CLI updates can bring changes to the gRPC interfaces, as the API might change. gRPC interfaces can be updated running the command
`yarn --cwd arduino-ide-extension generate-protocol` - Some CLI updates can bring changes to the gRPC interfaces, as the API might change. gRPC interfaces can be updated running the command
`yarn --cwd arduino-ide-extension generate-protocol`
### Update **clangd** and **ClangFormat** ### Update **clangd** and **ClangFormat**
@@ -72,11 +74,13 @@ The [**clangd** C++ language server](https://clangd.llvm.org/) and the [**ClangF
1. Submit a pull request in [the `arduino/tooling-project-assets` repository](https://github.com/arduino/tooling-project-assets) to update the version in the `vars.DEFAULT_CLANG_FORMAT_VERSION` field of [`Taskfile.yml`](https://github.com/arduino/tooling-project-assets/blob/main/Taskfile.yml). 1. Submit a pull request in [the `arduino/tooling-project-assets` repository](https://github.com/arduino/tooling-project-assets) to update the version in the `vars.DEFAULT_CLANG_FORMAT_VERSION` field of [`Taskfile.yml`](https://github.com/arduino/tooling-project-assets/blob/main/Taskfile.yml).
### Customize Icons ### Customize Icons
ArduinoIde uses a customized version of FontAwesome. ArduinoIde uses a customized version of FontAwesome.
In order to update/replace icons follow the following steps: In order to update/replace icons follow the following steps:
- import the file `arduino-icons.json` in [Icomoon](https://icomoon.io/app/#/projects)
- load it - import the file `arduino-icons.json` in [Icomoon](https://icomoon.io/app/#/projects)
- edit the icons as needed - load it
- !! download the **new** `arduino-icons.json` file and put it in this repo - edit the icons as needed
- Click on "Generate Font" in Icomoon, then download - !! download the **new** `arduino-icons.json` file and put it in this repo
- place the updated fonts in the `src/style/fonts` directory - Click on "Generate Font" in Icomoon, then download
- place the updated fonts in the `src/style/fonts` directory

View File

@@ -1,6 +1,6 @@
{ {
"name": "arduino-ide-extension", "name": "arduino-ide-extension",
"version": "2.2.0", "version": "2.3.3",
"description": "An extension for Theia building the Arduino IDE", "description": "An extension for Theia building the Arduino IDE",
"license": "AGPL-3.0-or-later", "license": "AGPL-3.0-or-later",
"scripts": { "scripts": {
@@ -24,29 +24,29 @@
}, },
"dependencies": { "dependencies": {
"@grpc/grpc-js": "^1.8.14", "@grpc/grpc-js": "^1.8.14",
"@theia/application-package": "1.39.0", "@theia/application-package": "1.41.0",
"@theia/core": "1.39.0", "@theia/core": "1.41.0",
"@theia/debug": "1.39.0", "@theia/debug": "1.41.0",
"@theia/editor": "1.39.0", "@theia/editor": "1.41.0",
"@theia/electron": "1.39.0", "@theia/electron": "1.41.0",
"@theia/filesystem": "1.39.0", "@theia/filesystem": "1.41.0",
"@theia/keymaps": "1.39.0", "@theia/keymaps": "1.41.0",
"@theia/markers": "1.39.0", "@theia/markers": "1.41.0",
"@theia/messages": "1.39.0", "@theia/messages": "1.41.0",
"@theia/monaco": "1.39.0", "@theia/monaco": "1.41.0",
"@theia/monaco-editor-core": "1.72.3", "@theia/monaco-editor-core": "1.72.3",
"@theia/navigator": "1.39.0", "@theia/navigator": "1.41.0",
"@theia/outline-view": "1.39.0", "@theia/outline-view": "1.41.0",
"@theia/output": "1.39.0", "@theia/output": "1.41.0",
"@theia/plugin-ext": "1.39.0", "@theia/plugin-ext": "1.41.0",
"@theia/preferences": "1.39.0", "@theia/preferences": "1.41.0",
"@theia/scm": "1.39.0", "@theia/scm": "1.41.0",
"@theia/search-in-workspace": "1.39.0", "@theia/search-in-workspace": "1.41.0",
"@theia/terminal": "1.39.0", "@theia/terminal": "1.41.0",
"@theia/typehierarchy": "1.39.0", "@theia/typehierarchy": "1.41.0",
"@theia/workspace": "1.39.0", "@theia/workspace": "1.41.0",
"@tippyjs/react": "^4.2.5", "@tippyjs/react": "^4.2.5",
"@types/auth0-js": "^9.14.0", "@types/auth0-js": "^9.21.3",
"@types/btoa": "^1.2.3", "@types/btoa": "^1.2.3",
"@types/dateformat": "^3.0.1", "@types/dateformat": "^3.0.1",
"@types/google-protobuf": "^3.7.2", "@types/google-protobuf": "^3.7.2",
@@ -60,18 +60,19 @@
"@types/temp": "^0.8.34", "@types/temp": "^0.8.34",
"arduino-serial-plotter-webapp": "0.2.0", "arduino-serial-plotter-webapp": "0.2.0",
"async-mutex": "^0.3.0", "async-mutex": "^0.3.0",
"auth0-js": "^9.14.0", "auth0-js": "^9.23.2",
"btoa": "^1.2.1", "btoa": "^1.2.1",
"classnames": "^2.3.1", "classnames": "^2.3.1",
"cpy": "^10.0.0",
"cross-fetch": "^3.1.5", "cross-fetch": "^3.1.5",
"dateformat": "^3.0.3", "dateformat": "^3.0.3",
"deepmerge": "^4.2.2", "deepmerge": "^4.2.2",
"drivelist": "^9.2.4", "drivelist": "^9.2.4",
"electron-updater": "^4.6.5", "electron-updater": "^4.6.5",
"fast-deep-equal": "^3.1.3",
"fast-json-stable-stringify": "^2.1.0", "fast-json-stable-stringify": "^2.1.0",
"fast-safe-stringify": "^2.1.1", "fast-safe-stringify": "^2.1.1",
"filename-reserved-regex": "^2.0.0", "filename-reserved-regex": "^2.0.0",
"fqbn": "^1.0.5",
"glob": "^7.1.6", "glob": "^7.1.6",
"google-protobuf": "^3.20.1", "google-protobuf": "^3.20.1",
"hash.js": "^1.1.7", "hash.js": "^1.1.7",
@@ -109,7 +110,7 @@
"devDependencies": { "devDependencies": {
"@octokit/rest": "^18.12.0", "@octokit/rest": "^18.12.0",
"@types/chai": "^4.2.7", "@types/chai": "^4.2.7",
"@types/mocha": "^5.2.7", "@types/mocha": "^10.0.0",
"@types/react-window": "^1.8.5", "@types/react-window": "^1.8.5",
"@xhmikosr/downloader": "^13.0.1", "@xhmikosr/downloader": "^13.0.1",
"chai": "^4.2.0", "chai": "^4.2.0",
@@ -118,19 +119,16 @@
"decompress-tarbz2": "^4.1.1", "decompress-tarbz2": "^4.1.1",
"decompress-targz": "^4.1.1", "decompress-targz": "^4.1.1",
"decompress-unzip": "^4.0.1", "decompress-unzip": "^4.0.1",
"grpc_tools_node_protoc_ts": "^4.1.0", "grpc_tools_node_protoc_ts": "^5.3.3",
"mocha": "^7.0.0", "mocha": "^10.2.0",
"mockdate": "^3.0.5", "mockdate": "^3.0.5",
"moment": "^2.24.0", "moment": "^2.24.0",
"ncp": "^2.0.0", "ncp": "^2.0.0",
"rimraf": "^2.6.1", "rimraf": "^2.6.1"
"shelljs": "^0.8.3",
"uuid": "^3.2.1",
"yargs": "^11.1.0"
}, },
"optionalDependencies": { "optionalDependencies": {
"grpc-tools": "^1.9.0", "grpc-tools": "^1.12.4",
"protoc": "^1.0.4" "@pingghost/protoc": "^1.0.2"
}, },
"mocha": { "mocha": {
"require": [ "require": [
@@ -172,13 +170,17 @@
], ],
"arduino": { "arduino": {
"arduino-cli": { "arduino-cli": {
"version": "0.34.0" "version": "1.0.4"
}, },
"arduino-fwuploader": { "arduino-fwuploader": {
"version": "2.4.0" "version": "2.4.1"
}, },
"arduino-language-server": { "arduino-language-server": {
"version": "0.7.4" "version": {
"owner": "arduino",
"repo": "arduino-language-server",
"commitish": "05ec308"
}
}, },
"clangd": { "clangd": {
"version": "14.0.0" "version": "14.0.0"

View File

@@ -2,7 +2,6 @@
(async () => { (async () => {
const path = require('path'); const path = require('path');
const shell = require('shelljs');
const semver = require('semver'); const semver = require('semver');
const moment = require('moment'); const moment = require('moment');
const downloader = require('./downloader'); const downloader = require('./downloader');
@@ -29,8 +28,8 @@
})(); })();
if (!version) { if (!version) {
shell.echo(`Could not retrieve CLI version info from the 'package.json'.`); console.log(`Could not retrieve CLI version info from the 'package.json'.`);
shell.exit(1); process.exit(1);
} }
const { platform, arch } = process; const { platform, arch } = process;
@@ -71,24 +70,24 @@
} }
})(); })();
if (!suffix) { if (!suffix) {
shell.echo(`The CLI is not available for ${platform} ${arch}.`); console.log(`The CLI is not available for ${platform} ${arch}.`);
shell.exit(1); process.exit(1);
} }
if (semver.valid(version)) { if (semver.valid(version)) {
const url = `https://downloads.arduino.cc/arduino-cli/arduino-cli_${version}_${suffix}`; const url = `https://downloads.arduino.cc/arduino-cli/arduino-cli_${version}_${suffix}`;
shell.echo( console.log(
`📦 Identified released version of the CLI. Downloading version ${version} from '${url}'` `📦 Identified released version of the CLI. Downloading version ${version} from '${url}'`
); );
await downloader.downloadUnzipFile(url, destinationPath, 'arduino-cli'); await downloader.downloadUnzipFile(url, destinationPath, 'arduino-cli');
} else if (moment(version, 'YYYYMMDD', true).isValid()) { } else if (moment(version, 'YYYYMMDD', true).isValid()) {
const url = `https://downloads.arduino.cc/arduino-cli/nightly/arduino-cli_nightly-${version}_${suffix}`; const url = `https://downloads.arduino.cc/arduino-cli/nightly/arduino-cli_nightly-${version}_${suffix}`;
shell.echo( console.log(
`🌙 Identified nightly version of the CLI. Downloading version ${version} from '${url}'` `🌙 Identified nightly version of the CLI. Downloading version ${version} from '${url}'`
); );
await downloader.downloadUnzipFile(url, destinationPath, 'arduino-cli'); await downloader.downloadUnzipFile(url, destinationPath, 'arduino-cli');
} else { } else {
shell.echo(`🔥 Could not interpret 'version': ${version}`); console.log(`🔥 Could not interpret 'version': ${version}`);
shell.exit(1); process.exit(1);
} }
} else { } else {
taskBuildFromGit(version, destinationPath, 'CLI'); taskBuildFromGit(version, destinationPath, 'CLI');

View File

@@ -1,14 +1,18 @@
// @ts-check // @ts-check
// The version to use. // The version to use.
const version = '1.10.0'; const version = '1.10.1';
(async () => { (async () => {
const os = require('node:os'); const os = require('node:os');
const { existsSync, promises: fs } = require('node:fs'); const {
existsSync,
promises: fs,
mkdirSync,
readdirSync,
cpSync,
} = require('node:fs');
const path = require('node:path'); const path = require('node:path');
const shell = require('shelljs');
const { v4 } = require('uuid');
const { exec } = require('./utils'); const { exec } = require('./utils');
const destination = path.join( const destination = path.join(
@@ -20,31 +24,38 @@ const version = '1.10.0';
'Examples' 'Examples'
); );
if (existsSync(destination)) { if (existsSync(destination)) {
shell.echo( console.log(
`Skipping Git checkout of the examples because the repository already exists: ${destination}` `Skipping Git checkout of the examples because the repository already exists: ${destination}`
); );
return; return;
} }
const repository = path.join(os.tmpdir(), `${v4()}-arduino-examples`); const repository = await fs.mkdtemp(
if (shell.mkdir('-p', repository).code !== 0) { path.join(os.tmpdir(), 'arduino-examples-')
shell.exit(1); );
}
exec( exec(
'git', 'git',
['clone', 'https://github.com/arduino/arduino-examples.git', repository], ['clone', 'https://github.com/arduino/arduino-examples.git', repository],
shell { logStdout: true }
); );
exec( exec(
'git', 'git',
['-C', repository, 'checkout', `tags/${version}`, '-b', version], ['-C', repository, 'checkout', `tags/${version}`, '-b', version],
shell { logStdout: true }
); );
shell.mkdir('-p', destination); mkdirSync(destination, { recursive: true });
shell.cp('-fR', path.join(repository, 'examples', '*'), destination); const examplesPath = path.join(repository, 'examples');
const exampleResources = readdirSync(examplesPath);
for (const exampleResource of exampleResources) {
cpSync(
path.join(examplesPath, exampleResource),
path.join(destination, exampleResource),
{ recursive: true }
);
}
const isSketch = async (pathLike) => { const isSketch = async (pathLike) => {
try { try {
@@ -104,5 +115,5 @@ const version = '1.10.0';
JSON.stringify(examples, null, 2), JSON.stringify(examples, null, 2),
{ encoding: 'utf8' } { encoding: 'utf8' }
); );
shell.echo(`Generated output to ${path.join(destination, 'examples.json')}`); console.log(`Generated output to ${path.join(destination, 'examples.json')}`);
})(); })();

View File

@@ -2,7 +2,6 @@
(async () => { (async () => {
const path = require('node:path'); const path = require('node:path');
const shell = require('shelljs');
const semver = require('semver'); const semver = require('semver');
const downloader = require('./downloader'); const downloader = require('./downloader');
const { taskBuildFromGit } = require('./utils'); const { taskBuildFromGit } = require('./utils');
@@ -28,10 +27,10 @@
})(); })();
if (!version) { if (!version) {
shell.echo( console.log(
`Could not retrieve Firmware Uploader version info from the 'package.json'.` `Could not retrieve Firmware Uploader version info from the 'package.json'.`
); );
shell.exit(1); process.exit(1);
} }
const { platform, arch } = process; const { platform, arch } = process;
@@ -51,7 +50,14 @@
const suffix = (() => { const suffix = (() => {
switch (platform) { switch (platform) {
case 'darwin': case 'darwin':
return 'macOS_64bit.tar.gz'; switch (arch) {
case 'arm64':
return 'macOS_ARM64.tar.gz';
case 'x64':
return 'macOS_64bit.tar.gz';
default:
return undefined;
}
case 'win32': case 'win32':
return 'Windows_64bit.zip'; return 'Windows_64bit.zip';
case 'linux': { case 'linux': {
@@ -71,14 +77,14 @@
} }
})(); })();
if (!suffix) { if (!suffix) {
shell.echo( console.log(
`The Firmware Uploader is not available for ${platform} ${arch}.` `The Firmware Uploader is not available for ${platform} ${arch}.`
); );
shell.exit(1); process.exit(1);
} }
if (semver.valid(version)) { if (semver.valid(version)) {
const url = `https://downloads.arduino.cc/arduino-fwuploader/arduino-fwuploader_${version}_${suffix}`; const url = `https://downloads.arduino.cc/arduino-fwuploader/arduino-fwuploader_${version}_${suffix}`;
shell.echo( console.log(
`📦 Identified released version of the Firmware Uploader. Downloading version ${version} from '${url}'` `📦 Identified released version of the Firmware Uploader. Downloading version ${version} from '${url}'`
); );
await downloader.downloadUnzipFile( await downloader.downloadUnzipFile(
@@ -87,8 +93,8 @@
'arduino-fwuploader' 'arduino-fwuploader'
); );
} else { } else {
shell.echo(`🔥 Could not interpret 'version': ${version}`); console.log(`🔥 Could not interpret 'version': ${version}`);
shell.exit(1); process.exit(1);
} }
} else { } else {
taskBuildFromGit(version, destinationPath, 'Firmware Uploader'); taskBuildFromGit(version, destinationPath, 'Firmware Uploader');

View File

@@ -5,7 +5,6 @@
(() => { (() => {
const path = require('path'); const path = require('path');
const shell = require('shelljs');
const downloader = require('./downloader'); const downloader = require('./downloader');
const { goBuildFromGit } = require('./utils'); const { goBuildFromGit } = require('./utils');
@@ -25,20 +24,20 @@
})(); })();
if (!DEFAULT_LS_VERSION) { if (!DEFAULT_LS_VERSION) {
shell.echo( console.log(
`Could not retrieve Arduino Language Server version info from the 'package.json'.` `Could not retrieve Arduino Language Server version info from the 'package.json'.`
); );
shell.exit(1); process.exit(1);
} }
if (!DEFAULT_CLANGD_VERSION) { if (!DEFAULT_CLANGD_VERSION) {
shell.echo( console.log(
`Could not retrieve clangd version info from the 'package.json'.` `Could not retrieve clangd version info from the 'package.json'.`
); );
shell.exit(1); process.exit(1);
} }
const yargs = require('yargs') const yargs = require('@theia/core/shared/yargs')
.option('ls-version', { .option('ls-version', {
alias: 'lv', alias: 'lv',
default: DEFAULT_LS_VERSION, default: DEFAULT_LS_VERSION,
@@ -114,10 +113,10 @@
throw new Error(`Unsupported platform/arch: ${platformArch}.`); throw new Error(`Unsupported platform/arch: ${platformArch}.`);
} }
if (!lsSuffix || !clangdSuffix) { if (!lsSuffix || !clangdSuffix) {
shell.echo( console.log(
`The arduino-language-server is not available for ${platform} ${arch}.` `The arduino-language-server is not available for ${platform} ${arch}.`
); );
shell.exit(1); process.exit(1);
} }
if (typeof lsVersion === 'string') { if (typeof lsVersion === 'string') {

View File

@@ -1,20 +1,19 @@
// @ts-check
const fs = require('fs'); const fs = require('fs');
const path = require('path'); const path = require('path');
const shell = require('shelljs');
const decompress = require('decompress'); const decompress = require('decompress');
const unzip = require('decompress-unzip'); const unzip = require('decompress-unzip');
const untargz = require('decompress-targz'); const untargz = require('decompress-targz');
const untarbz2 = require('decompress-tarbz2'); const untarbz2 = require('decompress-tarbz2');
process.on('unhandledRejection', (reason, _) => { process.on('unhandledRejection', (reason) => {
shell.echo(String(reason)); console.log(String(reason));
shell.exit(1); process.exit(1);
throw reason;
}); });
process.on('uncaughtException', (error) => { process.on('uncaughtException', (error) => {
shell.echo(String(error)); console.log(String(error));
shell.exit(1); process.exit(1);
throw error;
}); });
/** /**
@@ -30,55 +29,42 @@ exports.downloadUnzipFile = async (
force = false force = false
) => { ) => {
if (fs.existsSync(targetFile) && !force) { if (fs.existsSync(targetFile) && !force) {
shell.echo(`Skipping download because file already exists: ${targetFile}`); console.log(`Skipping download because file already exists: ${targetFile}`);
return; return;
} }
if (!fs.existsSync(path.dirname(targetFile))) { fs.mkdirSync(path.dirname(targetFile), { recursive: true });
if (shell.mkdir('-p', path.dirname(targetFile)).code !== 0) {
shell.echo('Could not create new directory.');
shell.exit(1);
}
}
const downloads = path.join(__dirname, '..', 'downloads'); const downloads = path.join(__dirname, '..', 'downloads');
if (shell.rm('-rf', targetFile, downloads).code !== 0) { fs.rmSync(targetFile, { recursive: true, force: true });
shell.exit(1); fs.rmSync(downloads, { recursive: true, force: true });
}
shell.echo(`>>> Downloading from '${url}'...`); console.log(`>>> Downloading from '${url}'...`);
const { default: download } = await import('@xhmikosr/downloader');
const data = await download(url); const data = await download(url);
shell.echo(`<<< Download succeeded.`); console.log(`<<< Download succeeded.`);
shell.echo('>>> Decompressing...'); console.log('>>> Decompressing...');
const files = await decompress(data, downloads, { const files = await decompress(data, downloads, {
plugins: [unzip(), untargz(), untarbz2()], plugins: [unzip(), untargz(), untarbz2()],
}); });
if (files.length === 0) { if (files.length === 0) {
shell.echo('Error ocurred while decompressing the archive.'); console.log('Error ocurred while decompressing the archive.');
shell.exit(1); process.exit(1);
} }
const fileIndex = files.findIndex((f) => f.path.startsWith(filePrefix)); const fileIndex = files.findIndex((f) => f.path.startsWith(filePrefix));
if (fileIndex === -1) { if (fileIndex === -1) {
shell.echo( console.log(
`The downloaded artifact does not contain any file with prefix ${filePrefix}.` `The downloaded artifact does not contain any file with prefix ${filePrefix}.`
); );
shell.exit(1); process.exit(1);
} }
shell.echo('<<< Decompressing succeeded.'); console.log('<<< Decompressing succeeded.');
if ( fs.renameSync(path.join(downloads, files[fileIndex].path), targetFile);
shell.mv('-f', path.join(downloads, files[fileIndex].path), targetFile)
.code !== 0
) {
shell.echo(`Could not move file to target path: ${targetFile}`);
shell.exit(1);
}
if (!fs.existsSync(targetFile)) { if (!fs.existsSync(targetFile)) {
shell.echo(`Could not find file: ${targetFile}`); console.log(`Could not find file: ${targetFile}`);
shell.exit(1); process.exit(1);
} }
shell.echo(`Done: ${targetFile}`); console.log(`Done: ${targetFile}`);
}; };
/** /**
@@ -86,7 +72,7 @@ exports.downloadUnzipFile = async (
* @param targetDir {string} Directory into which to decompress the archive * @param targetDir {string} Directory into which to decompress the archive
* @param targetFile {string} Path to the main file expected after decompressing * @param targetFile {string} Path to the main file expected after decompressing
* @param force {boolean} Whether to download even if the target file exists * @param force {boolean} Whether to download even if the target file exists
* @param decompressOptions {import('decompress').DecompressOptions} * @param decompressOptions {import('decompress').DecompressOptions|undefined} [decompressOptions]
*/ */
exports.downloadUnzipAll = async ( exports.downloadUnzipAll = async (
url, url,
@@ -96,22 +82,16 @@ exports.downloadUnzipAll = async (
decompressOptions = undefined decompressOptions = undefined
) => { ) => {
if (fs.existsSync(targetFile) && !force) { if (fs.existsSync(targetFile) && !force) {
shell.echo(`Skipping download because file already exists: ${targetFile}`); console.log(`Skipping download because file already exists: ${targetFile}`);
return; return;
} }
if (!fs.existsSync(targetDir)) { fs.mkdirSync(targetDir, { recursive: true });
if (shell.mkdir('-p', targetDir).code !== 0) {
shell.echo('Could not create new directory.');
shell.exit(1);
}
}
shell.echo(`>>> Downloading from '${url}'...`); console.log(`>>> Downloading from '${url}'...`);
const { default: download } = await import('@xhmikosr/downloader');
const data = await download(url); const data = await download(url);
shell.echo(`<<< Download succeeded.`); console.log(`<<< Download succeeded.`);
shell.echo('>>> Decompressing...'); console.log('>>> Decompressing...');
let options = { let options = {
plugins: [unzip(), untargz(), untarbz2()], plugins: [unzip(), untargz(), untarbz2()],
}; };
@@ -120,14 +100,27 @@ exports.downloadUnzipAll = async (
} }
const files = await decompress(data, targetDir, options); const files = await decompress(data, targetDir, options);
if (files.length === 0) { if (files.length === 0) {
shell.echo('Error ocurred while decompressing the archive.'); console.log('Error ocurred while decompressing the archive.');
shell.exit(1); process.exit(1);
} }
shell.echo('<<< Decompressing succeeded.'); console.log('<<< Decompressing succeeded.');
if (!fs.existsSync(targetFile)) { if (!fs.existsSync(targetFile)) {
shell.echo(`Could not find file: ${targetFile}`); console.log(`Could not find file: ${targetFile}`);
shell.exit(1); process.exit(1);
} }
shell.echo(`Done: ${targetFile}`); console.log(`Done: ${targetFile}`);
}; };
/**
* @param {string} url
* @returns {Promise<import('node:buffer').Buffer>}
*/
async function download(url) {
const { default: download } = await import('@xhmikosr/downloader');
/** @type {import('node:buffer').Buffer} */
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
const data = await download(url);
return data;
}

View File

@@ -3,22 +3,20 @@
(async () => { (async () => {
const os = require('node:os'); const os = require('node:os');
const path = require('node:path'); const path = require('node:path');
const { mkdirSync, promises: fs, rmSync } = require('node:fs');
const { exec } = require('./utils'); const { exec } = require('./utils');
const glob = require('glob'); const glob = require('glob');
const { v4 } = require('uuid'); const { SemVer, gte, valid: validSemVer } = require('semver');
const shell = require('shelljs'); // Use a node-protoc fork until apple arm32 is supported
const protoc = path.dirname(require('protoc/protoc')); // https://github.com/YePpHa/node-protoc/pull/10
const protoc = path.dirname(require('@pingghost/protoc/protoc'));
const repository = path.join(os.tmpdir(), `${v4()}-arduino-cli`); const repository = await fs.mkdtemp(path.join(os.tmpdir(), 'arduino-cli-'));
if (shell.mkdir('-p', repository).code !== 0) {
shell.exit(1);
}
const { owner, repo, commitish } = (() => { const { owner, repo, commitish } = (() => {
const pkg = require(path.join(__dirname, '..', 'package.json')); const pkg = require(path.join(__dirname, '..', 'package.json'));
if (!pkg) { if (!pkg) {
shell.echo(`Could not parse the 'package.json'.`); console.log(`Could not parse the 'package.json'.`);
shell.exit(1); process.exit(1);
} }
const defaultVersion = { const defaultVersion = {
@@ -48,21 +46,21 @@
// We assume an object with `owner`, `repo`, commitish?` properties. // We assume an object with `owner`, `repo`, commitish?` properties.
const { owner, repo, commitish } = version; const { owner, repo, commitish } = version;
if (!owner) { if (!owner) {
shell.echo(`Could not retrieve 'owner' from ${JSON.stringify(version)}`); console.log(`Could not retrieve 'owner' from ${JSON.stringify(version)}`);
shell.exit(1); process.exit(1);
} }
if (!repo) { if (!repo) {
shell.echo(`Could not retrieve 'repo' from ${JSON.stringify(version)}`); console.log(`Could not retrieve 'repo' from ${JSON.stringify(version)}`);
shell.exit(1); process.exit(1);
} }
return { owner, repo, commitish }; return { owner, repo, commitish };
})(); })();
const url = `https://github.com/${owner}/${repo}.git`; const url = `https://github.com/${owner}/${repo}.git`;
shell.echo(`>>> Cloning repository from '${url}'...`); console.log(`>>> Cloning repository from '${url}'...`);
exec('git', ['clone', url, repository], shell); exec('git', ['clone', url, repository], { logStdout: true });
shell.echo(`<<< Repository cloned.`); console.log(`<<< Repository cloned.`);
const { platform } = process; const { platform } = process;
const resourcesFolder = path.join( const resourcesFolder = path.join(
@@ -76,10 +74,12 @@
resourcesFolder, resourcesFolder,
`arduino-cli${platform === 'win32' ? '.exe' : ''}` `arduino-cli${platform === 'win32' ? '.exe' : ''}`
); );
const versionJson = exec(cli, ['version', '--format', 'json'], shell).trim(); const versionJson = exec(cli, ['version', '--format', 'json'], {
logStdout: true,
}).trim();
if (!versionJson) { if (!versionJson) {
shell.echo(`Could not retrieve the CLI version from ${cli}.`); console.log(`Could not retrieve the CLI version from ${cli}.`);
shell.exit(1); process.exit(1);
} }
// As of today (28.01.2021), the `VersionString` can be one of the followings: // As of today (28.01.2021), the `VersionString` can be one of the followings:
// - `nightly-YYYYMMDD` stands for the nightly build, we use the , the `commitish` from the `package.json` to check out the code. // - `nightly-YYYYMMDD` stands for the nightly build, we use the , the `commitish` from the `package.json` to check out the code.
@@ -96,52 +96,61 @@
} }
*/ */
const versionObject = JSON.parse(versionJson); const versionObject = JSON.parse(versionJson);
const version = versionObject.VersionString; let version = versionObject.VersionString;
if ( if (validSemVer(version)) {
version && // https://github.com/arduino/arduino-cli/pull/2374
!version.startsWith('nightly-') && if (gte(new SemVer(version, { loose: true }), new SemVer('0.35.0-rc.1'))) {
version !== '0.0.0-git' && version = `v${version}`;
version !== 'git-snapshot' }
) { console.log(`>>> Checking out tagged version: '${version}'...`);
shell.echo(`>>> Checking out tagged version: '${version}'...`); exec('git', ['-C', repository, 'fetch', '--all', '--tags'], {
exec('git', ['-C', repository, 'fetch', '--all', '--tags'], shell); logStdout: true,
});
exec( exec(
'git', 'git',
['-C', repository, 'checkout', `tags/${version}`, '-b', version], ['-C', repository, 'checkout', `tags/${version}`, '-b', version],
shell { logStdout: true }
); );
shell.echo(`<<< Checked out tagged version: '${version}'.`); console.log(`<<< Checked out tagged version: '${version}'.`);
} else if (commitish) { } else if (commitish) {
shell.echo( console.log(
`>>> Checking out commitish from 'package.json': '${commitish}'...` `>>> Checking out commitish from 'package.json': '${commitish}'...`
); );
exec('git', ['-C', repository, 'checkout', commitish], shell); exec('git', ['-C', repository, 'checkout', commitish], { logStdout: true });
shell.echo( console.log(
`<<< Checked out commitish from 'package.json': '${commitish}'.` `<<< Checked out commitish from 'package.json': '${commitish}'.`
); );
} else if (versionObject.Commit) { } else if (versionObject.Commit) {
shell.echo( console.log(
`>>> Checking out commitish from the CLI: '${versionObject.Commit}'...` `>>> Checking out commitish from the CLI: '${versionObject.Commit}'...`
); );
exec('git', ['-C', repository, 'checkout', versionObject.Commit], shell); exec('git', ['-C', repository, 'checkout', versionObject.Commit], {
shell.echo( logStdout: true,
});
console.log(
`<<< Checked out commitish from the CLI: '${versionObject.Commit}'.` `<<< Checked out commitish from the CLI: '${versionObject.Commit}'.`
); );
} else { } else {
shell.echo(`WARN: no 'git checkout'. Generating from the HEAD revision.`); console.log(`WARN: no 'git checkout'. Generating from the HEAD revision.`);
} }
shell.echo('>>> Generating TS/JS API from:'); console.log('>>> Generating TS/JS API from:');
exec('git', ['-C', repository, 'rev-parse', '--abbrev-ref', 'HEAD'], shell); exec('git', ['-C', repository, 'rev-parse', '--abbrev-ref', 'HEAD'], {
logStdout: true,
});
const rpc = path.join(repository, 'rpc'); const rpc = path.join(repository, 'rpc');
const out = path.join(__dirname, '..', 'src', 'node', 'cli-protocol'); const out = path.join(__dirname, '..', 'src', 'node', 'cli-protocol');
shell.mkdir('-p', out); // Must wipe the gen output folder. Otherwise, dangling service implementation remain in IDE2 code,
// although it has been removed from the proto file.
// For example, https://github.com/arduino/arduino-cli/commit/50a8bf5c3e61d5b661ccfcd6a055e82eeb510859.
rmSync(out, { recursive: true, maxRetries: 5, force: true });
mkdirSync(out, { recursive: true });
const protos = await new Promise((resolve) => const protos = await new Promise((resolve) =>
glob('**/*.proto', { cwd: rpc }, (error, matches) => { glob('**/*.proto', { cwd: rpc }, (error, matches) => {
if (error) { if (error) {
shell.echo(error.stack ?? error.message); console.log(error.stack ?? error.message);
resolve([]); resolve([]);
return; return;
} }
@@ -149,12 +158,11 @@
}) })
); );
if (!protos || protos.length === 0) { if (!protos || protos.length === 0) {
shell.echo(`Could not find any .proto files under ${rpc}.`); console.log(`Could not find any .proto files under ${rpc}.`);
shell.exit(1); process.exit(1);
} }
// Generate JS code from the `.proto` files. // Generate JS code from the `.proto` files.
exec( exec(
'grpc_tools_node_protoc', 'grpc_tools_node_protoc',
[ [
@@ -164,7 +172,7 @@
rpc, rpc,
...protos, ...protos,
], ],
shell { logStdout: true }
); );
// Generate the `.d.ts` files for JS. // Generate the `.d.ts` files for JS.
@@ -183,8 +191,8 @@
rpc, rpc,
...protos, ...protos,
], ],
shell { logStdout: true }
); );
shell.echo('<<< Generation was successful.'); console.log('<<< Generation was successful.');
})(); })();

View File

@@ -3,24 +3,21 @@
const exec = ( const exec = (
/** @type {string} */ command, /** @type {string} */ command,
/** @type {readonly string[]} */ args, /** @type {readonly string[]} */ args,
/** @type {import('shelljs')|undefined}*/ shell = undefined, /** @type {Partial<import('node:child_process').ExecFileSyncOptionsWithStringEncoding> & { logStdout?: boolean }|undefined} */ options = undefined
/** @type {import('node:child_process').ExecFileSyncOptionsWithStringEncoding|undefined} */ options = undefined
) => { ) => {
try { try {
const stdout = require('node:child_process').execFileSync( const stdout = require('node:child_process').execFileSync(command, args, {
command, encoding: 'utf8',
args, ...(options ?? {}),
options ? options : { encoding: 'utf8' } });
); if (options?.logStdout) {
if (shell) { console.log(stdout.trim());
shell.echo(stdout.trim());
} }
return stdout; return stdout;
} catch (err) { } catch (err) {
if (shell) { console.log(
shell.echo(err instanceof Error ? err.message : String(err)); `Failed to execute ${command} with args: ${JSON.stringify(args)}`
shell.exit(1); );
}
throw err; throw err;
} }
}; };
@@ -59,32 +56,31 @@ function buildFromGit(command, version, destinationPath, taskName) {
const fs = require('node:fs'); const fs = require('node:fs');
const path = require('node:path'); const path = require('node:path');
const temp = require('temp'); const temp = require('temp');
const shell = require('shelljs');
// We assume an object with `owner`, `repo`, commitish?` properties. // We assume an object with `owner`, `repo`, commitish?` properties.
if (typeof version !== 'object') { if (typeof version !== 'object') {
shell.echo( console.log(
`Expected a \`{ owner, repo, commitish }\` object. Got <${version}> instead.` `Expected a \`{ owner, repo, commitish }\` object. Got <${version}> instead.`
); );
} }
const { owner, repo, commitish } = version; const { owner, repo, commitish } = version;
if (!owner) { if (!owner) {
shell.echo(`Could not retrieve 'owner' from ${JSON.stringify(version)}`); console.log(`Could not retrieve 'owner' from ${JSON.stringify(version)}`);
shell.exit(1); process.exit(1);
} }
if (!repo) { if (!repo) {
shell.echo(`Could not retrieve 'repo' from ${JSON.stringify(version)}`); console.log(`Could not retrieve 'repo' from ${JSON.stringify(version)}`);
shell.exit(1); process.exit(1);
} }
const url = `https://github.com/${owner}/${repo}.git`; const url = `https://github.com/${owner}/${repo}.git`;
shell.echo( console.log(
`Building ${taskName} from ${url}. Commitish: ${ `Building ${taskName} from ${url}. Commitish: ${
commitish ? commitish : 'HEAD' commitish ? commitish : 'HEAD'
}` }`
); );
if (fs.existsSync(destinationPath)) { if (fs.existsSync(destinationPath)) {
shell.echo( console.log(
`Skipping the ${taskName} build because it already exists: ${destinationPath}` `Skipping the ${taskName} build because it already exists: ${destinationPath}`
); );
return; return;
@@ -97,48 +93,51 @@ function buildFromGit(command, version, destinationPath, taskName) {
'node', 'node',
'resources' 'resources'
); );
if (shell.mkdir('-p', resourcesFolder).code !== 0) { fs.mkdirSync(resourcesFolder, { recursive: true });
shell.echo('Could not create resources folder.');
shell.exit(1);
}
const tempRepoPath = temp.mkdirSync(); const tempRepoPath = temp.mkdirSync();
shell.echo(`>>> Cloning ${taskName} source to ${tempRepoPath}...`); console.log(`>>> Cloning ${taskName} source to ${tempRepoPath}...`);
exec('git', ['clone', url, tempRepoPath], shell); exec('git', ['clone', url, tempRepoPath], { logStdout: true });
shell.echo(`<<< Cloned ${taskName} repo.`); console.log(`<<< Cloned ${taskName} repo.`);
if (commitish) { if (commitish) {
shell.echo(`>>> Checking out ${commitish}...`); console.log(`>>> Checking out ${commitish}...`);
exec('git', ['-C', tempRepoPath, 'checkout', commitish], shell); exec('git', ['-C', tempRepoPath, 'checkout', commitish], {
shell.echo(`<<< Checked out ${commitish}.`); logStdout: true,
});
console.log(`<<< Checked out ${commitish}.`);
} }
exec('git', ['-C', tempRepoPath, 'rev-parse', '--short', 'HEAD'], shell); exec('git', ['-C', tempRepoPath, 'rev-parse', '--short', 'HEAD'], {
logStdout: true,
});
shell.echo(`>>> Building the ${taskName}...`); console.log(`>>> Building the ${taskName}...`);
exec(command, ['build'], shell, { cwd: tempRepoPath, encoding: 'utf8' }); exec(command, ['build'], {
shell.echo(`<<< Done ${taskName} build.`); cwd: tempRepoPath,
encoding: 'utf8',
logStdout: true,
});
console.log(`<<< Done ${taskName} build.`);
const binName = path.basename(destinationPath); const binName = path.basename(destinationPath);
if (!fs.existsSync(path.join(tempRepoPath, binName))) { if (!fs.existsSync(path.join(tempRepoPath, binName))) {
shell.echo( console.log(
`Could not find the ${taskName} at ${path.join(tempRepoPath, binName)}.` `Could not find the ${taskName} at ${path.join(tempRepoPath, binName)}.`
); );
shell.exit(1); process.exit(1);
} }
const binPath = path.join(tempRepoPath, binName); const binPath = path.join(tempRepoPath, binName);
shell.echo( console.log(
`>>> Copying ${taskName} from ${binPath} to ${destinationPath}...` `>>> Copying ${taskName} from ${binPath} to ${destinationPath}...`
); );
if (shell.cp(binPath, destinationPath).code !== 0) { fs.copyFileSync(binPath, destinationPath);
shell.exit(1); console.log(`<<< Copied the ${taskName}.`);
}
shell.echo(`<<< Copied the ${taskName}.`);
shell.echo(`<<< Verifying ${taskName}...`); console.log(`<<< Verifying ${taskName}...`);
if (!fs.existsSync(destinationPath)) { if (!fs.existsSync(destinationPath)) {
shell.exit(1); process.exit(1);
} }
shell.echo(`>>> Verified ${taskName}.`); console.log(`>>> Verified ${taskName}.`);
} }

View File

@@ -1,5 +1,5 @@
import '../../src/browser/style/index.css'; import '../../src/browser/style/index.css';
import { ContainerModule } from '@theia/core/shared/inversify'; import { Container, ContainerModule } from '@theia/core/shared/inversify';
import { WidgetFactory } from '@theia/core/lib/browser/widget-manager'; import { WidgetFactory } from '@theia/core/lib/browser/widget-manager';
import { CommandContribution } from '@theia/core/lib/common/command'; import { CommandContribution } from '@theia/core/lib/common/command';
import { bindViewContribution } from '@theia/core/lib/browser/shell/view-contribution'; import { bindViewContribution } from '@theia/core/lib/browser/shell/view-contribution';
@@ -180,7 +180,7 @@ import { TabBarRenderer } from './theia/core/tab-bars';
import { EditorCommandContribution } from './theia/editor/editor-command'; import { EditorCommandContribution } from './theia/editor/editor-command';
import { NavigatorTabBarDecorator as TheiaNavigatorTabBarDecorator } from '@theia/navigator/lib/browser/navigator-tab-bar-decorator'; import { NavigatorTabBarDecorator as TheiaNavigatorTabBarDecorator } from '@theia/navigator/lib/browser/navigator-tab-bar-decorator';
import { NavigatorTabBarDecorator } from './theia/navigator/navigator-tab-bar-decorator'; import { NavigatorTabBarDecorator } from './theia/navigator/navigator-tab-bar-decorator';
import { Debug } from './contributions/debug'; import { Debug, DebugDisabledStatusMessageSource } from './contributions/debug';
import { Sketchbook } from './contributions/sketchbook'; import { Sketchbook } from './contributions/sketchbook';
import { DebugFrontendApplicationContribution } from './theia/debug/debug-frontend-application-contribution'; import { DebugFrontendApplicationContribution } from './theia/debug/debug-frontend-application-contribution';
import { DebugFrontendApplicationContribution as TheiaDebugFrontendApplicationContribution } from '@theia/debug/lib/browser/debug-frontend-application-contribution'; import { DebugFrontendApplicationContribution as TheiaDebugFrontendApplicationContribution } from '@theia/debug/lib/browser/debug-frontend-application-contribution';
@@ -271,8 +271,8 @@ import { MonitorModel } from './monitor-model';
import { MonitorManagerProxyClientImpl } from './monitor-manager-proxy-client-impl'; import { MonitorManagerProxyClientImpl } from './monitor-manager-proxy-client-impl';
import { EditorManager as TheiaEditorManager } from '@theia/editor/lib/browser/editor-manager'; import { EditorManager as TheiaEditorManager } from '@theia/editor/lib/browser/editor-manager';
import { EditorManager } from './theia/editor/editor-manager'; import { EditorManager } from './theia/editor/editor-manager';
import { HostedPluginEvents } from './hosted-plugin-events'; import { HostedPluginEvents } from './hosted/hosted-plugin-events';
import { HostedPluginSupport } from './theia/plugin-ext/hosted-plugin'; import { HostedPluginSupportImpl } from './theia/plugin-ext/hosted-plugin';
import { HostedPluginSupport as TheiaHostedPluginSupport } from '@theia/plugin-ext/lib/hosted/browser/hosted-plugin'; import { HostedPluginSupport as TheiaHostedPluginSupport } from '@theia/plugin-ext/lib/hosted/browser/hosted-plugin';
import { Formatter, FormatterPath } from '../common/protocol/formatter'; import { Formatter, FormatterPath } from '../common/protocol/formatter';
import { Format } from './contributions/format'; import { Format } from './contributions/format';
@@ -358,6 +358,20 @@ import { MonacoEditorMenuContribution as TheiaMonacoEditorMenuContribution } fro
import { UpdateArduinoState } from './contributions/update-arduino-state'; import { UpdateArduinoState } from './contributions/update-arduino-state';
import { TerminalFrontendContribution } from './theia/terminal/terminal-frontend-contribution'; import { TerminalFrontendContribution } from './theia/terminal/terminal-frontend-contribution';
import { TerminalFrontendContribution as TheiaTerminalFrontendContribution } from '@theia/terminal/lib/browser/terminal-frontend-contribution'; import { TerminalFrontendContribution as TheiaTerminalFrontendContribution } from '@theia/terminal/lib/browser/terminal-frontend-contribution';
import { SelectionService } from '@theia/core/lib/common/selection-service';
import { CommandService } from '@theia/core/lib/common/command';
import { CorePreferences } from '@theia/core/lib/browser/core-preferences';
import { AutoSelectProgrammer } from './contributions/auto-select-programmer';
import { HostedPluginSupport } from './hosted/hosted-plugin-support';
import { DebugSessionManager as TheiaDebugSessionManager } from '@theia/debug/lib/browser/debug-session-manager';
import { DebugSessionManager } from './theia/debug/debug-session-manager';
import { DebugWidget as TheiaDebugWidget } from '@theia/debug/lib/browser/view/debug-widget';
import { DebugWidget } from './theia/debug/debug-widget';
import { DebugViewModel } from '@theia/debug/lib/browser/view/debug-view-model';
import { DebugSessionWidget } from '@theia/debug/lib/browser/view/debug-session-widget';
import { DebugConfigurationWidget } from './theia/debug/debug-configuration-widget';
import { DebugConfigurationWidget as TheiaDebugConfigurationWidget } from '@theia/debug/lib/browser/view/debug-configuration-widget';
import { DebugToolBar } from '@theia/debug/lib/browser/view/debug-toolbar-widget';
// Hack to fix copy/cut/paste issue after electron version update in Theia. // Hack to fix copy/cut/paste issue after electron version update in Theia.
// https://github.com/eclipse-theia/theia/issues/12487 // https://github.com/eclipse-theia/theia/issues/12487
@@ -451,6 +465,9 @@ export default new ContainerModule((bind, unbind, isBound, rebind) => {
// To be able to track, and update the menu based on the core settings (aka. board details) of the currently selected board. // To be able to track, and update the menu based on the core settings (aka. board details) of the currently selected board.
bind(BoardsDataStore).toSelf().inSingletonScope(); bind(BoardsDataStore).toSelf().inSingletonScope();
bind(FrontendApplicationContribution).toService(BoardsDataStore); bind(FrontendApplicationContribution).toService(BoardsDataStore);
bind(CommandContribution).toService(BoardsDataStore);
bind(StartupTaskProvider).toService(BoardsDataStore); // to inherit the boards config options, programmer, etc in a new window
// Logger for the Arduino daemon // Logger for the Arduino daemon
bind(ILogger) bind(ILogger)
.toDynamicValue((ctx) => { .toDynamicValue((ctx) => {
@@ -750,10 +767,13 @@ export default new ContainerModule((bind, unbind, isBound, rebind) => {
Contribution.configure(bind, CreateCloudCopy); Contribution.configure(bind, CreateCloudCopy);
Contribution.configure(bind, UpdateArduinoState); Contribution.configure(bind, UpdateArduinoState);
Contribution.configure(bind, BoardsDataMenuUpdater); Contribution.configure(bind, BoardsDataMenuUpdater);
Contribution.configure(bind, AutoSelectProgrammer);
bindContributionProvider(bind, StartupTaskProvider); bindContributionProvider(bind, StartupTaskProvider);
bind(StartupTaskProvider).toService(BoardsServiceProvider); // to reuse the boards config in another window bind(StartupTaskProvider).toService(BoardsServiceProvider); // to reuse the boards config in another window
bind(DebugDisabledStatusMessageSource).toService(Debug);
// Disabled the quick-pick customization from Theia when multiple formatters are available. // Disabled the quick-pick customization from Theia when multiple formatters are available.
// Use the default VS Code behavior, and pick the first one. In the IDE2, clang-format has `exclusive` selectors. // Use the default VS Code behavior, and pick the first one. In the IDE2, clang-format has `exclusive` selectors.
bind(MonacoFormattingConflictsContribution).toSelf().inSingletonScope(); bind(MonacoFormattingConflictsContribution).toSelf().inSingletonScope();
@@ -796,10 +816,19 @@ export default new ContainerModule((bind, unbind, isBound, rebind) => {
); );
const iconThemeService = const iconThemeService =
context.container.get<IconThemeService>(IconThemeService); context.container.get<IconThemeService>(IconThemeService);
const selectionService =
context.container.get<SelectionService>(SelectionService);
const commandService =
context.container.get<CommandService>(CommandService);
const corePreferences =
context.container.get<CorePreferences>(CorePreferences);
return new TabBarRenderer( return new TabBarRenderer(
contextMenuRenderer, contextMenuRenderer,
decoratorService, decoratorService,
iconThemeService iconThemeService,
selectionService,
commandService,
corePreferences
); );
}); });
@@ -842,6 +871,28 @@ export default new ContainerModule((bind, unbind, isBound, rebind) => {
// To be able to use a `launch.json` from outside of the workspace. // To be able to use a `launch.json` from outside of the workspace.
bind(DebugConfigurationManager).toSelf().inSingletonScope(); bind(DebugConfigurationManager).toSelf().inSingletonScope();
rebind(TheiaDebugConfigurationManager).toService(DebugConfigurationManager); rebind(TheiaDebugConfigurationManager).toService(DebugConfigurationManager);
// To update the currently selected debug config <select> option when starting a debug session.
bind(DebugSessionManager).toSelf().inSingletonScope();
rebind(TheiaDebugSessionManager).toService(DebugSessionManager);
// Customized debug widget with its customized config <select> to update it programmatically.
bind(WidgetFactory)
.toDynamicValue(({ container }) => ({
id: TheiaDebugWidget.ID,
createWidget: () => {
const child = new Container({ defaultScope: 'Singleton' });
child.parent = container;
child.bind(DebugViewModel).toSelf();
child.bind(DebugToolBar).toSelf();
child.bind(DebugSessionWidget).toSelf();
child.bind(DebugConfigurationWidget).toSelf(); // with the patched select
child // use the customized one in the Theia DI
.bind(TheiaDebugConfigurationWidget)
.toService(DebugConfigurationWidget);
child.bind(DebugWidget).toSelf();
return child.get(DebugWidget);
},
}))
.inSingletonScope();
// To avoid duplicate tabs use deepEqual instead of string equal: https://github.com/eclipse-theia/theia/issues/11309 // To avoid duplicate tabs use deepEqual instead of string equal: https://github.com/eclipse-theia/theia/issues/11309
bind(WidgetManager).toSelf().inSingletonScope(); bind(WidgetManager).toSelf().inSingletonScope();
@@ -970,8 +1021,9 @@ export default new ContainerModule((bind, unbind, isBound, rebind) => {
}) })
.inSingletonScope(); .inSingletonScope();
bind(HostedPluginSupport).toSelf().inSingletonScope(); bind(HostedPluginSupportImpl).toSelf().inSingletonScope();
rebind(TheiaHostedPluginSupport).toService(HostedPluginSupport); bind(HostedPluginSupport).toService(HostedPluginSupportImpl);
rebind(TheiaHostedPluginSupport).toService(HostedPluginSupportImpl);
bind(HostedPluginEvents).toSelf().inSingletonScope(); bind(HostedPluginEvents).toSelf().inSingletonScope();
bind(FrontendApplicationContribution).toService(HostedPluginEvents); bind(FrontendApplicationContribution).toService(HostedPluginEvents);

View File

@@ -54,11 +54,17 @@ export function isMonitorWidgetDockPanel(
return arg === 'bottom' || arg === 'right'; return arg === 'bottom' || arg === 'right';
} }
export const defaultAsyncWorkers = 0 as const;
export const minAsyncWorkers = defaultAsyncWorkers;
export const maxAsyncWorkers = 8 as const;
type StrictPreferenceSchemaProperties<T extends object> = { type StrictPreferenceSchemaProperties<T extends object> = {
[p in keyof T]: PreferenceSchemaProperty; [p in keyof T]: PreferenceSchemaProperty;
}; };
type ArduinoPreferenceSchemaProperties = type ArduinoPreferenceSchemaProperties =
StrictPreferenceSchemaProperties<ArduinoConfiguration> & { 'arduino.window.zoomLevel': PreferenceSchemaProperty }; StrictPreferenceSchemaProperties<ArduinoConfiguration> & {
'arduino.window.zoomLevel': PreferenceSchemaProperty;
};
const properties: ArduinoPreferenceSchemaProperties = { const properties: ArduinoPreferenceSchemaProperties = {
'arduino.language.log': { 'arduino.language.log': {
@@ -77,6 +83,16 @@ const properties: ArduinoPreferenceSchemaProperties = {
), ),
default: false, default: false,
}, },
'arduino.language.asyncWorkers': {
type: 'number',
description: nls.localize(
'arduino/preferences/language.asyncWorkers',
'Number of async workers used by the Arduino Language Server (clangd). Background index also uses this many workers. The minimum value is 0, and the maximum is 8. When it is 0, the language server uses all available cores. The default value is 0.'
),
minimum: minAsyncWorkers,
maximum: maxAsyncWorkers,
default: defaultAsyncWorkers,
},
'arduino.compile.verbose': { 'arduino.compile.verbose': {
type: 'boolean', type: 'boolean',
description: nls.localize( description: nls.localize(
@@ -212,6 +228,14 @@ const properties: ArduinoPreferenceSchemaProperties = {
), ),
default: 'https://api2.arduino.cc/create', default: 'https://api2.arduino.cc/create',
}, },
'arduino.cloud.sharedSpaceID': {
type: 'string',
description: nls.localize(
'arduino/preferences/cloud.sharedSpaceId',
'The ID of the Arduino Cloud shared space to load the sketchbook from. If empty, your private space is selected.'
),
default: '',
},
'arduino.auth.clientID': { 'arduino.auth.clientID': {
type: 'string', type: 'string',
description: nls.localize( description: nls.localize(
@@ -296,6 +320,7 @@ export const ArduinoConfigSchema: PreferenceSchema = {
export interface ArduinoConfiguration { export interface ArduinoConfiguration {
'arduino.language.log': boolean; 'arduino.language.log': boolean;
'arduino.language.realTimeDiagnostics': boolean; 'arduino.language.realTimeDiagnostics': boolean;
'arduino.language.asyncWorkers': number;
'arduino.compile.verbose': boolean; 'arduino.compile.verbose': boolean;
'arduino.compile.experimental': boolean; 'arduino.compile.experimental': boolean;
'arduino.compile.revealRange': ErrorRevealStrategy; 'arduino.compile.revealRange': ErrorRevealStrategy;
@@ -312,6 +337,7 @@ export interface ArduinoConfiguration {
'arduino.cloud.push.warn': boolean; 'arduino.cloud.push.warn': boolean;
'arduino.cloud.pushpublic.warn': boolean; 'arduino.cloud.pushpublic.warn': boolean;
'arduino.cloud.sketchSyncEndpoint': string; 'arduino.cloud.sketchSyncEndpoint': string;
'arduino.cloud.sharedSpaceID': string;
'arduino.auth.clientID': string; 'arduino.auth.clientID': string;
'arduino.auth.domain': string; 'arduino.auth.domain': string;
'arduino.auth.audience': string; 'arduino.auth.audience': string;

View File

@@ -48,16 +48,17 @@ namespace BoardsConfigComponent {
} }
} }
export abstract class Item<T> extends React.Component<{ class Item<T> extends React.Component<{
item: T; item: T;
label: string; label: string;
selected: boolean; selected: boolean;
onClick: (item: T) => void; onClick: (item: T) => void;
missing?: boolean; missing?: boolean;
details?: string; details?: string;
title?: string | ((item: T) => string);
}> { }> {
override render(): React.ReactNode { override render(): React.ReactNode {
const { selected, label, missing, details } = this.props; const { selected, label, missing, details, item } = this.props;
const classNames = ['item']; const classNames = ['item'];
if (selected) { if (selected) {
classNames.push('selected'); classNames.push('selected');
@@ -65,11 +66,15 @@ export abstract class Item<T> extends React.Component<{
if (missing === true) { if (missing === true) {
classNames.push('missing'); classNames.push('missing');
} }
let title = this.props.title ?? `${label}${!details ? '' : details}`;
if (typeof title === 'function') {
title = title(item);
}
return ( return (
<div <div
onClick={this.onClick} onClick={this.onClick}
className={classNames.join(' ')} className={classNames.join(' ')}
title={`${label}${!details ? '' : details}`} title={title}
> >
<div className="label">{label}</div> <div className="label">{label}</div>
{!details ? '' : <div className="details">{details}</div>} {!details ? '' : <div className="details">{details}</div>}
@@ -234,9 +239,20 @@ export class BoardsConfigComponent extends React.Component<
distinctBoards.set(key, board); distinctBoards.set(key, board);
} }
} }
const title = (board: Board.Detailed): string => {
const { details, manuallyInstalled } = board;
let label = board.name;
if (details) {
label += details;
}
if (manuallyInstalled) {
label += nls.localize('arduino/board/inSketchbook', ' (in Sketchbook)');
}
return label;
};
const boardsList = Array.from(distinctBoards.values()).map((board) => ( const boardsList = Array.from(distinctBoards.values()).map((board) => (
<Item<BoardWithPackage> <Item<Board.Detailed>
key={toKey(board)} key={toKey(board)}
item={board} item={board}
label={board.name} label={board.name}
@@ -244,6 +260,7 @@ export class BoardsConfigComponent extends React.Component<
selected={board.selected} selected={board.selected}
onClick={this.selectBoard} onClick={this.selectBoard}
missing={board.missing} missing={board.missing}
title={title}
/> />
)); ));

View File

@@ -1,21 +1,51 @@
import { FrontendApplicationContribution } from '@theia/core/lib/browser/frontend-application'; import { FrontendApplicationContribution } from '@theia/core/lib/browser/frontend-application';
import { LocalStorageService } from '@theia/core/lib/browser/storage-service'; import { FrontendApplicationStateService } from '@theia/core/lib/browser/frontend-application-state';
import { StorageService } from '@theia/core/lib/browser/storage-service';
import type {
Command,
CommandContribution,
CommandRegistry,
} from '@theia/core/lib/common/command';
import { DisposableCollection } from '@theia/core/lib/common/disposable'; import { DisposableCollection } from '@theia/core/lib/common/disposable';
import { Emitter, Event } from '@theia/core/lib/common/event'; import { Emitter, Event } from '@theia/core/lib/common/event';
import { ILogger } from '@theia/core/lib/common/logger'; import { ILogger } from '@theia/core/lib/common/logger';
import { deepClone } from '@theia/core/lib/common/objects'; import { deepClone, deepFreeze } from '@theia/core/lib/common/objects';
import type { Mutable } from '@theia/core/lib/common/types';
import { inject, injectable, named } from '@theia/core/shared/inversify'; import { inject, injectable, named } from '@theia/core/shared/inversify';
import { FQBN } from 'fqbn';
import { import {
BoardDetails, BoardDetails,
BoardsService, BoardsService,
ConfigOption, ConfigOption,
ConfigValue,
Programmer, Programmer,
isBoardIdentifierChangeEvent,
isProgrammer,
sanitizeFqbn,
} from '../../common/protocol'; } from '../../common/protocol';
import { notEmpty } from '../../common/utils'; import { notEmpty } from '../../common/utils';
import type {
StartupTask,
StartupTaskProvider,
} from '../../electron-common/startup-task';
import { NotificationCenter } from '../notification-center'; import { NotificationCenter } from '../notification-center';
import { BoardsServiceProvider } from './boards-service-provider';
export interface SelectConfigOptionParams {
readonly fqbn: string;
readonly optionsToUpdate: readonly Readonly<{
option: string;
selectedValue: string;
}>[];
}
@injectable() @injectable()
export class BoardsDataStore implements FrontendApplicationContribution { export class BoardsDataStore
implements
FrontendApplicationContribution,
StartupTaskProvider,
CommandContribution
{
@inject(ILogger) @inject(ILogger)
@named('store') @named('store')
private readonly logger: ILogger; private readonly logger: ILogger;
@@ -23,46 +53,140 @@ export class BoardsDataStore implements FrontendApplicationContribution {
private readonly boardsService: BoardsService; private readonly boardsService: BoardsService;
@inject(NotificationCenter) @inject(NotificationCenter)
private readonly notificationCenter: NotificationCenter; private readonly notificationCenter: NotificationCenter;
@inject(LocalStorageService) // When `@theia/workspace` is part of the application, the workspace-scoped storage service is the default implementation, and the `StorageService` symbol must be used for the injection.
private readonly storageService: LocalStorageService; // https://github.com/eclipse-theia/theia/blob/ba3722b04ff91eb6a4af6a571c9e263c77cdd8b5/packages/workspace/src/browser/workspace-frontend-module.ts#L97-L98
// In other words, store the data (such as the board configs) per sketch, not per IDE2 installation. https://github.com/arduino/arduino-ide/issues/2240
@inject(StorageService)
private readonly storageService: StorageService;
@inject(BoardsServiceProvider)
private readonly boardsServiceProvider: BoardsServiceProvider;
@inject(FrontendApplicationStateService)
private readonly appStateService: FrontendApplicationStateService;
private readonly onChangedEmitter = new Emitter<string[]>(); private readonly onDidChangeEmitter =
private readonly toDispose = new DisposableCollection(this.onChangedEmitter); new Emitter<BoardsDataStoreChangeEvent>();
private readonly toDispose = new DisposableCollection(
this.onDidChangeEmitter
);
private _selectedBoardData: BoardsDataStoreChange | undefined;
onStart(): void { onStart(): void {
this.toDispose.push( this.toDispose.pushAll([
this.boardsServiceProvider.onBoardsConfigDidChange((event) => {
if (isBoardIdentifierChangeEvent(event)) {
this.updateSelectedBoardData(
event.selectedBoard?.fqbn,
// If the change event comes from toolbar and the FQBN contains custom board options, change the currently selected options
// https://github.com/arduino/arduino-ide/issues/1588
event.reason === 'toolbar'
);
}
}),
this.notificationCenter.onPlatformDidInstall(async ({ item }) => { this.notificationCenter.onPlatformDidInstall(async ({ item }) => {
const dataDidChangePerFqbn: string[] = []; const boardsWithFqbn = item.boards
for (const fqbn of item.boards
.map(({ fqbn }) => fqbn) .map(({ fqbn }) => fqbn)
.filter(notEmpty) .filter(notEmpty);
.filter((fqbn) => !!fqbn)) { const changes: BoardsDataStoreChange[] = [];
for (const fqbn of boardsWithFqbn) {
const key = this.getStorageKey(fqbn); const key = this.getStorageKey(fqbn);
let data = await this.storageService.getData<ConfigOption[]>(key); const storedData =
if (!data || !data.length) { await this.storageService.getData<BoardsDataStore.Data>(key);
const details = await this.getBoardDetailsSafe(fqbn); if (!storedData) {
if (details) { // if no previously value is available for the board, do not update the cache
data = details.configOptions; continue;
if (data.length) { }
await this.storageService.setData(key, data); const details = await this.loadBoardDetails(fqbn);
dataDidChangePerFqbn.push(fqbn); if (details) {
} const data = createDataStoreEntry(details);
} await this.storageService.setData(key, data);
changes.push({ fqbn, data });
} }
} }
if (dataDidChangePerFqbn.length) { if (changes.length) {
this.fireChanged(...dataDidChangePerFqbn); this.fireChanged(...changes);
} }
}) }),
this.onDidChange((event) => {
const selectedFqbn =
this.boardsServiceProvider.boardsConfig.selectedBoard?.fqbn;
if (event.changes.find((change) => change.fqbn === selectedFqbn)) {
this.updateSelectedBoardData(selectedFqbn);
}
}),
]);
Promise.all([
this.boardsServiceProvider.ready,
this.appStateService.reachedState('ready'),
]).then(() =>
this.updateSelectedBoardData(
this.boardsServiceProvider.boardsConfig.selectedBoard?.fqbn
)
); );
} }
private async getSelectedBoardData(
fqbn: string | undefined
): Promise<BoardsDataStoreChange | undefined> {
if (!fqbn) {
return undefined;
} else {
const data = await this.getData(sanitizeFqbn(fqbn));
if (data === BoardsDataStore.Data.EMPTY) {
return undefined;
}
return { fqbn, data };
}
}
private async updateSelectedBoardData(
fqbn: string | undefined,
updateConfigOptions = false
): Promise<void> {
this._selectedBoardData = await this.getSelectedBoardData(fqbn);
if (fqbn && updateConfigOptions) {
const { options } = new FQBN(fqbn);
if (options) {
const optionsToUpdate = Object.entries(options).map(([key, value]) => ({
option: key,
selectedValue: value,
}));
const params = { fqbn, optionsToUpdate };
await this.selectConfigOption(params);
this._selectedBoardData = await this.getSelectedBoardData(fqbn); // reload the updated data
}
}
}
onStop(): void { onStop(): void {
this.toDispose.dispose(); this.toDispose.dispose();
} }
get onChanged(): Event<string[]> { registerCommands(registry: CommandRegistry): void {
return this.onChangedEmitter.event; registry.registerCommand(USE_INHERITED_DATA, {
execute: async (arg: unknown) => {
if (isBoardsDataStoreChange(arg)) {
await this.setData(arg);
this.fireChanged(arg);
}
},
});
}
tasks(): StartupTask[] {
if (!this._selectedBoardData) {
return [];
}
return [
{
command: USE_INHERITED_DATA.id,
args: [this._selectedBoardData],
},
];
}
get onDidChange(): Event<BoardsDataStoreChangeEvent> {
return this.onDidChangeEmitter.event;
} }
async appendConfigToFqbn( async appendConfigToFqbn(
@@ -72,7 +196,7 @@ export class BoardsDataStore implements FrontendApplicationContribution {
return undefined; return undefined;
} }
const { configOptions } = await this.getData(fqbn); const { configOptions } = await this.getData(fqbn);
return ConfigOption.decorate(fqbn, configOptions); return new FQBN(fqbn).withConfigOptions(...configOptions).toString();
} }
async getData(fqbn: string | undefined): Promise<BoardsDataStore.Data> { async getData(fqbn: string | undefined): Promise<BoardsDataStore.Data> {
@@ -81,22 +205,19 @@ export class BoardsDataStore implements FrontendApplicationContribution {
} }
const key = this.getStorageKey(fqbn); const key = this.getStorageKey(fqbn);
let data = await this.storageService.getData< const storedData = await this.storageService.getData<
BoardsDataStore.Data | undefined BoardsDataStore.Data | undefined
>(key, undefined); >(key, undefined);
if (BoardsDataStore.Data.is(data)) { if (BoardsDataStore.Data.is(storedData)) {
return data; return storedData;
} }
const boardDetails = await this.getBoardDetailsSafe(fqbn); const boardDetails = await this.loadBoardDetails(fqbn);
if (!boardDetails) { if (!boardDetails) {
return BoardsDataStore.Data.EMPTY; return BoardsDataStore.Data.EMPTY;
} }
data = { const data = createDataStoreEntry(boardDetails);
configOptions: boardDetails.configOptions,
programmers: boardDetails.programmers,
};
await this.storageService.setData(key, data); await this.storageService.setData(key, data);
return data; return data;
} }
@@ -108,59 +229,68 @@ export class BoardsDataStore implements FrontendApplicationContribution {
fqbn: string; fqbn: string;
selectedProgrammer: Programmer; selectedProgrammer: Programmer;
}): Promise<boolean> { }): Promise<boolean> {
const data = deepClone(await this.getData(fqbn)); const sanitizedFQBN = sanitizeFqbn(fqbn);
const { programmers } = data; const storedData = deepClone(await this.getData(sanitizedFQBN));
const { programmers } = storedData;
if (!programmers.find((p) => Programmer.equals(selectedProgrammer, p))) { if (!programmers.find((p) => Programmer.equals(selectedProgrammer, p))) {
return false; return false;
} }
await this.setData({ const change: BoardsDataStoreChange = {
fqbn, fqbn: sanitizedFQBN,
data: { ...data, selectedProgrammer }, data: { ...storedData, selectedProgrammer },
}); };
this.fireChanged(fqbn); await this.setData(change);
this.fireChanged(change);
return true; return true;
} }
async selectConfigOption({ async selectConfigOption(params: SelectConfigOptionParams): Promise<boolean> {
fqbn, const { fqbn, optionsToUpdate } = params;
option, if (!optionsToUpdate.length) {
selectedValue,
}: {
fqbn: string;
option: string;
selectedValue: string;
}): Promise<boolean> {
const data = deepClone(await this.getData(fqbn));
const { configOptions } = data;
const configOption = configOptions.find((c) => c.option === option);
if (!configOption) {
return false; return false;
} }
let updated = false;
for (const value of configOption.values) { const sanitizedFQBN = sanitizeFqbn(fqbn);
if (value.value === selectedValue) { const mutableData = deepClone(await this.getData(sanitizedFQBN));
(value as any).selected = true; let didChange = false;
updated = true;
} else { for (const { option, selectedValue } of optionsToUpdate) {
(value as any).selected = false; const { configOptions } = mutableData;
const configOption = configOptions.find((c) => c.option === option);
if (configOption) {
const configOptionValueIndex = configOption.values.findIndex(
(configOptionValue) => configOptionValue.value === selectedValue
);
if (configOptionValueIndex >= 0) {
// unselect all
configOption.values
.map((value) => value as Mutable<ConfigValue>)
.forEach((value) => (value.selected = false));
const mutableConfigValue: Mutable<ConfigValue> =
configOption.values[configOptionValueIndex];
// make the new value `selected`
mutableConfigValue.selected = true;
didChange = true;
}
} }
} }
if (!updated) {
if (!didChange) {
return false; return false;
} }
await this.setData({ fqbn, data });
this.fireChanged(fqbn); const change: BoardsDataStoreChange = {
fqbn: sanitizedFQBN,
data: mutableData,
};
await this.setData(change);
this.fireChanged(change);
return true; return true;
} }
protected async setData({ protected async setData(change: BoardsDataStoreChange): Promise<void> {
fqbn, const { fqbn, data } = change;
data,
}: {
fqbn: string;
data: BoardsDataStore.Data;
}): Promise<void> {
const key = this.getStorageKey(fqbn); const key = this.getStorageKey(fqbn);
return this.storageService.setData(key, data); return this.storageService.setData(key, data);
} }
@@ -169,11 +299,9 @@ export class BoardsDataStore implements FrontendApplicationContribution {
return `.arduinoIDE-configOptions-${fqbn}`; return `.arduinoIDE-configOptions-${fqbn}`;
} }
protected async getBoardDetailsSafe( async loadBoardDetails(fqbn: string): Promise<BoardDetails | undefined> {
fqbn: string
): Promise<BoardDetails | undefined> {
try { try {
const details = this.boardsService.getBoardDetails({ fqbn }); const details = await this.boardsService.getBoardDetails({ fqbn });
return details; return details;
} catch (err) { } catch (err) {
if ( if (
@@ -194,8 +322,8 @@ export class BoardsDataStore implements FrontendApplicationContribution {
} }
} }
protected fireChanged(...fqbn: string[]): void { protected fireChanged(...changes: BoardsDataStoreChange[]): void {
this.onChangedEmitter.fire(fqbn); this.onDidChangeEmitter.fire({ changes });
} }
} }
@@ -204,20 +332,86 @@ export namespace BoardsDataStore {
readonly configOptions: ConfigOption[]; readonly configOptions: ConfigOption[];
readonly programmers: Programmer[]; readonly programmers: Programmer[];
readonly selectedProgrammer?: Programmer; readonly selectedProgrammer?: Programmer;
readonly defaultProgrammerId?: string;
} }
export namespace Data { export namespace Data {
export const EMPTY: Data = { export const EMPTY: Data = deepFreeze({
configOptions: [], configOptions: [],
programmers: [], programmers: [],
}; });
export function is(arg: any): arg is Data {
export function is(arg: unknown): arg is Data {
return ( return (
!!arg && typeof arg === 'object' &&
'configOptions' in arg && arg !== null &&
Array.isArray(arg['configOptions']) && Array.isArray((<Data>arg).configOptions) &&
'programmers' in arg && Array.isArray((<Data>arg).programmers) &&
Array.isArray(arg['programmers']) ((<Data>arg).selectedProgrammer === undefined ||
isProgrammer((<Data>arg).selectedProgrammer)) &&
((<Data>arg).defaultProgrammerId === undefined ||
typeof (<Data>arg).defaultProgrammerId === 'string')
); );
} }
} }
} }
export function isEmptyData(data: BoardsDataStore.Data): boolean {
return (
Boolean(!data.configOptions.length) &&
Boolean(!data.programmers.length) &&
Boolean(!data.selectedProgrammer) &&
Boolean(!data.defaultProgrammerId)
);
}
export function findDefaultProgrammer(
programmers: readonly Programmer[],
defaultProgrammerId: string | undefined | BoardsDataStore.Data
): Programmer | undefined {
if (!defaultProgrammerId) {
return undefined;
}
const id =
typeof defaultProgrammerId === 'string'
? defaultProgrammerId
: defaultProgrammerId.defaultProgrammerId;
return programmers.find((p) => p.id === id);
}
function createDataStoreEntry(details: BoardDetails): BoardsDataStore.Data {
const configOptions = details.configOptions.slice();
const programmers = details.programmers.slice();
const { defaultProgrammerId } = details;
const selectedProgrammer = findDefaultProgrammer(
programmers,
defaultProgrammerId
);
const data = {
configOptions,
programmers,
...(selectedProgrammer ? { selectedProgrammer } : {}),
...(defaultProgrammerId ? { defaultProgrammerId } : {}),
};
return data;
}
export interface BoardsDataStoreChange {
readonly fqbn: string;
readonly data: BoardsDataStore.Data;
}
function isBoardsDataStoreChange(arg: unknown): arg is BoardsDataStoreChange {
return (
typeof arg === 'object' &&
arg !== null &&
typeof (<BoardsDataStoreChange>arg).fqbn === 'string' &&
BoardsDataStore.Data.is((<BoardsDataStoreChange>arg).data)
);
}
export interface BoardsDataStoreChangeEvent {
readonly changes: readonly BoardsDataStoreChange[];
}
const USE_INHERITED_DATA: Command = {
id: 'arduino-use-inherited-boards-data',
};

View File

@@ -12,6 +12,7 @@ import { Emitter } from '@theia/core/lib/common/event';
import { ILogger } from '@theia/core/lib/common/logger'; import { ILogger } from '@theia/core/lib/common/logger';
import { MessageService } from '@theia/core/lib/common/message-service'; import { MessageService } from '@theia/core/lib/common/message-service';
import { nls } from '@theia/core/lib/common/nls'; import { nls } from '@theia/core/lib/common/nls';
import { deepClone } from '@theia/core/lib/common/objects';
import { Deferred } from '@theia/core/lib/common/promise-util'; import { Deferred } from '@theia/core/lib/common/promise-util';
import type { Mutable } from '@theia/core/lib/common/types'; import type { Mutable } from '@theia/core/lib/common/types';
import { inject, injectable, optional } from '@theia/core/shared/inversify'; import { inject, injectable, optional } from '@theia/core/shared/inversify';
@@ -21,31 +22,32 @@ import {
} from '@theia/output/lib/browser/output-channel'; } from '@theia/output/lib/browser/output-channel';
import { import {
BoardIdentifier, BoardIdentifier,
boardIdentifierEquals, BoardUserField,
BoardWithPackage,
BoardsConfig, BoardsConfig,
BoardsConfigChangeEvent, BoardsConfigChangeEvent,
BoardsPackage, BoardsPackage,
BoardsService, BoardsService,
BoardUserField,
BoardWithPackage,
DetectedPorts, DetectedPorts,
Port,
PortIdentifier,
boardIdentifierEquals,
emptyBoardsConfig, emptyBoardsConfig,
isBoardIdentifier, isBoardIdentifier,
isBoardIdentifierChangeEvent, isBoardIdentifierChangeEvent,
isPortIdentifier, isPortIdentifier,
isPortIdentifierChangeEvent, isPortIdentifierChangeEvent,
Port,
PortIdentifier,
portIdentifierEquals, portIdentifierEquals,
sanitizeFqbn,
serializePlatformIdentifier, serializePlatformIdentifier,
} from '../../common/protocol'; } from '../../common/protocol';
import { import {
BoardList, BoardList,
BoardListHistory, BoardListHistory,
createBoardList,
EditBoardsConfigActionParams, EditBoardsConfigActionParams,
isBoardListHistory,
SelectBoardsConfigActionParams, SelectBoardsConfigActionParams,
createBoardList,
isBoardListHistory,
} from '../../common/protocol/board-list'; } from '../../common/protocol/board-list';
import type { Defined } from '../../common/types'; import type { Defined } from '../../common/types';
import type { import type {
@@ -104,6 +106,21 @@ type BoardListHistoryUpdateResult =
type BoardToSelect = BoardIdentifier | undefined | 'ignore-board'; type BoardToSelect = BoardIdentifier | undefined | 'ignore-board';
type PortToSelect = PortIdentifier | undefined | 'ignore-port'; type PortToSelect = PortIdentifier | undefined | 'ignore-port';
function sanitizeBoardToSelectFQBN(board: BoardToSelect): BoardToSelect {
if (isBoardIdentifier(board)) {
return sanitizeBoardIdentifierFQBN(board);
}
return board;
}
function sanitizeBoardIdentifierFQBN(board: BoardIdentifier): BoardIdentifier {
if (board.fqbn) {
const copy: Mutable<BoardIdentifier> = deepClone(board);
copy.fqbn = sanitizeFqbn(board.fqbn);
return copy;
}
return board;
}
interface UpdateBoardListHistoryParams { interface UpdateBoardListHistoryParams {
readonly portToSelect: PortToSelect; readonly portToSelect: PortToSelect;
readonly boardToSelect: BoardToSelect; readonly boardToSelect: BoardToSelect;
@@ -136,6 +153,9 @@ export interface BoardListUIActions {
} }
export type BoardListUI = BoardList & BoardListUIActions; export type BoardListUI = BoardList & BoardListUIActions;
export type BoardsConfigChangeEventUI = BoardsConfigChangeEvent &
Readonly<{ reason?: UpdateBoardsConfigReason }>;
@injectable() @injectable()
export class BoardListDumper implements Disposable { export class BoardListDumper implements Disposable {
@inject(OutputChannelManager) @inject(OutputChannelManager)
@@ -190,7 +210,7 @@ export class BoardsServiceProvider
private _ready = new Deferred<void>(); private _ready = new Deferred<void>();
private readonly boardsConfigDidChangeEmitter = private readonly boardsConfigDidChangeEmitter =
new Emitter<BoardsConfigChangeEvent>(); new Emitter<BoardsConfigChangeEventUI>();
readonly onBoardsConfigDidChange = this.boardsConfigDidChangeEmitter.event; readonly onBoardsConfigDidChange = this.boardsConfigDidChangeEmitter.event;
private readonly boardListDidChangeEmitter = new Emitter<BoardListUI>(); private readonly boardListDidChangeEmitter = new Emitter<BoardListUI>();
@@ -353,7 +373,8 @@ export class BoardsServiceProvider
portToSelect !== 'ignore-port' && portToSelect !== 'ignore-port' &&
!portIdentifierEquals(portToSelect, previousSelectedPort); !portIdentifierEquals(portToSelect, previousSelectedPort);
const boardDidChangeEvent = boardDidChange const boardDidChangeEvent = boardDidChange
? { selectedBoard: boardToSelect, previousSelectedBoard } ? // The change event must always contain any custom board options. Hence the board to select is not sanitized.
{ selectedBoard: boardToSelect, previousSelectedBoard }
: undefined; : undefined;
const portDidChangeEvent = portDidChange const portDidChangeEvent = portDidChange
? { selectedPort: portToSelect, previousSelectedPort } ? { selectedPort: portToSelect, previousSelectedPort }
@@ -374,16 +395,31 @@ export class BoardsServiceProvider
return false; return false;
} }
this.maybeUpdateBoardListHistory({ portToSelect, boardToSelect }); // unlike for the board change event, every persistent state must not contain custom board config options in the FQBN
this.maybeUpdateBoardsData({ boardToSelect, reason }); const sanitizedBoardToSelect = sanitizeBoardToSelectFQBN(boardToSelect);
this.maybeUpdateBoardListHistory({
portToSelect,
boardToSelect: sanitizedBoardToSelect,
});
this.maybeUpdateBoardsData({
boardToSelect: sanitizedBoardToSelect,
reason,
});
if (isBoardIdentifierChangeEvent(event)) { if (isBoardIdentifierChangeEvent(event)) {
this._boardsConfig.selectedBoard = event.selectedBoard; this._boardsConfig.selectedBoard = event.selectedBoard
? sanitizeBoardIdentifierFQBN(event.selectedBoard)
: event.selectedBoard;
} }
if (isPortIdentifierChangeEvent(event)) { if (isPortIdentifierChangeEvent(event)) {
this._boardsConfig.selectedPort = event.selectedPort; this._boardsConfig.selectedPort = event.selectedPort;
} }
if (reason) {
event = Object.assign(event, { reason });
}
this.boardsConfigDidChangeEmitter.fire(event); this.boardsConfigDidChangeEmitter.fire(event);
this.refreshBoardList(); this.refreshBoardList();
this.saveState(); this.saveState();

View File

@@ -0,0 +1,123 @@
import type { MaybePromise } from '@theia/core/lib/common/types';
import { inject, injectable } from '@theia/core/shared/inversify';
import {
BoardDetails,
Programmer,
isBoardIdentifierChangeEvent,
} from '../../common/protocol';
import {
BoardsDataStore,
findDefaultProgrammer,
isEmptyData,
} from '../boards/boards-data-store';
import { BoardsServiceProvider } from '../boards/boards-service-provider';
import { Contribution } from './contribution';
/**
* Before CLI 0.35.0-rc.3, there was no `programmer#default` property in the `board details` response.
* This method does the programmer migration in the data store. If there is a programmer selected, it's a noop.
* If no programmer is selected, it forcefully reloads the details from the CLI and updates it in the local storage.
*/
@injectable()
export class AutoSelectProgrammer extends Contribution {
@inject(BoardsServiceProvider)
private readonly boardsServiceProvider: BoardsServiceProvider;
@inject(BoardsDataStore)
private readonly boardsDataStore: BoardsDataStore;
override onStart(): void {
this.boardsServiceProvider.onBoardsConfigDidChange((event) => {
if (isBoardIdentifierChangeEvent(event)) {
this.ensureProgrammerIsSelected();
}
});
}
override onReady(): void {
this.boardsServiceProvider.ready.then(() =>
this.ensureProgrammerIsSelected()
);
}
private async ensureProgrammerIsSelected(): Promise<boolean> {
return ensureProgrammerIsSelected({
fqbn: this.boardsServiceProvider.boardsConfig.selectedBoard?.fqbn,
getData: (fqbn) => this.boardsDataStore.getData(fqbn),
loadBoardDetails: (fqbn) => this.boardsDataStore.loadBoardDetails(fqbn),
selectProgrammer: (arg) => this.boardsDataStore.selectProgrammer(arg),
});
}
}
interface EnsureProgrammerIsSelectedParams {
fqbn: string | undefined;
getData: (fqbn: string | undefined) => MaybePromise<BoardsDataStore.Data>;
loadBoardDetails: (fqbn: string) => MaybePromise<BoardDetails | undefined>;
selectProgrammer(options: {
fqbn: string;
selectedProgrammer: Programmer;
}): MaybePromise<boolean>;
}
export async function ensureProgrammerIsSelected(
params: EnsureProgrammerIsSelectedParams
): Promise<boolean> {
const { fqbn, getData, loadBoardDetails, selectProgrammer } = params;
if (!fqbn) {
return false;
}
console.debug(`Ensuring a programmer is selected for ${fqbn}...`);
const data = await getData(fqbn);
if (isEmptyData(data)) {
// For example, the platform is not installed.
console.debug(`Skipping. No boards data is available for ${fqbn}.`);
return false;
}
if (data.selectedProgrammer) {
console.debug(
`A programmer is already selected for ${fqbn}: '${data.selectedProgrammer.id}'.`
);
return true;
}
let programmer = findDefaultProgrammer(data.programmers, data);
if (programmer) {
// select the programmer if the default info is available
const result = await selectProgrammer({
fqbn,
selectedProgrammer: programmer,
});
if (result) {
console.debug(`Selected '${programmer.id}' programmer for ${fqbn}.`);
return result;
}
}
console.debug(`Reloading board details for ${fqbn}...`);
const reloadedData = await loadBoardDetails(fqbn);
if (!reloadedData) {
console.debug(`Skipping. No board details found for ${fqbn}.`);
return false;
}
if (!reloadedData.programmers.length) {
console.debug(`Skipping. ${fqbn} does not have programmers.`);
return false;
}
programmer = findDefaultProgrammer(reloadedData.programmers, reloadedData);
if (!programmer) {
console.debug(
`Skipping. Could not find a default programmer for ${fqbn}. Programmers were: `
);
return false;
}
const result = await selectProgrammer({
fqbn,
selectedProgrammer: programmer,
});
if (result) {
console.debug(`Selected '${programmer.id}' programmer for ${fqbn}.`);
} else {
console.debug(
`Could not select '${programmer.id}' programmer for ${fqbn}.`
);
}
return result;
}

View File

@@ -35,7 +35,7 @@ export class BoardsDataMenuUpdater extends Contribution {
private readonly toDisposeOnBoardChange = new DisposableCollection(); private readonly toDisposeOnBoardChange = new DisposableCollection();
override onStart(): void { override onStart(): void {
this.boardsDataStore.onChanged(() => this.boardsDataStore.onDidChange(() =>
this.updateMenuActions( this.updateMenuActions(
this.boardsServiceProvider.boardsConfig.selectedBoard this.boardsServiceProvider.boardsConfig.selectedBoard
) )
@@ -87,8 +87,7 @@ export class BoardsDataMenuUpdater extends Contribution {
execute: () => execute: () =>
this.boardsDataStore.selectConfigOption({ this.boardsDataStore.selectConfigOption({
fqbn, fqbn,
option, optionsToUpdate: [{ option, selectedValue: value.value }],
selectedValue: value.value,
}), }),
isToggled: () => value.selected, isToggled: () => value.selected,
}; };

View File

@@ -37,11 +37,15 @@ export class BurnBootloader extends CoreServiceContribution {
'arduino/bootloader/burningBootloader', 'arduino/bootloader/burningBootloader',
'Burning bootloader...' 'Burning bootloader...'
), ),
task: (progressId, coreService) => task: (progressId, coreService, token) =>
coreService.burnBootloader({ coreService.burnBootloader(
...options, {
progressId, ...options,
}), progressId,
},
token
),
cancelable: true,
}); });
this.messageService.info( this.messageService.info(
nls.localize( nls.localize(

View File

@@ -1,83 +1,89 @@
import { ClipboardService } from '@theia/core/lib/browser/clipboard-service';
import {
FrontendApplication,
FrontendApplicationContribution,
} from '@theia/core/lib/browser/frontend-application';
import { FrontendApplicationStateService } from '@theia/core/lib/browser/frontend-application-state';
import {
KeybindingContribution,
KeybindingRegistry,
} from '@theia/core/lib/browser/keybinding';
import { LabelProvider } from '@theia/core/lib/browser/label-provider';
import { OpenerService, open } from '@theia/core/lib/browser/opener-service';
import { Saveable } from '@theia/core/lib/browser/saveable';
import { ApplicationShell } from '@theia/core/lib/browser/shell/application-shell';
import {
TabBarToolbarContribution,
TabBarToolbarRegistry,
} from '@theia/core/lib/browser/shell/tab-bar-toolbar';
import { CancellationToken } from '@theia/core/lib/common/cancellation';
import {
Command,
CommandContribution,
CommandRegistry,
CommandService,
} from '@theia/core/lib/common/command';
import {
Disposable,
DisposableCollection,
} from '@theia/core/lib/common/disposable';
import { EnvVariablesServer } from '@theia/core/lib/common/env-variables';
import { ILogger } from '@theia/core/lib/common/logger';
import {
MenuContribution,
MenuModelRegistry,
} from '@theia/core/lib/common/menu';
import { MessageService } from '@theia/core/lib/common/message-service';
import { MessageType } from '@theia/core/lib/common/message-service-protocol';
import { nls } from '@theia/core/lib/common/nls';
import { MaybePromise, isObject } from '@theia/core/lib/common/types';
import URI from '@theia/core/lib/common/uri';
import { import {
inject, inject,
injectable, injectable,
interfaces, interfaces,
postConstruct, postConstruct,
} from '@theia/core/shared/inversify'; } from '@theia/core/shared/inversify';
import URI from '@theia/core/lib/common/uri';
import { ILogger } from '@theia/core/lib/common/logger';
import {
Disposable,
DisposableCollection,
} from '@theia/core/lib/common/disposable';
import { Saveable } from '@theia/core/lib/browser/saveable';
import { FileService } from '@theia/filesystem/lib/browser/file-service';
import { MaybePromise } from '@theia/core/lib/common/types';
import { LabelProvider } from '@theia/core/lib/browser/label-provider';
import { EditorManager } from '@theia/editor/lib/browser/editor-manager'; import { EditorManager } from '@theia/editor/lib/browser/editor-manager';
import { MessageService } from '@theia/core/lib/common/message-service'; import { FileService } from '@theia/filesystem/lib/browser/file-service';
import { EnvVariablesServer } from '@theia/core/lib/common/env-variables'; import { NotificationManager } from '@theia/messages/lib/browser/notifications-manager';
import { open, OpenerService } from '@theia/core/lib/browser/opener-service'; import { OutputChannelSeverity } from '@theia/output/lib/browser/output-channel';
import { MainMenuManager } from '../../common/main-menu-manager';
import { userAbort } from '../../common/nls';
import { import {
MenuModelRegistry, CoreError,
MenuContribution, CoreService,
} from '@theia/core/lib/common/menu'; FileSystemExt,
ResponseServiceClient,
Sketch,
SketchesService,
} from '../../common/protocol';
import { import {
KeybindingRegistry, ExecuteWithProgress,
KeybindingContribution, UserAbortApplicationError,
} from '@theia/core/lib/browser/keybinding'; } from '../../common/protocol/progressible';
import { import { ArduinoPreferences } from '../arduino-preferences';
TabBarToolbarContribution, import { BoardsDataStore } from '../boards/boards-data-store';
TabBarToolbarRegistry, import { BoardsServiceProvider } from '../boards/boards-service-provider';
} from '@theia/core/lib/browser/shell/tab-bar-toolbar'; import { ConfigServiceClient } from '../config/config-service-client';
import { import { DialogService } from '../dialog-service';
FrontendApplicationContribution,
FrontendApplication,
} from '@theia/core/lib/browser/frontend-application';
import {
Command,
CommandRegistry,
CommandContribution,
CommandService,
} from '@theia/core/lib/common/command';
import { SettingsService } from '../dialogs/settings/settings'; import { SettingsService } from '../dialogs/settings/settings';
import { import {
CurrentSketch, CurrentSketch,
SketchesServiceClientImpl, SketchesServiceClientImpl,
} from '../sketches-service-client-impl'; } from '../sketches-service-client-impl';
import {
SketchesService,
FileSystemExt,
Sketch,
CoreService,
CoreError,
ResponseServiceClient,
} from '../../common/protocol';
import { ArduinoPreferences } from '../arduino-preferences';
import { FrontendApplicationStateService } from '@theia/core/lib/browser/frontend-application-state';
import { nls } from '@theia/core';
import { OutputChannelManager } from '../theia/output/output-channel';
import { ClipboardService } from '@theia/core/lib/browser/clipboard-service';
import { ExecuteWithProgress } from '../../common/protocol/progressible';
import { BoardsServiceProvider } from '../boards/boards-service-provider';
import { BoardsDataStore } from '../boards/boards-data-store';
import { NotificationManager } from '@theia/messages/lib/browser/notifications-manager';
import { MessageType } from '@theia/core/lib/common/message-service-protocol';
import { WorkspaceService } from '../theia/workspace/workspace-service';
import { MainMenuManager } from '../../common/main-menu-manager';
import { ConfigServiceClient } from '../config/config-service-client';
import { ApplicationShell } from '@theia/core/lib/browser/shell/application-shell';
import { DialogService } from '../dialog-service';
import { ApplicationConnectionStatusContribution } from '../theia/core/connection-status-service'; import { ApplicationConnectionStatusContribution } from '../theia/core/connection-status-service';
import { OutputChannelManager } from '../theia/output/output-channel';
import { WorkspaceService } from '../theia/workspace/workspace-service';
export { export {
Command, Command,
CommandRegistry, CommandRegistry,
MenuModelRegistry,
KeybindingRegistry, KeybindingRegistry,
MenuModelRegistry,
Sketch,
TabBarToolbarRegistry, TabBarToolbarRegistry,
URI, URI,
Sketch,
open, open,
}; };
@@ -247,6 +253,12 @@ export abstract class CoreServiceContribution extends SketchContribution {
} }
protected handleError(error: unknown): void { protected handleError(error: unknown): void {
if (isObject(error) && UserAbortApplicationError.is(error)) {
this.outputChannelManager
.getChannel('Arduino')
.appendLine(userAbort, OutputChannelSeverity.Warning);
return;
}
this.tryToastErrorMessage(error); this.tryToastErrorMessage(error);
} }
@@ -293,7 +305,13 @@ export abstract class CoreServiceContribution extends SketchContribution {
protected async doWithProgress<T>(options: { protected async doWithProgress<T>(options: {
progressText: string; progressText: string;
keepOutput?: boolean; keepOutput?: boolean;
task: (progressId: string, coreService: CoreService) => Promise<T>; task: (
progressId: string,
coreService: CoreService,
cancellationToken?: CancellationToken
) => Promise<T>;
// false by default
cancelable?: boolean;
}): Promise<T> { }): Promise<T> {
const toDisposeOnComplete = new DisposableCollection( const toDisposeOnComplete = new DisposableCollection(
this.maybeActivateMonitorWidget() this.maybeActivateMonitorWidget()
@@ -306,8 +324,10 @@ export abstract class CoreServiceContribution extends SketchContribution {
messageService: this.messageService, messageService: this.messageService,
responseService: this.responseService, responseService: this.responseService,
progressText, progressText,
run: ({ progressId }) => task(progressId, this.coreService), run: ({ progressId, cancellationToken }) =>
task(progressId, this.coreService, cancellationToken),
keepOutput, keepOutput,
cancelable: options.cancelable,
}); });
toDisposeOnComplete.dispose(); toDisposeOnComplete.dispose();
return result; return result;

View File

@@ -1,106 +1,173 @@
import { Emitter, Event } from '@theia/core/lib/common/event';
import { MenuModelRegistry } from '@theia/core/lib/common/menu/menu-model-registry';
import { nls } from '@theia/core/lib/common/nls';
import { MaybePromise } from '@theia/core/lib/common/types';
import { inject, injectable } from '@theia/core/shared/inversify'; import { inject, injectable } from '@theia/core/shared/inversify';
import { Event, Emitter } from '@theia/core/lib/common/event'; import { noBoardSelected } from '../../common/nls';
import { HostedPluginSupport } from '@theia/plugin-ext/lib/hosted/browser/hosted-plugin';
import { ArduinoToolbar } from '../toolbar/arduino-toolbar';
import { NotificationCenter } from '../notification-center';
import { import {
Board, BoardDetails,
BoardIdentifier, BoardIdentifier,
BoardsService, BoardsService,
CheckDebugEnabledParams,
ExecutableService, ExecutableService,
SketchRef,
isBoardIdentifierChangeEvent, isBoardIdentifierChangeEvent,
Sketch, isCompileSummary,
} from '../../common/protocol'; } from '../../common/protocol';
import { BoardsDataStore } from '../boards/boards-data-store';
import { BoardsServiceProvider } from '../boards/boards-service-provider'; import { BoardsServiceProvider } from '../boards/boards-service-provider';
import { HostedPluginSupport } from '../hosted/hosted-plugin-support';
import { ArduinoMenus } from '../menu/arduino-menus';
import { NotificationCenter } from '../notification-center';
import { CurrentSketch } from '../sketches-service-client-impl';
import { ArduinoToolbar } from '../toolbar/arduino-toolbar';
import { import {
URI,
Command, Command,
CommandRegistry, CommandRegistry,
SketchContribution, SketchContribution,
TabBarToolbarRegistry, TabBarToolbarRegistry,
URI,
} from './contribution'; } from './contribution';
import { MaybePromise, MenuModelRegistry, nls } from '@theia/core/lib/common';
import { CurrentSketch } from '../sketches-service-client-impl';
import { ArduinoMenus } from '../menu/arduino-menus';
const COMPILE_FOR_DEBUG_KEY = 'arduino-compile-for-debug'; const COMPILE_FOR_DEBUG_KEY = 'arduino-compile-for-debug';
interface StartDebugParams {
/**
* Absolute filesystem path to the Arduino CLI executable.
*/
readonly cliPath: string;
/**
* The the board to debug.
*/
readonly board: Readonly<{ fqbn: string; name?: string }>;
/**
* Absolute filesystem path of the sketch to debug.
*/
readonly sketchPath: string;
/**
* Location where the `launch.json` will be created on the fly before starting every debug session.
* If not defined, it falls back to `sketchPath/.vscode/launch.json`.
*/
readonly launchConfigsDirPath?: string;
/**
* Absolute path to the `arduino-cli.yaml` file. If not specified, it falls back to `~/.arduinoIDE/arduino-cli.yaml`.
*/
readonly cliConfigPath?: string;
/**
* Programmer for the debugging.
*/
readonly programmer?: string;
/**
* Custom progress title to use when getting the debug information from the CLI.
*/
readonly title?: string;
}
type StartDebugResult = boolean;
export const DebugDisabledStatusMessageSource = Symbol(
'DebugDisabledStatusMessageSource'
);
export interface DebugDisabledStatusMessageSource {
/**
* `undefined` if debugging is enabled (for the currently selected board + programmer + config options).
* Otherwise, it's the human readable message why it's disabled.
*/
get message(): string | undefined;
/**
* Emits an event when {@link message} changes.
*/
get onDidChangeMessage(): Event<string | undefined>;
}
@injectable() @injectable()
export class Debug extends SketchContribution { export class Debug
extends SketchContribution
implements DebugDisabledStatusMessageSource
{
@inject(HostedPluginSupport) @inject(HostedPluginSupport)
private readonly hostedPluginSupport: HostedPluginSupport; private readonly hostedPluginSupport: HostedPluginSupport;
@inject(NotificationCenter) @inject(NotificationCenter)
private readonly notificationCenter: NotificationCenter; private readonly notificationCenter: NotificationCenter;
@inject(ExecutableService) @inject(ExecutableService)
private readonly executableService: ExecutableService; private readonly executableService: ExecutableService;
@inject(BoardsService) @inject(BoardsService)
private readonly boardService: BoardsService; private readonly boardService: BoardsService;
@inject(BoardsServiceProvider) @inject(BoardsServiceProvider)
private readonly boardsServiceProvider: BoardsServiceProvider; private readonly boardsServiceProvider: BoardsServiceProvider;
@inject(BoardsDataStore)
private readonly boardsDataStore: BoardsDataStore;
/** /**
* If `undefined`, debugging is enabled. Otherwise, the reason why it's disabled. * If `undefined`, debugging is enabled. Otherwise, the human-readable reason why it's disabled.
*/ */
private _disabledMessages?: string = nls.localize( private _message?: string = noBoardSelected; // Initial pessimism.
'arduino/common/noBoardSelected', private readonly didChangeMessageEmitter = new Emitter<string | undefined>();
'No board selected' readonly onDidChangeMessage = this.didChangeMessageEmitter.event;
); // Initial pessimism.
private disabledMessageDidChangeEmitter = new Emitter<string | undefined>();
private onDisabledMessageDidChange =
this.disabledMessageDidChangeEmitter.event;
private get disabledMessage(): string | undefined { get message(): string | undefined {
return this._disabledMessages; return this._message;
} }
private set disabledMessage(message: string | undefined) { private set message(message: string | undefined) {
this._disabledMessages = message; this._message = message;
this.disabledMessageDidChangeEmitter.fire(this._disabledMessages); this.didChangeMessageEmitter.fire(this._message);
} }
private readonly debugToolbarItem = { private readonly debugToolbarItem = {
id: Debug.Commands.START_DEBUGGING.id, id: Debug.Commands.START_DEBUGGING.id,
command: Debug.Commands.START_DEBUGGING.id, command: Debug.Commands.START_DEBUGGING.id,
tooltip: `${ tooltip: `${
this.disabledMessage this.message
? nls.localize( ? nls.localize(
'arduino/debug/debugWithMessage', 'arduino/debug/debugWithMessage',
'Debug - {0}', 'Debug - {0}',
this.disabledMessage this.message
) )
: Debug.Commands.START_DEBUGGING.label : Debug.Commands.START_DEBUGGING.label
}`, }`,
priority: 3, priority: 3,
onDidChange: this.onDisabledMessageDidChange as Event<void>, onDidChange: this.onDidChangeMessage as Event<void>,
}; };
override onStart(): void { override onStart(): void {
this.onDisabledMessageDidChange( this.onDidChangeMessage(
() => () =>
(this.debugToolbarItem.tooltip = `${ (this.debugToolbarItem.tooltip = `${
this.disabledMessage this.message
? nls.localize( ? nls.localize(
'arduino/debug/debugWithMessage', 'arduino/debug/debugWithMessage',
'Debug - {0}', 'Debug - {0}',
this.disabledMessage this.message
) )
: Debug.Commands.START_DEBUGGING.label : Debug.Commands.START_DEBUGGING.label
}`) }`)
); );
this.boardsServiceProvider.onBoardsConfigDidChange((event) => { this.boardsServiceProvider.onBoardsConfigDidChange((event) => {
if (isBoardIdentifierChangeEvent(event)) { if (isBoardIdentifierChangeEvent(event)) {
this.refreshState(event.selectedBoard); this.updateMessage();
}
});
this.notificationCenter.onPlatformDidInstall(() => this.updateMessage());
this.notificationCenter.onPlatformDidUninstall(() => this.updateMessage());
this.boardsDataStore.onDidChange((event) => {
const selectedFqbn =
this.boardsServiceProvider.boardsConfig.selectedBoard?.fqbn;
if (event.changes.find((change) => change.fqbn === selectedFqbn)) {
this.updateMessage();
}
});
this.commandService.onDidExecuteCommand((event) => {
const { commandId, args } = event;
if (
commandId === 'arduino.languageserver.notifyBuildDidComplete' &&
isCompileSummary(args[0])
) {
this.updateMessage();
} }
}); });
this.notificationCenter.onPlatformDidInstall(() => this.refreshState());
this.notificationCenter.onPlatformDidUninstall(() => this.refreshState());
} }
override onReady(): MaybePromise<void> { override onReady(): void {
this.refreshState(); this.boardsServiceProvider.ready.then(() => this.updateMessage());
} }
override registerCommands(registry: CommandRegistry): void { override registerCommands(registry: CommandRegistry): void {
@@ -108,7 +175,7 @@ export class Debug extends SketchContribution {
execute: () => this.startDebug(), execute: () => this.startDebug(),
isVisible: (widget) => isVisible: (widget) =>
ArduinoToolbar.is(widget) && widget.side === 'left', ArduinoToolbar.is(widget) && widget.side === 'left',
isEnabled: () => !this.disabledMessage, isEnabled: () => !this.message,
}); });
registry.registerCommand(Debug.Commands.TOGGLE_OPTIMIZE_FOR_DEBUG, { registry.registerCommand(Debug.Commands.TOGGLE_OPTIMIZE_FOR_DEBUG, {
execute: () => this.toggleCompileForDebug(), execute: () => this.toggleCompileForDebug(),
@@ -131,94 +198,56 @@ export class Debug extends SketchContribution {
}); });
} }
private async refreshState( private async updateMessage(): Promise<void> {
board: Board | undefined = this.boardsServiceProvider.boardsConfig try {
await this.isDebugEnabled();
this.message = undefined;
} catch (err) {
let message = String(err);
if (err instanceof Error) {
message = err.message;
}
this.message = message;
}
}
private async isDebugEnabled(
board: BoardIdentifier | undefined = this.boardsServiceProvider.boardsConfig
.selectedBoard .selectedBoard
): Promise<void> { ): Promise<string> {
if (!board) { const debugFqbn = await isDebugEnabled(
this.disabledMessage = nls.localize( board,
'arduino/common/noBoardSelected', (fqbn) => this.boardService.getBoardDetails({ fqbn }),
'No board selected' (fqbn) => this.boardsDataStore.getData(fqbn),
); (fqbn) => this.boardsDataStore.appendConfigToFqbn(fqbn),
return; (params) => this.boardService.checkDebugEnabled(params)
} );
const fqbn = board.fqbn; return debugFqbn;
if (!fqbn) {
this.disabledMessage = nls.localize(
'arduino/debug/noPlatformInstalledFor',
"Platform is not installed for '{0}'",
board.name
);
return;
}
const details = await this.boardService.getBoardDetails({ fqbn });
if (!details) {
this.disabledMessage = nls.localize(
'arduino/debug/noPlatformInstalledFor',
"Platform is not installed for '{0}'",
board.name
);
return;
}
const { debuggingSupported } = details;
if (!debuggingSupported) {
this.disabledMessage = nls.localize(
'arduino/debug/debuggingNotSupported',
"Debugging is not supported by '{0}'",
board.name
);
} else {
this.disabledMessage = undefined;
}
} }
private async startDebug( private async startDebug(
board: BoardIdentifier | undefined = this.boardsServiceProvider.boardsConfig board: BoardIdentifier | undefined = this.boardsServiceProvider.boardsConfig
.selectedBoard .selectedBoard,
): Promise<void> { sketch:
if (!board) { | CurrentSketch
return; | undefined = this.sketchServiceClient.tryGetCurrentSketch()
): Promise<StartDebugResult> {
if (!CurrentSketch.isValid(sketch)) {
return false;
} }
const { name, fqbn } = board; const params = await this.createStartDebugParams(board);
if (!fqbn) { if (!params) {
return; return false;
} }
await this.hostedPluginSupport.didStart; await this.hostedPluginSupport.didStart;
const [sketch, executables] = await Promise.all([
this.sketchServiceClient.currentSketch(),
this.executableService.list(),
]);
if (!CurrentSketch.isValid(sketch)) {
return;
}
const ideTempFolderUri = await this.sketchesService.getIdeTempFolderUri(
sketch
);
const [cliPath, sketchPath, configPath] = await Promise.all([
this.fileService.fsPath(new URI(executables.cliUri)),
this.fileService.fsPath(new URI(sketch.uri)),
this.fileService.fsPath(new URI(ideTempFolderUri)),
]);
const config = {
cliPath,
board: {
fqbn,
name,
},
sketchPath,
configPath,
};
try { try {
await this.commandService.executeCommand('arduino.debug.start', config); const result = await this.debug(params);
return Boolean(result);
} catch (err) { } catch (err) {
if (await this.isSketchNotVerifiedError(err, sketch)) { if (await this.isSketchNotVerifiedError(err, sketch)) {
const yes = nls.localize('vscode/extensionsUtils/yes', 'Yes'); const yes = nls.localize('vscode/extensionsUtils/yes', 'Yes');
const answer = await this.messageService.error( const answer = await this.messageService.error(
nls.localize( sketchIsNotCompiled(sketch.name),
'arduino/debug/sketchIsNotCompiled',
"Sketch '{0}' must be verified before starting a debug session. Please verify the sketch and start debugging again. Do you want to verify the sketch now?",
sketch.name
),
yes yes
); );
if (answer === yes) { if (answer === yes) {
@@ -230,6 +259,16 @@ export class Debug extends SketchContribution {
); );
} }
} }
return false;
}
private async debug(
params: StartDebugParams
): Promise<StartDebugResult | undefined> {
return this.commandService.executeCommand<StartDebugResult>(
'arduino.debug.start',
params
);
} }
get compileForDebug(): boolean { get compileForDebug(): boolean {
@@ -237,7 +276,7 @@ export class Debug extends SketchContribution {
return value === 'true'; return value === 'true';
} }
async toggleCompileForDebug(): Promise<void> { private toggleCompileForDebug(): void {
const oldState = this.compileForDebug; const oldState = this.compileForDebug;
const newState = !oldState; const newState = !oldState;
window.localStorage.setItem(COMPILE_FOR_DEBUG_KEY, String(newState)); window.localStorage.setItem(COMPILE_FOR_DEBUG_KEY, String(newState));
@@ -246,7 +285,7 @@ export class Debug extends SketchContribution {
private async isSketchNotVerifiedError( private async isSketchNotVerifiedError(
err: unknown, err: unknown,
sketch: Sketch sketch: SketchRef
): Promise<boolean> { ): Promise<boolean> {
if (err instanceof Error) { if (err instanceof Error) {
try { try {
@@ -260,6 +299,48 @@ export class Debug extends SketchContribution {
} }
return false; return false;
} }
private async createStartDebugParams(
board: BoardIdentifier | undefined
): Promise<StartDebugParams | undefined> {
if (!board || !board.fqbn) {
return undefined;
}
let debugFqbn: string | undefined = undefined;
try {
debugFqbn = await this.isDebugEnabled(board);
} catch {}
if (!debugFqbn) {
return undefined;
}
const [sketch, executables, boardsData] = await Promise.all([
this.sketchServiceClient.currentSketch(),
this.executableService.list(),
this.boardsDataStore.getData(board.fqbn),
]);
if (!CurrentSketch.isValid(sketch)) {
return undefined;
}
const ideTempFolderUri = await this.sketchesService.getIdeTempFolderUri(
sketch
);
const [cliPath, sketchPath, launchConfigsDirPath] = await Promise.all([
this.fileService.fsPath(new URI(executables.cliUri)),
this.fileService.fsPath(new URI(sketch.uri)),
this.fileService.fsPath(new URI(ideTempFolderUri)),
]);
return {
board: { fqbn: debugFqbn, name: board.name },
cliPath,
sketchPath,
launchConfigsDirPath,
programmer: boardsData.selectedProgrammer?.id,
title: nls.localize(
'arduino/debug/getDebugInfo',
'Getting debug info...'
),
};
}
} }
export namespace Debug { export namespace Debug {
export namespace Commands { export namespace Commands {
@@ -284,3 +365,78 @@ export namespace Debug {
}; };
} }
} }
/**
* Resolves with the FQBN to use for the `debug --info --programmer p --fqbn $FQBN` command. Otherwise, rejects.
*
* (non-API)
*/
export async function isDebugEnabled(
board: BoardIdentifier | undefined,
getDetails: (fqbn: string) => MaybePromise<BoardDetails | undefined>,
getData: (fqbn: string) => MaybePromise<BoardsDataStore.Data>,
appendConfigToFqbn: (fqbn: string) => MaybePromise<string | undefined>,
checkDebugEnabled: (params: CheckDebugEnabledParams) => MaybePromise<string>
): Promise<string> {
if (!board) {
throw new Error(noBoardSelected);
}
const { fqbn } = board;
if (!fqbn) {
throw new Error(noPlatformInstalledFor(board.name));
}
const [details, data, fqbnWithConfig] = await Promise.all([
getDetails(fqbn),
getData(fqbn),
appendConfigToFqbn(fqbn),
]);
if (!details) {
throw new Error(noPlatformInstalledFor(board.name));
}
if (!fqbnWithConfig) {
throw new Error(
`Failed to append boards config to the FQBN. Original FQBN was: ${fqbn}`
);
}
const params = {
fqbn: fqbnWithConfig,
programmer: data.selectedProgrammer?.id,
};
try {
const debugFqbn = await checkDebugEnabled(params);
return debugFqbn;
} catch (err) {
throw new Error(debuggingNotSupported(board.name));
}
}
/**
* (non-API)
*/
export function sketchIsNotCompiled(sketchName: string): string {
return nls.localize(
'arduino/debug/sketchIsNotCompiled',
"Sketch '{0}' must be verified before starting a debug session. Please verify the sketch and start debugging again. Do you want to verify the sketch now?",
sketchName
);
}
/**
* (non-API)
*/
export function noPlatformInstalledFor(boardName: string): string {
return nls.localize(
'arduino/debug/noPlatformInstalledFor',
"Platform is not installed for '{0}'",
boardName
);
}
/**
* (non-API)
*/
export function debuggingNotSupported(boardName: string): string {
return nls.localize(
'arduino/debug/debuggingNotSupported',
"Debugging is not supported by '{0}'",
boardName
);
}

View File

@@ -300,8 +300,8 @@ export class LibraryExamples extends Examples {
this.notificationCenter.onLibraryDidUninstall(() => this.update()); this.notificationCenter.onLibraryDidUninstall(() => this.update());
} }
override async onReady(): Promise<void> { override onReady(): void {
this.update(); // no `await` this.boardsServiceProvider.ready.then(() => this.update());
} }
protected override handleBoardChanged(board: Board | undefined): void { protected override handleBoardChanged(board: Board | undefined): void {

View File

@@ -2,7 +2,6 @@ import PQueue from 'p-queue';
import { inject, injectable } from '@theia/core/shared/inversify'; import { inject, injectable } from '@theia/core/shared/inversify';
import URI from '@theia/core/lib/common/uri'; import URI from '@theia/core/lib/common/uri';
import { MonacoEditor } from '@theia/monaco/lib/browser/monaco-editor'; import { MonacoEditor } from '@theia/monaco/lib/browser/monaco-editor';
import { EditorManager } from '@theia/editor/lib/browser';
import { MenuModelRegistry, MenuPath } from '@theia/core/lib/common/menu'; import { MenuModelRegistry, MenuPath } from '@theia/core/lib/common/menu';
import { import {
Disposable, Disposable,
@@ -22,28 +21,25 @@ import { CurrentSketch } from '../sketches-service-client-impl';
@injectable() @injectable()
export class IncludeLibrary extends SketchContribution { export class IncludeLibrary extends SketchContribution {
@inject(CommandRegistry) @inject(CommandRegistry)
protected readonly commandRegistry: CommandRegistry; private readonly commandRegistry: CommandRegistry;
@inject(MenuModelRegistry) @inject(MenuModelRegistry)
protected readonly menuRegistry: MenuModelRegistry; private readonly menuRegistry: MenuModelRegistry;
@inject(MainMenuManager) @inject(MainMenuManager)
protected readonly mainMenuManager: MainMenuManager; private readonly mainMenuManager: MainMenuManager;
@inject(EditorManager)
protected override readonly editorManager: EditorManager;
@inject(NotificationCenter) @inject(NotificationCenter)
protected readonly notificationCenter: NotificationCenter; private readonly notificationCenter: NotificationCenter;
@inject(BoardsServiceProvider) @inject(BoardsServiceProvider)
protected readonly boardsServiceProvider: BoardsServiceProvider; private readonly boardsServiceProvider: BoardsServiceProvider;
@inject(LibraryService) @inject(LibraryService)
protected readonly libraryService: LibraryService; private readonly libraryService: LibraryService;
protected readonly queue = new PQueue({ autoStart: true, concurrency: 1 }); private readonly queue = new PQueue({ autoStart: true, concurrency: 1 });
protected readonly toDispose = new DisposableCollection(); private readonly toDispose = new DisposableCollection();
override onStart(): void { override onStart(): void {
this.boardsServiceProvider.onBoardsConfigDidChange(() => this.boardsServiceProvider.onBoardsConfigDidChange(() =>
@@ -56,8 +52,8 @@ export class IncludeLibrary extends SketchContribution {
this.notificationCenter.onDidReinitialize(() => this.updateMenuActions()); this.notificationCenter.onDidReinitialize(() => this.updateMenuActions());
} }
override async onReady(): Promise<void> { override onReady(): void {
this.updateMenuActions(); this.boardsServiceProvider.ready.then(() => this.updateMenuActions());
} }
override registerMenus(registry: MenuModelRegistry): void { override registerMenus(registry: MenuModelRegistry): void {
@@ -93,7 +89,7 @@ export class IncludeLibrary extends SketchContribution {
}); });
} }
protected async updateMenuActions(): Promise<void> { private async updateMenuActions(): Promise<void> {
return this.queue.add(async () => { return this.queue.add(async () => {
this.toDispose.dispose(); this.toDispose.dispose();
this.mainMenuManager.update(); this.mainMenuManager.update();
@@ -139,7 +135,7 @@ export class IncludeLibrary extends SketchContribution {
}); });
} }
protected registerLibrary( private registerLibrary(
libraryOrPlaceholder: LibraryPackage | string, libraryOrPlaceholder: LibraryPackage | string,
menuPath: MenuPath menuPath: MenuPath
): Disposable { ): Disposable {
@@ -172,7 +168,7 @@ export class IncludeLibrary extends SketchContribution {
); );
} }
protected async includeLibrary(library: LibraryPackage): Promise<void> { private async includeLibrary(library: LibraryPackage): Promise<void> {
const sketch = await this.sketchServiceClient.currentSketch(); const sketch = await this.sketchServiceClient.currentSketch();
if (!CurrentSketch.isValid(sketch)) { if (!CurrentSketch.isValid(sketch)) {
return; return;

View File

@@ -6,40 +6,105 @@ import { inject, injectable } from '@theia/core/shared/inversify';
import { Mutex } from 'async-mutex'; import { Mutex } from 'async-mutex';
import { import {
ArduinoDaemon, ArduinoDaemon,
assertSanitizedFqbn,
BoardIdentifier, BoardIdentifier,
BoardsService, BoardsService,
ExecutableService, ExecutableService,
isBoardIdentifierChangeEvent, isBoardIdentifierChangeEvent,
sanitizeFqbn, sanitizeFqbn,
} from '../../common/protocol'; } from '../../common/protocol';
import { CurrentSketch } from '../sketches-service-client-impl'; import {
import { BoardsServiceProvider } from '../boards/boards-service-provider'; defaultAsyncWorkers,
import { HostedPluginEvents } from '../hosted-plugin-events'; maxAsyncWorkers,
import { NotificationCenter } from '../notification-center'; minAsyncWorkers,
import { SketchContribution, URI } from './contribution'; } from '../arduino-preferences';
import { BoardsDataStore } from '../boards/boards-data-store'; import { BoardsDataStore } from '../boards/boards-data-store';
import { BoardsServiceProvider } from '../boards/boards-service-provider';
import { HostedPluginEvents } from '../hosted/hosted-plugin-events';
import { NotificationCenter } from '../notification-center';
import { CurrentSketch } from '../sketches-service-client-impl';
import { SketchContribution, URI } from './contribution';
interface DaemonAddress {
/**
* The host where the Arduino CLI daemon is available.
*/
readonly hostname: string;
/**
* The port where the Arduino CLI daemon is listening.
*/
readonly port: number;
/**
* The [id](https://arduino.github.io/arduino-cli/latest/rpc/commands/#instance) of the initialized core Arduino client instance.
*/
readonly instance: number;
}
interface StartLanguageServerParams {
/**
* Absolute filesystem path to the Arduino Language Server executable.
*/
readonly lsPath: string;
/**
* The hostname and the port for the gRPC channel connecting to the Arduino CLI daemon.
* The `instance` number is for the initialized core Arduino client.
*/
readonly daemonAddress: DaemonAddress;
/**
* Absolute filesystem path to [`clangd`](https://clangd.llvm.org/).
*/
readonly clangdPath: string;
/**
* The board is relevant to start a specific "flavor" of the language.
*/
readonly board: { fqbn: string; name?: string };
/**
* `true` if the LS should generate the log files into the default location. The default location is the `cwd` of the process.
* It's very often the same as the workspace root of the IDE, aka the sketch folder.
* When it is a string, it is the absolute filesystem path to the folder to generate the log files.
* If `string`, but the path is inaccessible, the log files will be generated into the default location.
*/
readonly log?: boolean | string;
/**
* Optional `env` for the language server process.
*/
readonly env?: NodeJS.ProcessEnv;
/**
* Additional flags for the Arduino Language server process.
*/
readonly flags?: readonly string[];
/**
* Set to `true`, to enable `Diagnostics`.
*/
readonly realTimeDiagnostics?: boolean;
/**
* If `true`, the logging is not forwarded to the _Output_ view via the language client.
*/
readonly silentOutput?: boolean;
/**
* Number of async workers used by `clangd`. Background index also uses this many workers. If `0`, `clangd` uses all available cores. It's `0` by default.
*/
readonly jobs?: number;
}
/**
* The FQBN the language server runs with or `undefined` if it could not start.
*/
type StartLanguageServerResult = string | undefined;
@injectable() @injectable()
export class InoLanguage extends SketchContribution { export class InoLanguage extends SketchContribution {
@inject(HostedPluginEvents) @inject(HostedPluginEvents)
private readonly hostedPluginEvents: HostedPluginEvents; private readonly hostedPluginEvents: HostedPluginEvents;
@inject(ExecutableService) @inject(ExecutableService)
private readonly executableService: ExecutableService; private readonly executableService: ExecutableService;
@inject(ArduinoDaemon) @inject(ArduinoDaemon)
private readonly daemon: ArduinoDaemon; private readonly daemon: ArduinoDaemon;
@inject(BoardsService) @inject(BoardsService)
private readonly boardsService: BoardsService; private readonly boardsService: BoardsService;
@inject(BoardsServiceProvider) @inject(BoardsServiceProvider)
private readonly boardsServiceProvider: BoardsServiceProvider; private readonly boardsServiceProvider: BoardsServiceProvider;
@inject(NotificationCenter) @inject(NotificationCenter)
private readonly notificationCenter: NotificationCenter; private readonly notificationCenter: NotificationCenter;
@inject(BoardsDataStore) @inject(BoardsDataStore)
private readonly boardDataStore: BoardsDataStore; private readonly boardDataStore: BoardsDataStore;
@@ -80,6 +145,7 @@ export class InoLanguage extends SketchContribution {
switch (preferenceName) { switch (preferenceName) {
case 'arduino.language.log': case 'arduino.language.log':
case 'arduino.language.realTimeDiagnostics': case 'arduino.language.realTimeDiagnostics':
case 'arduino.language.asyncWorkers':
forceRestart(); forceRestart();
} }
} }
@@ -90,30 +156,30 @@ export class InoLanguage extends SketchContribution {
this.notificationCenter.onPlatformDidInstall(() => forceRestart()), this.notificationCenter.onPlatformDidInstall(() => forceRestart()),
this.notificationCenter.onPlatformDidUninstall(() => forceRestart()), this.notificationCenter.onPlatformDidUninstall(() => forceRestart()),
this.notificationCenter.onDidReinitialize(() => forceRestart()), this.notificationCenter.onDidReinitialize(() => forceRestart()),
this.boardDataStore.onChanged((dataChangePerFqbn) => { this.boardDataStore.onDidChange((event) => {
if (this.languageServerFqbn) { if (this.languageServerFqbn) {
const sanitizedFqbn = sanitizeFqbn(this.languageServerFqbn); const sanitizedFQBN = sanitizeFqbn(this.languageServerFqbn);
if (!sanitizeFqbn) { // The incoming FQBNs might contain custom boards configs, sanitize them before the comparison.
throw new Error( // https://github.com/arduino/arduino-ide/pull/2113#pullrequestreview-1499998328
`Failed to sanitize the FQBN of the running language server. FQBN with the board settings was: ${this.languageServerFqbn}` const matchingChange = event.changes.find(
); (change) => sanitizedFQBN === sanitizeFqbn(change.fqbn)
}
const matchingFqbn = dataChangePerFqbn.find(
(fqbn) => sanitizedFqbn === fqbn
); );
const { boardsConfig } = this.boardsServiceProvider; const { boardsConfig } = this.boardsServiceProvider;
if ( if (
matchingFqbn && matchingChange &&
boardsConfig.selectedBoard?.fqbn === matchingFqbn boardsConfig.selectedBoard?.fqbn === matchingChange.fqbn
) { ) {
start(boardsConfig.selectedBoard); start(boardsConfig.selectedBoard);
} }
} }
}), }),
]); ]);
this.boardsServiceProvider.ready.then(() => Promise.all([
start(this.boardsServiceProvider.boardsConfig.selectedBoard) this.boardsServiceProvider.ready,
); this.preferences.ready,
]).then(() => {
start(this.boardsServiceProvider.boardsConfig.selectedBoard);
});
} }
onStop(): void { onStop(): void {
@@ -126,7 +192,7 @@ export class InoLanguage extends SketchContribution {
forceStart = false forceStart = false
): Promise<void> { ): Promise<void> {
const port = await this.daemon.tryGetPort(); const port = await this.daemon.tryGetPort();
if (!port) { if (typeof port !== 'number') {
return; return;
} }
const release = await this.languageServerStartMutex.acquire(); const release = await this.languageServerStartMutex.acquire();
@@ -158,7 +224,6 @@ export class InoLanguage extends SketchContribution {
} }
return; return;
} }
assertSanitizedFqbn(fqbn);
const fqbnWithConfig = await this.boardDataStore.appendConfigToFqbn(fqbn); const fqbnWithConfig = await this.boardDataStore.appendConfigToFqbn(fqbn);
if (!fqbnWithConfig) { if (!fqbnWithConfig) {
throw new Error( throw new Error(
@@ -169,11 +234,16 @@ export class InoLanguage extends SketchContribution {
// NOOP // NOOP
return; return;
} }
this.logger.info(`Starting language server: ${fqbnWithConfig}`);
const log = this.preferences.get('arduino.language.log'); const log = this.preferences.get('arduino.language.log');
const realTimeDiagnostics = this.preferences.get( const realTimeDiagnostics = this.preferences.get(
'arduino.language.realTimeDiagnostics' 'arduino.language.realTimeDiagnostics'
); );
const jobs = this.getAsyncWorkersPreferenceSafe();
this.logger.info(
`Starting language server: ${fqbnWithConfig}${
jobs ? ` (async worker count: ${jobs})` : ''
}`
);
let currentSketchPath: string | undefined = undefined; let currentSketchPath: string | undefined = undefined;
if (log) { if (log) {
const currentSketch = await this.sketchServiceClient.currentSketch(); const currentSketch = await this.sketchServiceClient.currentSketch();
@@ -197,22 +267,23 @@ export class InoLanguage extends SketchContribution {
); );
toDisposeOnRelease.push(Disposable.create(() => clearTimeout(timer))); toDisposeOnRelease.push(Disposable.create(() => clearTimeout(timer)));
}), }),
this.commandService.executeCommand<string>( this.start({
'arduino.languageserver.start', lsPath,
{ daemonAddress: {
lsPath, hostname: 'localhost',
cliDaemonAddr: `localhost:${port}`, port,
clangdPath, instance: 1, // TODO: get it from the backend
log: currentSketchPath ? currentSketchPath : log, },
cliDaemonInstance: '1', clangdPath,
board: { log: currentSketchPath ? currentSketchPath : log,
fqbn: fqbnWithConfig, board: {
name: name ? `"${name}"` : undefined, fqbn: fqbnWithConfig,
}, name,
realTimeDiagnostics, },
silentOutput: true, realTimeDiagnostics,
} silentOutput: true,
), jobs,
}),
]); ]);
} catch (e) { } catch (e) {
console.log(`Failed to start language server. Original FQBN: ${fqbn}`, e); console.log(`Failed to start language server. Original FQBN: ${fqbn}`, e);
@@ -222,4 +293,28 @@ export class InoLanguage extends SketchContribution {
release(); release();
} }
} }
// The Theia preference UI validation is bogus.
// To restrict the number of jobs to a valid value.
private getAsyncWorkersPreferenceSafe(): number {
const jobs = this.preferences.get(
'arduino.language.asyncWorkers',
defaultAsyncWorkers
);
if (jobs < minAsyncWorkers) {
return minAsyncWorkers;
}
if (jobs > maxAsyncWorkers) {
return maxAsyncWorkers;
}
return jobs;
}
private async start(
params: StartLanguageServerParams
): Promise<StartLanguageServerResult | undefined> {
return this.commandService.executeCommand<StartLanguageServerResult>(
'arduino.languageserver.start',
params
);
}
} }

View File

@@ -3,10 +3,12 @@ import { NavigatableWidget } from '@theia/core/lib/browser/navigatable';
import { Saveable } from '@theia/core/lib/browser/saveable'; import { Saveable } from '@theia/core/lib/browser/saveable';
import { ApplicationShell } from '@theia/core/lib/browser/shell/application-shell'; import { ApplicationShell } from '@theia/core/lib/browser/shell/application-shell';
import { WindowService } from '@theia/core/lib/browser/window/window-service'; import { WindowService } from '@theia/core/lib/browser/window/window-service';
import { ApplicationError } from '@theia/core/lib/common/application-error';
import { nls } from '@theia/core/lib/common/nls'; import { nls } from '@theia/core/lib/common/nls';
import { inject, injectable } from '@theia/core/shared/inversify'; import { inject, injectable } from '@theia/core/shared/inversify';
import { EditorManager } from '@theia/editor/lib/browser/editor-manager'; import { EditorManager } from '@theia/editor/lib/browser/editor-manager';
import { WorkspaceInput } from '@theia/workspace/lib/browser/workspace-service'; import { WorkspaceInput } from '@theia/workspace/lib/browser/workspace-service';
import { SketchesError } from '../../common/protocol';
import { StartupTasks } from '../../electron-common/startup-task'; import { StartupTasks } from '../../electron-common/startup-task';
import { ArduinoMenus } from '../menu/arduino-menus'; import { ArduinoMenus } from '../menu/arduino-menus';
import { CurrentSketch } from '../sketches-service-client-impl'; import { CurrentSketch } from '../sketches-service-client-impl';
@@ -35,7 +37,29 @@ export class SaveAsSketch extends CloudSketchContribution {
override registerCommands(registry: CommandRegistry): void { override registerCommands(registry: CommandRegistry): void {
registry.registerCommand(SaveAsSketch.Commands.SAVE_AS_SKETCH, { registry.registerCommand(SaveAsSketch.Commands.SAVE_AS_SKETCH, {
execute: (args) => this.saveAs(args), execute: async (args) => {
try {
return await this.saveAs(args);
} catch (err) {
let message = String(err);
if (ApplicationError.is(err)) {
if (SketchesError.SketchAlreadyContainsThisFile.is(err)) {
message = nls.localize(
'arduino/sketch/sketchAlreadyContainsThisFileMessage',
'Failed to save sketch "{0}" as "{1}". {2}',
err.data.sourceSketchName,
err.data.targetSketchName,
err.message
);
} else {
message = err.message;
}
} else if (err instanceof Error) {
message = err.message;
}
this.messageService.error(message);
}
},
}); });
} }
@@ -58,13 +82,14 @@ export class SaveAsSketch extends CloudSketchContribution {
* Resolves `true` if the sketch was successfully saved as something. * Resolves `true` if the sketch was successfully saved as something.
*/ */
private async saveAs( private async saveAs(
{ params = SaveAsSketch.Options.DEFAULT
): Promise<boolean> {
const {
execOnlyIfTemp, execOnlyIfTemp,
openAfterMove, openAfterMove,
wipeOriginal, wipeOriginal,
markAsRecentlyOpened, markAsRecentlyOpened,
}: SaveAsSketch.Options = SaveAsSketch.Options.DEFAULT } = params;
): Promise<boolean> {
assertConnectedToBackend({ assertConnectedToBackend({
connectionStatusService: this.connectionStatusService, connectionStatusService: this.connectionStatusService,
messageService: this.messageService, messageService: this.messageService,

View File

@@ -19,16 +19,18 @@ export class SelectedBoard extends Contribution {
private readonly boardsServiceProvider: BoardsServiceProvider; private readonly boardsServiceProvider: BoardsServiceProvider;
override onStart(): void { override onStart(): void {
this.boardsServiceProvider.onBoardListDidChange(() => this.boardsServiceProvider.onBoardListDidChange((boardList) =>
this.update(this.boardsServiceProvider.boardList) this.update(boardList)
); );
} }
override onReady(): void { override onReady(): void {
this.update(this.boardsServiceProvider.boardList); this.boardsServiceProvider.ready.then(() => this.update());
} }
private update(boardList: BoardList): void { private update(
boardList: BoardList = this.boardsServiceProvider.boardList
): void {
const { selectedBoard, selectedPort } = boardList.boardsConfig; const { selectedBoard, selectedPort } = boardList.boardsConfig;
this.statusBar.setElement('arduino-selected-board', { this.statusBar.setElement('arduino-selected-board', {
alignment: StatusBarAlignment.RIGHT, alignment: StatusBarAlignment.RIGHT,

View File

@@ -1,7 +1,7 @@
import { DisposableCollection } from '@theia/core/lib/common/disposable'; import { DisposableCollection } from '@theia/core/lib/common/disposable';
import URI from '@theia/core/lib/common/uri'; import URI from '@theia/core/lib/common/uri';
import { inject, injectable } from '@theia/core/shared/inversify'; import { inject, injectable } from '@theia/core/shared/inversify';
import { HostedPluginSupport } from '@theia/plugin-ext/lib/hosted/browser/hosted-plugin'; import { HostedPluginSupport } from '../hosted/hosted-plugin-support';
import type { ArduinoState } from 'vscode-arduino-api'; import type { ArduinoState } from 'vscode-arduino-api';
import { import {
BoardsService, BoardsService,
@@ -21,7 +21,10 @@ import { BoardsServiceProvider } from '../boards/boards-service-provider';
import { CurrentSketch } from '../sketches-service-client-impl'; import { CurrentSketch } from '../sketches-service-client-impl';
import { SketchContribution } from './contribution'; import { SketchContribution } from './contribution';
interface UpdateStateParams<T extends ArduinoState> { /**
* (non-API) exported for tests
*/
export interface UpdateStateParams<T extends ArduinoState = ArduinoState> {
readonly key: keyof T; readonly key: keyof T;
readonly value: T[keyof T]; readonly value: T[keyof T];
} }
@@ -65,10 +68,13 @@ export class UpdateArduinoState extends SketchContribution {
this.updateCompileSummary(args[0]); this.updateCompileSummary(args[0]);
} }
}), }),
this.boardsDataStore.onChanged((fqbn) => { this.boardsDataStore.onDidChange((event) => {
const selectedFqbn = const selectedFqbn =
this.boardsServiceProvider.boardsConfig.selectedBoard?.fqbn; this.boardsServiceProvider.boardsConfig.selectedBoard?.fqbn;
if (selectedFqbn && fqbn.includes(selectedFqbn)) { if (
selectedFqbn &&
event.changes.find((change) => change.fqbn === selectedFqbn)
) {
this.updateBoardDetails(selectedFqbn); this.updateBoardDetails(selectedFqbn);
} }
}), }),
@@ -76,7 +82,9 @@ export class UpdateArduinoState extends SketchContribution {
} }
override onReady(): void { override onReady(): void {
this.updateBoardsConfig(this.boardsServiceProvider.boardsConfig); // TODO: verify! this.boardsServiceProvider.ready.then(() => {
this.updateBoardsConfig(this.boardsServiceProvider.boardsConfig);
});
this.updateSketchPath(this.sketchServiceClient.tryGetCurrentSketch()); this.updateSketchPath(this.sketchServiceClient.tryGetCurrentSketch());
this.updateUserDirPath(this.configService.tryGetSketchDirUri()); this.updateUserDirPath(this.configService.tryGetSketchDirUri());
this.updateDataDirPath(this.configService.tryGetDataDirUri()); this.updateDataDirPath(this.configService.tryGetDataDirUri());

View File

@@ -45,10 +45,7 @@ export namespace UploadFirmware {
export namespace Commands { export namespace Commands {
export const OPEN: Command = { export const OPEN: Command = {
id: 'arduino-upload-firmware-open', id: 'arduino-upload-firmware-open',
label: nls.localize( label: nls.localize('arduino/firmware/updater', 'Firmware Updater'),
'arduino/firmware/updater',
'Firmware Updater'
),
category: 'Arduino', category: 'Arduino',
}; };
} }

View File

@@ -1,7 +1,8 @@
import { Emitter } from '@theia/core/lib/common/event'; import { Emitter } from '@theia/core/lib/common/event';
import { nls } from '@theia/core/lib/common/nls'; import { nls } from '@theia/core/lib/common/nls';
import { inject, injectable } from '@theia/core/shared/inversify'; import { inject, injectable } from '@theia/core/shared/inversify';
import { CoreService, sanitizeFqbn } from '../../common/protocol'; import { FQBN } from 'fqbn';
import { CoreService } from '../../common/protocol';
import { ArduinoMenus } from '../menu/arduino-menus'; import { ArduinoMenus } from '../menu/arduino-menus';
import { CurrentSketch } from '../sketches-service-client-impl'; import { CurrentSketch } from '../sketches-service-client-impl';
import { ArduinoToolbar } from '../toolbar/arduino-toolbar'; import { ArduinoToolbar } from '../toolbar/arduino-toolbar';
@@ -126,6 +127,7 @@ export class UploadSketch extends CoreServiceContribution {
usingProgrammer, usingProgrammer,
verifyOptions verifyOptions
); );
if (!uploadOptions) { if (!uploadOptions) {
return; return;
} }
@@ -136,10 +138,37 @@ export class UploadSketch extends CoreServiceContribution {
const uploadResponse = await this.doWithProgress({ const uploadResponse = await this.doWithProgress({
progressText: nls.localize('arduino/sketch/uploading', 'Uploading...'), progressText: nls.localize('arduino/sketch/uploading', 'Uploading...'),
task: (progressId, coreService) => task: async (progressId, coreService, token) => {
coreService.upload({ ...uploadOptions, progressId }), try {
return await coreService.upload(
{ ...uploadOptions, progressId },
token
);
} catch (err) {
if (err.code === 4005) {
const uploadWithProgrammerOptions = await this.uploadOptions(
true,
verifyOptions
);
if (uploadWithProgrammerOptions) {
return coreService.upload(
{ ...uploadWithProgrammerOptions, progressId },
token
);
}
} else {
throw err;
}
}
},
keepOutput: true, keepOutput: true,
cancelable: true,
}); });
if (!uploadResponse) {
return;
}
// the port update is NOOP if nothing has changed // the port update is NOOP if nothing has changed
this.boardsServiceProvider.updateConfig(uploadResponse.portAfterUpload); this.boardsServiceProvider.updateConfig(uploadResponse.portAfterUpload);
@@ -172,7 +201,11 @@ export class UploadSketch extends CoreServiceContribution {
const [fqbn, { selectedProgrammer: programmer }, verify, verbose] = const [fqbn, { selectedProgrammer: programmer }, verify, verbose] =
await Promise.all([ await Promise.all([
verifyOptions.fqbn, // already decorated FQBN verifyOptions.fqbn, // already decorated FQBN
this.boardsDataStore.getData(sanitizeFqbn(verifyOptions.fqbn)), this.boardsDataStore.getData(
verifyOptions.fqbn
? new FQBN(verifyOptions.fqbn).toString(true)
: undefined
),
this.preferences.get('arduino.upload.verify'), this.preferences.get('arduino.upload.verify'),
this.preferences.get('arduino.upload.verbose'), this.preferences.get('arduino.upload.verbose'),
]); ]);

View File

@@ -1,10 +1,10 @@
import { nls } from '@theia/core/lib/common/nls';
import { inject, injectable } from '@theia/core/shared/inversify'; import { inject, injectable } from '@theia/core/shared/inversify';
import { nls } from '@theia/core/lib/common';
import { BoardUserField, CoreError } from '../../common/protocol'; import { BoardUserField, CoreError } from '../../common/protocol';
import { BoardsServiceProvider } from '../boards/boards-service-provider'; import { BoardsServiceProvider } from '../boards/boards-service-provider';
import { UserFieldsDialog } from '../dialogs/user-fields/user-fields-dialog'; import { UserFieldsDialog } from '../dialogs/user-fields/user-fields-dialog';
import { ArduinoMenus } from '../menu/arduino-menus'; import { ArduinoMenus } from '../menu/arduino-menus';
import { MenuModelRegistry, Contribution } from './contribution'; import { Contribution, MenuModelRegistry } from './contribution';
import { UploadSketch } from './upload-sketch'; import { UploadSketch } from './upload-sketch';
@injectable() @injectable()
@@ -21,12 +21,11 @@ export class UserFields extends Contribution {
protected override init(): void { protected override init(): void {
super.init(); super.init();
this.boardsServiceProvider.onBoardsConfigDidChange(async () => { this.boardsServiceProvider.onBoardsConfigDidChange(() => this.refresh());
const userFields = }
await this.boardsServiceProvider.selectedBoardUserFields();
this.boardRequiresUserFields = userFields.length > 0; override onReady(): void {
this.menuManager.update(); this.boardsServiceProvider.ready.then(() => this.refresh());
});
} }
override registerMenus(registry: MenuModelRegistry): void { override registerMenus(registry: MenuModelRegistry): void {
@@ -37,6 +36,13 @@ export class UserFields extends Contribution {
}); });
} }
private async refresh(): Promise<void> {
const userFields =
await this.boardsServiceProvider.selectedBoardUserFields();
this.boardRequiresUserFields = userFields.length > 0;
this.menuManager.update();
}
private selectedFqbnAddress(): string | undefined { private selectedFqbnAddress(): string | undefined {
const { boardsConfig } = this.boardsServiceProvider; const { boardsConfig } = this.boardsServiceProvider;
const fqbn = boardsConfig.selectedBoard?.fqbn; const fqbn = boardsConfig.selectedBoard?.fqbn;

View File

@@ -1,18 +1,18 @@
import { inject, injectable } from '@theia/core/shared/inversify';
import { Emitter } from '@theia/core/lib/common/event'; import { Emitter } from '@theia/core/lib/common/event';
import { nls } from '@theia/core/lib/common/nls';
import { inject, injectable } from '@theia/core/shared/inversify';
import type { CoreService } from '../../common/protocol';
import { ArduinoMenus } from '../menu/arduino-menus'; import { ArduinoMenus } from '../menu/arduino-menus';
import { CurrentSketch } from '../sketches-service-client-impl';
import { ArduinoToolbar } from '../toolbar/arduino-toolbar'; import { ArduinoToolbar } from '../toolbar/arduino-toolbar';
import { import {
CoreServiceContribution,
Command, Command,
CommandRegistry, CommandRegistry,
MenuModelRegistry, CoreServiceContribution,
KeybindingRegistry, KeybindingRegistry,
MenuModelRegistry,
TabBarToolbarRegistry, TabBarToolbarRegistry,
} from './contribution'; } from './contribution';
import { nls } from '@theia/core/lib/common';
import { CurrentSketch } from '../sketches-service-client-impl';
import { CoreService } from '../../common/protocol';
import { CoreErrorHandler } from './core-error-handler'; import { CoreErrorHandler } from './core-error-handler';
export interface VerifySketchParams { export interface VerifySketchParams {
@@ -131,11 +131,15 @@ export class VerifySketch extends CoreServiceContribution {
'arduino/sketch/compile', 'arduino/sketch/compile',
'Compiling sketch...' 'Compiling sketch...'
), ),
task: (progressId, coreService) => task: (progressId, coreService, token) =>
coreService.compile({ coreService.compile(
...options, {
progressId, ...options,
}), progressId,
},
token
),
cancelable: true,
}); });
this.messageService.info( this.messageService.info(
nls.localize('arduino/sketch/doneCompiling', 'Done compiling.'), nls.localize('arduino/sketch/doneCompiling', 'Done compiling.'),

View File

@@ -179,7 +179,8 @@ export class CreateApi {
); );
}) })
.catch((reason) => { .catch((reason) => {
if (reason?.status === 404) return [] as Create.Resource[]; if (reason?.status === 404)
return [] as Create.Resource[]; // TODO: must not swallow 404
else throw reason; else throw reason;
}); });
} }
@@ -486,18 +487,12 @@ export class CreateApi {
await this.run(url, init, ResponseResultProvider.NOOP); await this.run(url, init, ResponseResultProvider.NOOP);
} }
private fetchCounter = 0;
private async run<T>( private async run<T>(
requestInfo: URL, requestInfo: URL,
init: RequestInit | undefined, init: RequestInit | undefined,
resultProvider: ResponseResultProvider = ResponseResultProvider.JSON resultProvider: ResponseResultProvider = ResponseResultProvider.JSON
): Promise<T> { ): Promise<T> {
const fetchCount = `[${++this.fetchCounter}]`;
const fetchStart = performance.now();
const method = init?.method ? `${init.method}: ` : '';
const url = requestInfo.toString();
const response = await fetch(requestInfo.toString(), init); const response = await fetch(requestInfo.toString(), init);
const fetchEnd = performance.now();
if (!response.ok) { if (!response.ok) {
let details: string | undefined = undefined; let details: string | undefined = undefined;
try { try {
@@ -508,28 +503,25 @@ export class CreateApi {
const { statusText, status } = response; const { statusText, status } = response;
throw new CreateError(statusText, status, details); throw new CreateError(statusText, status, details);
} }
const parseStart = performance.now();
const result = await resultProvider(response); const result = await resultProvider(response);
const parseEnd = performance.now();
console.debug(
`HTTP ${fetchCount} ${method}${url} [fetch: ${(
fetchEnd - fetchStart
).toFixed(2)} ms, parse: ${(parseEnd - parseStart).toFixed(
2
)} ms] body: ${
typeof result === 'string' ? result : JSON.stringify(result)
}`
);
return result; return result;
} }
private async headers(): Promise<Record<string, string>> { private async headers(): Promise<Record<string, string>> {
const token = await this.token(); const token = await this.token();
return { const headers: Record<string, string> = {
'content-type': 'application/json', 'content-type': 'application/json',
accept: 'application/json', accept: 'application/json',
authorization: `Bearer ${token}`, authorization: `Bearer ${token}`,
}; };
const sharedSpaceID =
this.arduinoPreferences['arduino.cloud.sharedSpaceID'];
if (sharedSpaceID) {
headers['x-organization'] = sharedSpaceID;
}
return headers;
} }
private domain(apiVersion = 'v2'): string { private domain(apiVersion = 'v2'): string {

View File

@@ -82,6 +82,13 @@ export function isNotFound(err: unknown): err is NotFoundError {
return isErrorWithStatusOf(err, 404); return isErrorWithStatusOf(err, 404);
} }
export type UnprocessableContentError = CreateError & { status: 422 };
export function isUnprocessableContent(
err: unknown
): err is UnprocessableContentError {
return isErrorWithStatusOf(err, 422);
}
function isErrorWithStatusOf( function isErrorWithStatusOf(
err: unknown, err: unknown,
status: number status: number

View File

@@ -69,6 +69,7 @@ export const CertificateUploaderComponent = ({
const onItemSelect = React.useCallback( const onItemSelect = React.useCallback(
(item: BoardOptionValue | null) => { (item: BoardOptionValue | null) => {
if (!item) { if (!item) {
setSelectedItem(null);
return; return;
} }
const board = item.board; const board = item.board;

View File

@@ -1,8 +1,9 @@
import { nls } from '@theia/core/lib/common'; import { nls } from '@theia/core/lib/common';
import React from '@theia/core/shared/react'; import React from '@theia/core/shared/react';
import type { import {
BoardList, boardListItemEquals,
BoardListItemWithBoard, type BoardList,
type BoardListItemWithBoard,
} from '../../../common/protocol/board-list'; } from '../../../common/protocol/board-list';
import { ArduinoSelect } from '../../widgets/arduino-select'; import { ArduinoSelect } from '../../widgets/arduino-select';
@@ -75,7 +76,9 @@ export const SelectBoardComponent = ({
setSelectOptions(boardOptions); setSelectOptions(boardOptions);
if (selectedItem) { if (selectedItem) {
selBoard = updatableBoards.indexOf(selectedItem); selBoard = updatableBoards.findIndex((board) =>
boardListItemEquals(board, selectedItem)
);
} }
selectOption(boardOptions[selBoard] || null); selectOption(boardOptions[selBoard] || null);

View File

@@ -104,6 +104,7 @@ export const FirmwareUploaderComponent = ({
const onItemSelect = React.useCallback( const onItemSelect = React.useCallback(
(item: BoardListItemWithBoard | null) => { (item: BoardListItemWithBoard | null) => {
if (!item) { if (!item) {
setSelectedItem(null);
return; return;
} }
const board = item.board; const board = item.board;

View File

@@ -1,7 +1,7 @@
import { DisposableCollection, Emitter, Event } from '@theia/core'; import { DisposableCollection, Emitter, Event } from '@theia/core';
import { FrontendApplicationContribution } from '@theia/core/lib/browser'; import { FrontendApplicationContribution } from '@theia/core/lib/browser';
import { inject, injectable } from '@theia/core/shared/inversify'; import { inject, injectable } from '@theia/core/shared/inversify';
import { HostedPluginSupport } from './theia/plugin-ext/hosted-plugin'; import { HostedPluginSupport } from './hosted-plugin-support';
/** /**
* Frontend contribution to watch VS Code extension start/stop events from Theia. * Frontend contribution to watch VS Code extension start/stop events from Theia.

View File

@@ -0,0 +1,14 @@
import type { Event } from '@theia/core/lib/common/event';
/*
This implementation hides the default HostedPluginSupport implementation from Theia to be able to test it.
Otherwise, the default implementation fails at require time due to the `import.meta` in the Theia plugin worker code.
https://github.com/eclipse-theia/theia/blob/964f69ca3b3a5fb87ffa0177fb300b74ba0ca39f/packages/plugin-ext/src/hosted/browser/plugin-worker.ts#L30-L32
*/
export const HostedPluginSupport = Symbol('HostedPluginSupport');
export interface HostedPluginSupport {
readonly didStart: Promise<void>;
readonly onDidLoad: Event<void>;
readonly onDidCloseConnection: Event<void>;
}

View File

@@ -12,15 +12,13 @@ import {
LibrarySearch, LibrarySearch,
LibraryService, LibraryService,
} from '../../common/protocol/library-service'; } from '../../common/protocol/library-service';
import { import { ListWidget } from '../widgets/component-list/list-widget';
ListWidget,
UserAbortError,
} from '../widgets/component-list/list-widget';
import { Installable } from '../../common/protocol'; import { Installable } from '../../common/protocol';
import { ListItemRenderer } from '../widgets/component-list/list-item-renderer'; import { ListItemRenderer } from '../widgets/component-list/list-item-renderer';
import { nls } from '@theia/core/lib/common'; import { nls } from '@theia/core/lib/common';
import { LibraryFilterRenderer } from '../widgets/component-list/filter-renderer'; import { LibraryFilterRenderer } from '../widgets/component-list/filter-renderer';
import { findChildTheiaButton, splitByBoldTag } from '../utils/dom'; import { findChildTheiaButton, splitByBoldTag } from '../utils/dom';
import { UserAbortError } from '../../common/protocol/progressible';
@injectable() @injectable()
export class LibraryListWidget extends ListWidget< export class LibraryListWidget extends ListWidget<

View File

@@ -46,7 +46,7 @@ export class NotificationCenter
new Emitter<ProgressMessage>(); new Emitter<ProgressMessage>();
private readonly indexUpdateDidFailEmitter = private readonly indexUpdateDidFailEmitter =
new Emitter<IndexUpdateDidFailParams>(); new Emitter<IndexUpdateDidFailParams>();
private readonly daemonDidStartEmitter = new Emitter<string>(); private readonly daemonDidStartEmitter = new Emitter<number>();
private readonly daemonDidStopEmitter = new Emitter<void>(); private readonly daemonDidStopEmitter = new Emitter<void>();
private readonly configDidChangeEmitter = new Emitter<ConfigState>(); private readonly configDidChangeEmitter = new Emitter<ConfigState>();
private readonly platformDidInstallEmitter = new Emitter<{ private readonly platformDidInstallEmitter = new Emitter<{
@@ -136,7 +136,7 @@ export class NotificationCenter
this.indexUpdateDidFailEmitter.fire(params); this.indexUpdateDidFailEmitter.fire(params);
} }
notifyDaemonDidStart(port: string): void { notifyDaemonDidStart(port: number): void {
this.daemonDidStartEmitter.fire(port); this.daemonDidStartEmitter.fire(port);
} }

View File

@@ -67,6 +67,7 @@ export class SketchesServiceClientImpl
); );
private _currentSketch: CurrentSketch | undefined; private _currentSketch: CurrentSketch | undefined;
private _currentIdeTempFolderUri: URI | undefined;
private currentSketchLoaded = new Deferred<CurrentSketch>(); private currentSketchLoaded = new Deferred<CurrentSketch>();
onStart(): void { onStart(): void {
@@ -74,7 +75,10 @@ export class SketchesServiceClientImpl
this.watchSketchbookDir(sketchDirUri); this.watchSketchbookDir(sketchDirUri);
const refreshCurrentSketch = async () => { const refreshCurrentSketch = async () => {
const currentSketch = await this.loadCurrentSketch(); const currentSketch = await this.loadCurrentSketch();
this.useCurrentSketch(currentSketch); const ideTempFolderUri = await this.getIdeTempFolderUriForSketch(
currentSketch
);
this.useCurrentSketch(currentSketch, ideTempFolderUri);
}; };
this.toDispose.push( this.toDispose.push(
this.configService.onDidChangeSketchDirUri((sketchDirUri) => { this.configService.onDidChangeSketchDirUri((sketchDirUri) => {
@@ -141,7 +145,10 @@ export class SketchesServiceClientImpl
} }
if (!Sketch.sameAs(this._currentSketch, reloadedSketch)) { if (!Sketch.sameAs(this._currentSketch, reloadedSketch)) {
this.useCurrentSketch(reloadedSketch, true); const ideTempFolderUri = await this.getIdeTempFolderUriForSketch(
reloadedSketch
);
this.useCurrentSketch(reloadedSketch, ideTempFolderUri, true);
} }
return; return;
} }
@@ -179,11 +186,23 @@ export class SketchesServiceClientImpl
]); ]);
} }
private async getIdeTempFolderUriForSketch(
sketch: CurrentSketch
): Promise<URI | undefined> {
if (CurrentSketch.isValid(sketch)) {
const uri = await this.sketchesService.getIdeTempFolderUri(sketch);
return new URI(uri);
}
return undefined;
}
private useCurrentSketch( private useCurrentSketch(
currentSketch: CurrentSketch, currentSketch: CurrentSketch,
ideTempFolderUri: URI | undefined,
reassignPromise = false reassignPromise = false
) { ) {
this._currentSketch = currentSketch; this._currentSketch = currentSketch;
this._currentIdeTempFolderUri = ideTempFolderUri;
if (reassignPromise) { if (reassignPromise) {
this.currentSketchLoaded = new Deferred(); this.currentSketchLoaded = new Deferred();
} }
@@ -273,6 +292,14 @@ export class SketchesServiceClientImpl
return false; return false;
} }
if (
this._currentIdeTempFolderUri &&
this._currentIdeTempFolderUri.resolve('launch.json').toString() ===
toCheck.toString()
) {
return false;
}
const isCloudSketch = toCheck const isCloudSketch = toCheck
.toString() .toString()
.includes(`${REMOTE_SKETCHBOOK_FOLDER}/${ARDUINO_CLOUD_FOLDER}`); .includes(`${REMOTE_SKETCHBOOK_FOLDER}/${ARDUINO_CLOUD_FOLDER}`);

View File

@@ -38,7 +38,9 @@
.arduino-select__control.arduino-select__control--menu-is-open { .arduino-select__control.arduino-select__control--menu-is-open {
border: 1px solid !important; border: 1px solid !important;
border-color: var(--theia-focusBorder) !important; border-color: var(--theia-focusBorder) !important;
border-bottom-color: var(--theia-sideBar-background) !important; /* if the bottom border color has the same color as the background of the control, we make the border "invisible" */ border-bottom-color: var(
--theia-sideBar-background
) !important; /* if the bottom border color has the same color as the background of the control, we make the border "invisible" */
} }
.arduino-select__value-container .arduino-select__single-value { .arduino-select__value-container .arduino-select__single-value {

View File

@@ -1,324 +1,324 @@
#select-board-dialog-container > .dialogBlock { #select-board-dialog-container > .dialogBlock {
width: 640px; width: 640px;
height: 500px; height: 500px;
} }
div#select-board-dialog { div#select-board-dialog {
margin: 5px; margin: 5px;
height: 100%; height: 100%;
} }
div#select-board-dialog .selectBoardContainer { div#select-board-dialog .selectBoardContainer {
display: flex; display: flex;
gap: 10px; gap: 10px;
overflow: hidden; overflow: hidden;
max-height: 100%; max-height: 100%;
height: 100%; height: 100%;
} }
.select-board-dialog .head { .select-board-dialog .head {
margin: 5px; margin: 5px;
} }
.dialogContent.select-board-dialog { .dialogContent.select-board-dialog {
height: 100%; height: 100%;
} }
div.dialogContent.select-board-dialog > div.head .title { div.dialogContent.select-board-dialog > div.head .title {
font-weight: 400; font-weight: 400;
letter-spacing: 0.02em; letter-spacing: 0.02em;
font-size: 1.2em; font-size: 1.2em;
color: var(--theia-editorWidget-foreground); color: var(--theia-editorWidget-foreground);
margin-bottom: 10px; margin-bottom: 10px;
} }
div#select-board-dialog .selectBoardContainer .list .item.selected { div#select-board-dialog .selectBoardContainer .list .item.selected {
background: var(--theia-secondaryButton-hoverBackground); background: var(--theia-secondaryButton-hoverBackground);
} }
div#select-board-dialog .selectBoardContainer .list .item.selected i { div#select-board-dialog .selectBoardContainer .list .item.selected i {
color: var(--theia-arduino-branding-primary); color: var(--theia-arduino-branding-primary);
} }
#select-board-dialog .selectBoardContainer .search, #select-board-dialog .selectBoardContainer .search,
#select-board-dialog .selectBoardContainer .search input, #select-board-dialog .selectBoardContainer .search input,
#select-board-dialog .selectBoardContainer .list, #select-board-dialog .selectBoardContainer .list,
#select-board-dialog .selectBoardContainer .list { #select-board-dialog .selectBoardContainer .list {
background: var(--theia-editor-background); background: var(--theia-editor-background);
} }
#select-board-dialog .selectBoardContainer .search input { #select-board-dialog .selectBoardContainer .search input {
border: none; border: none;
width: 100%; width: 100%;
height: auto; height: auto;
max-height: 37px; max-height: 37px;
padding: 10px 5px 10px 10px; padding: 10px 5px 10px 10px;
margin: 0; margin: 0;
vertical-align: top; vertical-align: top;
display: flex; display: flex;
color: var(--theia-input-foreground); color: var(--theia-input-foreground);
} }
#select-board-dialog .selectBoardContainer .search input:focus { #select-board-dialog .selectBoardContainer .search input:focus {
box-shadow: none; box-shadow: none;
} }
#select-board-dialog .selectBoardContainer .container { #select-board-dialog .selectBoardContainer .container {
flex: 1; flex: 1;
overflow: hidden; overflow: hidden;
max-height: 100%; max-height: 100%;
} }
#select-board-dialog .selectBoardContainer .container .content { #select-board-dialog .selectBoardContainer .container .content {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
max-height: 100%; max-height: 100%;
height: 100%; height: 100%;
} }
#select-board-dialog .selectBoardContainer .left.container .content { #select-board-dialog .selectBoardContainer .left.container .content {
margin: 0 5px 0 0; margin: 0 5px 0 0;
} }
#select-board-dialog .selectBoardContainer .right.container .content { #select-board-dialog .selectBoardContainer .right.container .content {
margin: 0 0 0 5px; margin: 0 0 0 5px;
} }
#select-board-dialog .selectBoardContainer .container .content .title { #select-board-dialog .selectBoardContainer .container .content .title {
color: var(--theia-editorWidget-foreground); color: var(--theia-editorWidget-foreground);
padding: 0px 0px 10px 0px; padding: 0px 0px 10px 0px;
text-transform: uppercase; text-transform: uppercase;
} }
#select-board-dialog .selectBoardContainer .container .content .footer { #select-board-dialog .selectBoardContainer .container .content .footer {
padding: 10px 5px 10px 0px; padding: 10px 5px 10px 0px;
} }
#select-board-dialog .selectBoardContainer .container .content .loading { #select-board-dialog .selectBoardContainer .container .content .loading {
font-size: var(--theia-ui-font-size1); font-size: var(--theia-ui-font-size1);
color: var(--theia-editorWidget-foreground); color: var(--theia-editorWidget-foreground);
padding: 10px 5px 10px 10px; padding: 10px 5px 10px 10px;
text-transform: uppercase; text-transform: uppercase;
/* The max, min-height comes from `.list` 200px + 47px top padding - 2 * 10px top padding */ /* The max, min-height comes from `.list` 200px + 47px top padding - 2 * 10px top padding */
max-height: 227px; max-height: 227px;
min-height: 227px; min-height: 227px;
} }
#select-board-dialog .selectBoardContainer .list .item { #select-board-dialog .selectBoardContainer .list .item {
padding: 10px 5px 10px 10px; padding: 10px 5px 10px 10px;
display: flex; display: flex;
white-space: nowrap; white-space: nowrap;
overflow-x: hidden; overflow-x: hidden;
flex: 1 0; flex: 1 0;
} }
#select-board-dialog .selectBoardContainer .list .item .selected-icon { #select-board-dialog .selectBoardContainer .list .item .selected-icon {
margin-left: auto; margin-left: auto;
} }
#select-board-dialog .selectBoardContainer .list .item .details { #select-board-dialog .selectBoardContainer .list .item .details {
font-size: var(--theia-ui-font-size1); font-size: var(--theia-ui-font-size1);
opacity: var(--theia-mod-disabled-opacity); opacity: var(--theia-mod-disabled-opacity);
width: 155px; /* used heuristics for the calculation */ width: 155px; /* used heuristics for the calculation */
white-space: pre; white-space: pre;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
} }
#select-board-dialog .selectBoardContainer .list .item.missing { #select-board-dialog .selectBoardContainer .list .item.missing {
opacity: var(--theia-mod-disabled-opacity); opacity: var(--theia-mod-disabled-opacity);
} }
#select-board-dialog .selectBoardContainer .list .item:hover { #select-board-dialog .selectBoardContainer .list .item:hover {
background: var(--theia-secondaryButton-hoverBackground); background: var(--theia-secondaryButton-hoverBackground);
} }
#select-board-dialog .selectBoardContainer .list .label { #select-board-dialog .selectBoardContainer .list .label {
white-space: pre; white-space: pre;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
} }
#select-board-dialog .selectBoardContainer .list { #select-board-dialog .selectBoardContainer .list {
max-height: 200px; max-height: 200px;
overflow-y: auto; overflow-y: auto;
flex: 1; flex: 1;
} }
#select-board-dialog .selectBoardContainer .ports.list { #select-board-dialog .selectBoardContainer .ports.list {
margin: 47px 0px 0px 0px; /* 47 is 37 as input height for the `Boards`, plus 10 margin bottom. */ margin: 47px 0px 0px 0px; /* 47 is 37 as input height for the `Boards`, plus 10 margin bottom. */
} }
#select-board-dialog .selectBoardContainer .search { #select-board-dialog .selectBoardContainer .search {
margin-bottom: 10px; margin-bottom: 10px;
display: flex; display: flex;
align-items: center; align-items: center;
padding-right: 5px; padding-right: 5px;
} }
.arduino-boards-toolbar-item-container { .arduino-boards-toolbar-item-container {
align-items: center; align-items: center;
background: var(--theia-arduino-toolbar-dropdown-background); background: var(--theia-arduino-toolbar-dropdown-background);
border-radius: 1px; border-radius: 1px;
color: var(--theia-arduino-toolbar-dropdown-label); color: var(--theia-arduino-toolbar-dropdown-label);
border: 1px solid var(--theia-arduino-toolbar-dropdown-border); border: 1px solid var(--theia-arduino-toolbar-dropdown-border);
display: flex; display: flex;
gap: 10px; gap: 10px;
height: var(--arduino-button-height); height: var(--arduino-button-height);
margin: 0 4px; margin: 0 4px;
overflow: hidden; overflow: hidden;
padding: 0 10px; padding: 0 10px;
width: 210px; width: 210px;
} }
.arduino-boards-toolbar-item--protocol, .arduino-boards-toolbar-item--protocol,
.arduino-boards-dropdown-item--protocol { .arduino-boards-dropdown-item--protocol {
align-items: center; align-items: center;
display: flex; display: flex;
font-size: 16px; font-size: 16px;
} }
.arduino-boards-toolbar-item--protocol, .arduino-boards-toolbar-item--protocol,
.arduino-boards-dropdown-item--protocol { .arduino-boards-dropdown-item--protocol {
color: var(--theia-arduino-toolbar-dropdown-label); color: var(--theia-arduino-toolbar-dropdown-label);
} }
.arduino-boards-toolbar-item-container .arduino-boards-toolbar-item { .arduino-boards-toolbar-item-container .arduino-boards-toolbar-item {
display: flex; display: flex;
align-items: baseline; align-items: baseline;
width: 100%; width: 100%;
} }
.arduino-boards-toolbar-item--label { .arduino-boards-toolbar-item--label {
width: 100%; width: 100%;
} }
.arduino-boards-toolbar-item--label-connected { .arduino-boards-toolbar-item--label-connected {
font-family: 'Open Sans Bold'; font-family: "Open Sans Bold";
font-style: normal; font-style: normal;
font-weight: 700; font-weight: 700;
font-size: 14px; font-size: 14px;
} }
.arduino-boards-toolbar-item-container .caret { .arduino-boards-toolbar-item-container .caret {
width: 10px; width: 10px;
margin-right: 5px; margin-right: 5px;
} }
.arduino-boards-dropdown-list { .arduino-boards-dropdown-list {
margin: -1px; margin: -1px;
z-index: 1; z-index: 1;
border: 1px solid var(--theia-arduino-toolbar-dropdown-border); border: 1px solid var(--theia-arduino-toolbar-dropdown-border);
font-family: 'Open Sans'; font-family: "Open Sans";
font-style: normal; font-style: normal;
font-weight: 400; font-weight: 400;
font-size: 12px; font-size: 12px;
} }
.arduino-boards-dropdown-list:focus { .arduino-boards-dropdown-list:focus {
border: 1px solid var(--theia-arduino-toolbar-dropdown-borderActive); border: 1px solid var(--theia-arduino-toolbar-dropdown-borderActive);
} }
.arduino-boards-dropdown-list--items-container { .arduino-boards-dropdown-list--items-container {
overflow: auto; overflow: auto;
max-height: 404px; max-height: 404px;
background: var(--theia-arduino-toolbar-dropdown-background); background: var(--theia-arduino-toolbar-dropdown-background);
} }
.arduino-boards-dropdown-list--items-container::-webkit-scrollbar { .arduino-boards-dropdown-list--items-container::-webkit-scrollbar {
background: var(--theia-arduino-toolbar-dropdown-background); background: var(--theia-arduino-toolbar-dropdown-background);
} }
.arduino-boards-dropdown-item { .arduino-boards-dropdown-item {
background: var(--theia-arduino-toolbar-dropdown-background); background: var(--theia-arduino-toolbar-dropdown-background);
color: var(--theia-arduino-toolbar-dropdown-label); color: var(--theia-arduino-toolbar-dropdown-label);
cursor: default; cursor: default;
display: flex; display: flex;
font-size: var(--theia-ui-font-size1); font-size: var(--theia-ui-font-size1);
justify-content: space-between; justify-content: space-between;
padding: 10px; padding: 10px;
} }
.arduino-boards-dropdown-item--board-header { .arduino-boards-dropdown-item--board-header {
display: flex; display: flex;
align-items: center; align-items: center;
} }
.arduino-boards-dropdown-item--label { .arduino-boards-dropdown-item--label {
overflow: hidden; overflow: hidden;
flex: 1; flex: 1;
} }
/* Redefine default codicon size https://github.com/microsoft/vscode/commit/38cd0a377b7abef34fb07fe770fc633e68819ba6 */ /* Redefine default codicon size https://github.com/microsoft/vscode/commit/38cd0a377b7abef34fb07fe770fc633e68819ba6 */
.arduino-boards-dropdown-item .codicon[class*='codicon-'] { .arduino-boards-dropdown-item .codicon[class*="codicon-"] {
font-size: 14px; font-size: 14px;
} }
.arduino-boards-dropdown-item .p-TabBar-toolbar { .arduino-boards-dropdown-item .p-TabBar-toolbar {
padding: 0px; padding: 0px;
margin: 0px; margin: 0px;
flex-direction: column; flex-direction: column;
} }
.arduino-boards-dropdown-item .p-TabBar-toolbar .item { .arduino-boards-dropdown-item .p-TabBar-toolbar .item {
margin: 0px; margin: 0px;
} }
.arduino-boards-dropdown-item .p-TabBar-toolbar .item .action-label { .arduino-boards-dropdown-item .p-TabBar-toolbar .item .action-label {
padding: 0px; padding: 0px;
} }
.arduino-boards-dropdown-item--board-label { .arduino-boards-dropdown-item--board-label {
font-size: 14px; font-size: 14px;
} }
.arduino-boards-dropdown-item .arduino-boards-dropdown-item--protocol { .arduino-boards-dropdown-item .arduino-boards-dropdown-item--protocol {
margin-right: 10px; margin-right: 10px;
} }
.arduino-boards-dropdown-item--port-label { .arduino-boards-dropdown-item--port-label {
font-size: 12px; font-size: 12px;
} }
.arduino-boards-dropdown-item:hover { .arduino-boards-dropdown-item:hover {
background: var(--theia-arduino-toolbar-dropdown-option-backgroundHover); background: var(--theia-arduino-toolbar-dropdown-option-backgroundHover);
} }
.arduino-boards-dropdown-item--selected, .arduino-boards-dropdown-item--selected,
.arduino-boards-dropdown-item--selected:hover { .arduino-boards-dropdown-item--selected:hover {
background: var(--theia-arduino-toolbar-dropdown-option-backgroundSelected); background: var(--theia-arduino-toolbar-dropdown-option-backgroundSelected);
border: 1px solid var(--theia-arduino-toolbar-dropdown-option-backgroundSelected); border: 1px solid
var(--theia-arduino-toolbar-dropdown-option-backgroundSelected);
} }
.arduino-boards-dropdown-item--selected .arduino-boards-dropdown-item--selected
.arduino-boards-dropdown-item--port-label { .arduino-boards-dropdown-item--port-label {
color: var(--theia-arduino-toolbar-dropdown-label); color: var(--theia-arduino-toolbar-dropdown-label);
} }
.arduino-boards-dropdown-item--selected .fa { .arduino-boards-dropdown-item--selected .fa {
color: var(--theia-arduino-toolbar-dropdown-iconSelected); color: var(--theia-arduino-toolbar-dropdown-iconSelected);
} }
.arduino-board-dropdown-footer { .arduino-board-dropdown-footer {
color: var(--theia-secondaryButton-foreground); color: var(--theia-secondaryButton-foreground);
border-top: 1px solid var(--theia-dropdown-border); border-top: 1px solid var(--theia-dropdown-border);
} }
@media only screen and (max-height: 400px) { @media only screen and (max-height: 400px) {
div.dialogContent.select-board-dialog > div.head { div.dialogContent.select-board-dialog > div.head {
display: none; display: none;
} }
#select-board-dialog .selectBoardContainer .container .content .title { #select-board-dialog .selectBoardContainer .container .content .title {
display: none; display: none;
} }
} }
#select-board-dialog .no-result { #select-board-dialog .no-result {
text-transform: uppercase; text-transform: uppercase;
height: 100%; height: 100%;
user-select: none; user-select: none;
padding: 10px 5px; padding: 10px 5px;
overflow-wrap: break-word; overflow-wrap: break-word;
} }

View File

@@ -1,76 +1,75 @@
#certificate-uploader-dialog-container > .dialogBlock { #certificate-uploader-dialog-container > .dialogBlock {
width: 600px; width: 600px;
} }
.certificate-uploader-dialog .theia-select { .certificate-uploader-dialog .theia-select {
border: none !important; border: none !important;
} }
.certificate-uploader-dialog .arduino-select__control { .certificate-uploader-dialog .arduino-select__control {
height: 31px; height: 31px;
background: var(--theia-dropdown-background) !important; background: var(--theia-dropdown-background) !important;
} }
.certificate-uploader-dialog .dialogRow > button{ .certificate-uploader-dialog .dialogRow > button {
margin-right: 3px; margin-right: 3px;
} }
.certificate-uploader-dialog .certificate-list { .certificate-uploader-dialog .certificate-list {
border: 1px solid var(--theia-editorWidget-border); border: 1px solid var(--theia-editorWidget-border);
border-radius: 2px;; border-radius: 2px;
color: var(--theia-editor-foreground); color: var(--theia-editor-foreground);
background-color: var(--theia-editor-background); background-color: var(--theia-editor-background);
overflow: auto; overflow: auto;
height: 120px; height: 120px;
flex: 1; flex: 1;
} }
.certificate-uploader-dialog .certificate-list .certificate-row { .certificate-uploader-dialog .certificate-list .certificate-row {
display: flex; display: flex;
padding: 6px 10px 5px 10px padding: 6px 10px 5px 10px;
} }
.certificate-uploader-dialog .certificate-list .certificate-row:hover { .certificate-uploader-dialog .certificate-list .certificate-row:hover {
background-color: var(--theia-list-activeSelectionBackground); background-color: var(--theia-list-activeSelectionBackground);
} }
.certificate-uploader-dialog .upload-status { .certificate-uploader-dialog .upload-status {
display: flex; display: flex;
align-items: center; align-items: center;
flex: 1; flex: 1;
} }
.certificate-uploader-dialog .success { .certificate-uploader-dialog .success {
display: flex; display: flex;
align-items: center; align-items: center;
color: #1DA086; color: #1da086;
} }
.certificate-uploader-dialog .warn { .certificate-uploader-dialog .warn {
color: #C11F09; color: #c11f09;
} }
.certificate-uploader-dialog .status-icon { .certificate-uploader-dialog .status-icon {
margin-right: 10px; margin-right: 10px;
} }
.certificate-uploader-dialog .add-cert-btn { .certificate-uploader-dialog .add-cert-btn {
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: space-between; justify-content: space-between;
} }
.certificate-uploader-dialog .add-cert-btn .caret { .certificate-uploader-dialog .add-cert-btn .caret {
margin-left: 6px; margin-left: 6px;
} }
.certificate-add { .certificate-add {
padding: 16px; padding: 16px;
border-radius: 3px; border-radius: 3px;
border: 1px solid var(--theia-editorWidget-border); border: 1px solid var(--theia-editorWidget-border);
color: var(--theia-editorWidget-foreground); color: var(--theia-editorWidget-foreground);
background-color: var(--theia-editorWidget-background); background-color: var(--theia-editorWidget-background);
} }
.certificate-add input { .certificate-add input {
margin-top: 12px; margin-top: 12px;
padding: 0 12px; padding: 0 12px;
width: 100%; width: 100%;
box-sizing: border-box; box-sizing: border-box;
} }

View File

@@ -23,8 +23,7 @@
-webkit-mask-size: 100%; -webkit-mask-size: 100%;
} }
.p-mod-current .p-mod-current .cloud-sketchbook-tree-icon {
.cloud-sketchbook-tree-icon {
background-color: var(--theia-foreground); background-color: var(--theia-foreground);
-webkit-mask: url(../icons/arduino-cloud-filled.svg); -webkit-mask: url(../icons/arduino-cloud-filled.svg);
-webkit-mask-position: center; -webkit-mask-position: center;
@@ -33,49 +32,49 @@
} }
.sketchbook-trees-container .sketchbook-trees-container
.p-TabBar[data-orientation="horizontal"] .p-TabBar[data-orientation="horizontal"]
> .p-TabBar-content { > .p-TabBar-content {
justify-content: center; justify-content: center;
border-bottom: 1px solid var(--theia-tree-indentGuidesStroke); border-bottom: 1px solid var(--theia-tree-indentGuidesStroke);
} }
.sketchbook-trees-container .sketchbook-trees-container
.p-Widget.p-TabBar.p-DockPanel-tabBar .p-Widget.p-TabBar.p-DockPanel-tabBar
> ul > ul
> li.p-TabBar-tab > li.p-TabBar-tab
> div.p-TabBar-tabLabel { > div.p-TabBar-tabLabel {
display: none; display: none;
width: 0px; width: 0px;
max-width: 0px; max-width: 0px;
} }
.sketchbook-trees-container .sketchbook-trees-container
.p-Widget.p-TabBar.p-DockPanel-tabBar .p-Widget.p-TabBar.p-DockPanel-tabBar
> ul > ul
> li.p-TabBar-tab > li.p-TabBar-tab
> div.p-TabBar-tabCloseIcon { > div.p-TabBar-tabCloseIcon {
display: none; display: none;
width: 0px; width: 0px;
max-width: 0px; max-width: 0px;
} }
.sketchbook-trees-container .sketchbook-trees-container
.p-TabBar[data-orientation="horizontal"] .p-TabBar[data-orientation="horizontal"]
.p-TabBar-tab { .p-TabBar-tab {
min-width: 55px; min-width: 55px;
} }
.sketchbook-trees-container .sketchbook-trees-container
.p-Widget.p-TabBar.p-DockPanel-tabBar .p-Widget.p-TabBar.p-DockPanel-tabBar
> ul > ul
> li.p-TabBar-tab { > li.p-TabBar-tab {
padding-left: 20px; padding-left: 20px;
} }
.sketchbook-trees-container .sketchbook-trees-container
.p-Widget.p-TabBar.p-DockPanel-tabBar .p-Widget.p-TabBar.p-DockPanel-tabBar
> ul > ul
> li.p-TabBar-tab.p-mod-current { > li.p-TabBar-tab.p-mod-current {
border-bottom: 2px solid var(--theia-activityBar-activeBorder); border-bottom: 2px solid var(--theia-activityBar-activeBorder);
} }

View File

@@ -1,4 +1,4 @@
.codicon-debug-alt:before { .codicon-debug-alt:before {
font-family: 'FontAwesome' !important; font-family: "FontAwesome" !important;
content: "\e905" !important content: "\e905" !important;
} }

View File

@@ -1,36 +1,13 @@
/* TODO: remove after https://github.com/eclipse-theia/theia/pull/9256/ */ /* Naive way of hiding the debug widget when the debug functionality is disabled https://github.com/arduino/arduino-ide/issues/14 */
.theia-debug-container .debug-toolbar.hidden,
/* To fix colors in Theia. */ .theia-debug-container .theia-session-container.hidden {
.theia-debug-hover-title.number, visibility: hidden;
.theia-debug-console-variable.number {
color: var(--theia-variable-number-variable-color);
}
.theia-debug-hover-title.boolean,
.theia-debug-console-variable.boolean {
color: var(--theia-variable-boolean-variable-color);
}
.theia-debug-hover-title.string,
.theia-debug-console-variable.string {
color: var(--theia-variable-string-variable-color);
} }
/* To unset the default debug hover dimension. */ .theia-debug-container .status-message {
.theia-debug-hover { font-family: "Open Sans";
min-width: unset; font-style: normal;
min-height: unset; font-size: 12px;
width: unset;
height: unset;
}
/* To adjust the left padding in the hover title. */ padding: 10px;
.theia-debug-hover-title {
padding-left: 5px;
}
/* Use the default Theia dimensions only iff the expression is complex (`!!expression.hasChildren~) */
.theia-debug-hover.complex-value {
min-width: 324px;
min-height: 324px;
width: 324px;
height: 324px;
} }

View File

@@ -56,14 +56,23 @@
} }
.p-Widget.dialogOverlay .dialogControl .spinner, .p-Widget.dialogOverlay .dialogControl .spinner,
.p-Widget.dialogOverlay .dialogBlock .dialogContent .dialogSection .dialogRow .spinner { .p-Widget.dialogOverlay
.dialogBlock
.dialogContent
.dialogSection
.dialogRow
.spinner {
background: var(--theia-icon-loading) center center no-repeat; background: var(--theia-icon-loading) center center no-repeat;
animation: theia-spin 1.25s linear infinite; animation: theia-spin 1.25s linear infinite;
width: 30px; width: 30px;
height: 30px; height: 30px;
} }
.p-Widget.dialogOverlay .dialogBlock .dialogContent .dialogSection .dialogRow:first-child { .p-Widget.dialogOverlay
.dialogBlock
.dialogContent
.dialogSection
.dialogRow:first-child {
margin-top: 0px; margin-top: 0px;
height: 32px; height: 32px;
} }
@@ -78,7 +87,7 @@
} }
.fa.disabled { .fa.disabled {
opacity: .4; opacity: 0.4;
} }
@media only screen and (max-height: 560px) { @media only screen and (max-height: 560px) {

View File

@@ -1,7 +1,9 @@
/* Show the dirty indicator on unclosable widgets. On hover, it should still show the dot instead of the X. */ /* Show the dirty indicator on unclosable widgets. On hover, it should still show the dot instead of the X. */
/* https://github.com/arduino/arduino-pro-ide/issues/380 */ /* https://github.com/arduino/arduino-pro-ide/issues/380 */
.p-TabBar.theia-app-centers .p-TabBar-tab.p-mod-closable.a-mod-uncloseable.theia-mod-dirty > .p-TabBar-tabCloseIcon:before { .p-TabBar.theia-app-centers
content: "\ea71"; .p-TabBar-tab.p-mod-closable.a-mod-uncloseable.theia-mod-dirty
> .p-TabBar-tabCloseIcon:before {
content: "\ea71";
} }
.monaco-list-row.show-file-icons.focused { .monaco-list-row.show-file-icons.focused {

View File

@@ -1,31 +1,31 @@
#firmware-uploader-dialog-container > .dialogBlock { #firmware-uploader-dialog-container > .dialogBlock {
width: 600px; width: 600px;
} }
.firmware-uploader-dialog .theia-select { .firmware-uploader-dialog .theia-select {
border: none !important; border: none !important;
} }
.firmware-uploader-dialog .arduino-select__control { .firmware-uploader-dialog .arduino-select__control {
height: 31px; height: 31px;
background: var(--theia-input-background) !important; background: var(--theia-input-background) !important;
} }
.firmware-uploader-dialog .dialogRow > button{ .firmware-uploader-dialog .dialogRow > button {
margin-right: 3px; margin-right: 3px;
} }
.firmware-uploader-dialog #firmware-select { .firmware-uploader-dialog #firmware-select {
flex: unset; flex: unset;
} }
.firmware-uploader-dialog .success { .firmware-uploader-dialog .success {
color: #1DA086; color: #1da086;
} }
.firmware-uploader-dialog .warn { .firmware-uploader-dialog .warn {
color: #C11F09; color: #c11f09;
} }
.firmware-uploader-dialog .status-icon { .firmware-uploader-dialog .status-icon {
margin-right: 10px; margin-right: 10px;
} }

View File

@@ -1,699 +1,698 @@
@font-face { @font-face {
font-family: 'Open Sans'; font-family: "Open Sans";
src: url('fonts/OpenSans-Regular-webfont.woff') format('woff'); src: url("fonts/OpenSans-Regular-webfont.woff") format("woff");
} }
@font-face { @font-face {
font-family: 'Open Sans Bold'; font-family: "Open Sans Bold";
src: url('fonts/OpenSans-Bold-webfont.woff') format('woff'); src: url("fonts/OpenSans-Bold-webfont.woff") format("woff");
} }
@font-face { @font-face {
font-family: 'FontAwesome'; font-family: "FontAwesome";
src: src: url("fonts/FontAwesome.ttf?h959em") format("truetype"),
url('fonts/FontAwesome.ttf?h959em') format('truetype'), url("fonts/FontAwesome.woff?h959em") format("woff"),
url('fonts/FontAwesome.woff?h959em') format('woff'), url("fonts/FontAwesome.svg?h959em#FontAwesome") format("svg");
url('fonts/FontAwesome.svg?h959em#FontAwesome') format('svg'); font-weight: normal;
font-weight: normal; font-style: normal;
font-style: normal; font-display: block;
font-display: block;
} }
.fa { .fa {
/* use !important to prevent issues with browser extensions that change fonts */ /* use !important to prevent issues with browser extensions that change fonts */
font-family: 'FontAwesome' !important; font-family: "FontAwesome" !important;
speak: never; speak: never;
font-style: normal; font-style: normal;
font-weight: normal; font-weight: normal;
font-variant: normal; font-variant: normal;
text-transform: none; text-transform: none;
line-height: 1; line-height: 1;
/* Better Font Rendering =========== */ /* Better Font Rendering =========== */
-webkit-font-smoothing: antialiased; -webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale; -moz-osx-font-smoothing: grayscale;
} }
.fa-arduino-verify:before { .fa-arduino-verify:before {
content: "\e90b"; content: "\e90b";
} }
.fa-arduino-upload:before { .fa-arduino-upload:before {
content: "\e90c"; content: "\e90c";
} }
.fa-arduino-monitor:before { .fa-arduino-monitor:before {
content: "\e90d"; content: "\e90d";
} }
.fa-arduino-sketch-tabs-menu:before { .fa-arduino-sketch-tabs-menu:before {
content: "\e90e"; content: "\e90e";
} }
.fa-arduino-plotter:before { .fa-arduino-plotter:before {
content: "\e90f"; content: "\e90f";
} }
.fa-fa-check:before { .fa-fa-check:before {
content: "\e90a"; content: "\e90a";
} }
.fa-arduino-technology-3dimensionscube:before { .fa-arduino-technology-3dimensionscube:before {
content: "\e906"; content: "\e906";
} }
.fa-arduino-technology-usb:before { .fa-arduino-technology-usb:before {
content: "\e907"; content: "\e907";
} }
.fa-arduino-technology-connection:before { .fa-arduino-technology-connection:before {
content: "\e908"; content: "\e908";
} }
.fa-arduino-technology-bluetooth:before { .fa-arduino-technology-bluetooth:before {
content: "\e909"; content: "\e909";
} }
.fa-arduino-debugger:before { .fa-arduino-debugger:before {
content: "\e905"; content: "\e905";
} }
.fa-arduino-search:before { .fa-arduino-search:before {
content: "\e901"; content: "\e901";
} }
.fa-arduino-boards:before { .fa-arduino-boards:before {
content: "\e902"; content: "\e902";
} }
.fa-arduino-library:before { .fa-arduino-library:before {
content: "\e903"; content: "\e903";
} }
.fa-arduino-folder:before { .fa-arduino-folder:before {
content: "\e904"; content: "\e904";
} }
.fa-reload:before { .fa-reload:before {
content: "\e900"; content: "\e900";
} }
.fa-asterisk:before { .fa-asterisk:before {
content: "\f069"; content: "\f069";
} }
.fa-plus:before { .fa-plus:before {
content: "\f067"; content: "\f067";
} }
.fa-question:before { .fa-question:before {
content: "\f128"; content: "\f128";
} }
.fa-minus:before { .fa-minus:before {
content: "\f068"; content: "\f068";
} }
.fa-music:before { .fa-music:before {
content: "\f001"; content: "\f001";
} }
.fa-search:before { .fa-search:before {
content: "\f002"; content: "\f002";
} }
.fa-envelope-o:before { .fa-envelope-o:before {
content: "\f003"; content: "\f003";
} }
.fa-heart:before { .fa-heart:before {
content: "\f004"; content: "\f004";
} }
.fa-star:before { .fa-star:before {
content: "\f005"; content: "\f005";
} }
.fa-star-o:before { .fa-star-o:before {
content: "\f006"; content: "\f006";
} }
.fa-user:before { .fa-user:before {
content: "\f007"; content: "\f007";
} }
.fa-film:before { .fa-film:before {
content: "\f008"; content: "\f008";
} }
.fa-th-large:before { .fa-th-large:before {
content: "\f009"; content: "\f009";
} }
.fa-th:before { .fa-th:before {
content: "\f00a"; content: "\f00a";
} }
.fa-th-list:before { .fa-th-list:before {
content: "\f00b"; content: "\f00b";
} }
.fa-close:before { .fa-close:before {
content: "\f00d"; content: "\f00d";
} }
.fa-remove:before { .fa-remove:before {
content: "\f00d"; content: "\f00d";
} }
.fa-times:before { .fa-times:before {
content: "\f00d"; content: "\f00d";
} }
.fa-search-plus:before { .fa-search-plus:before {
content: "\f00e"; content: "\f00e";
} }
.fa-search-minus:before { .fa-search-minus:before {
content: "\f010"; content: "\f010";
} }
.fa-power-off:before { .fa-power-off:before {
content: "\f011"; content: "\f011";
} }
.fa-signal:before { .fa-signal:before {
content: "\f012"; content: "\f012";
} }
.fa-cog:before { .fa-cog:before {
content: "\f013"; content: "\f013";
} }
.fa-gear:before { .fa-gear:before {
content: "\f013"; content: "\f013";
} }
.fa-trash-o:before { .fa-trash-o:before {
content: "\f014"; content: "\f014";
} }
.fa-home:before { .fa-home:before {
content: "\f015"; content: "\f015";
} }
.fa-file-o:before { .fa-file-o:before {
content: "\f016"; content: "\f016";
} }
.fa-clock-o:before { .fa-clock-o:before {
content: "\f017"; content: "\f017";
} }
.fa-download:before { .fa-download:before {
content: "\f019"; content: "\f019";
} }
.fa-arrow-circle-o-down:before { .fa-arrow-circle-o-down:before {
content: "\f01a"; content: "\f01a";
} }
.fa-arrow-circle-o-up:before { .fa-arrow-circle-o-up:before {
content: "\f01b"; content: "\f01b";
} }
.fa-inbox:before { .fa-inbox:before {
content: "\f01c"; content: "\f01c";
} }
.fa-play-circle-o:before { .fa-play-circle-o:before {
content: "\f01d"; content: "\f01d";
} }
.fa-repeat:before { .fa-repeat:before {
content: "\f01e"; content: "\f01e";
} }
.fa-rotate-right:before { .fa-rotate-right:before {
content: "\f01e"; content: "\f01e";
} }
.fa-refresh:before { .fa-refresh:before {
content: "\f021"; content: "\f021";
} }
.fa-list-alt:before { .fa-list-alt:before {
content: "\f022"; content: "\f022";
} }
.fa-lock:before { .fa-lock:before {
content: "\f023"; content: "\f023";
} }
.fa-volume-off:before { .fa-volume-off:before {
content: "\f026"; content: "\f026";
} }
.fa-volume-down:before { .fa-volume-down:before {
content: "\f027"; content: "\f027";
} }
.fa-volume-up:before { .fa-volume-up:before {
content: "\f028"; content: "\f028";
} }
.fa-qrcode:before { .fa-qrcode:before {
content: "\f029"; content: "\f029";
} }
.fa-tag:before { .fa-tag:before {
content: "\f02b"; content: "\f02b";
} }
.fa-tags:before { .fa-tags:before {
content: "\f02c"; content: "\f02c";
} }
.fa-book:before { .fa-book:before {
content: "\f02d"; content: "\f02d";
} }
.fa-print:before { .fa-print:before {
content: "\f02f"; content: "\f02f";
} }
.fa-text-height:before { .fa-text-height:before {
content: "\f034"; content: "\f034";
} }
.fa-text-width:before { .fa-text-width:before {
content: "\f035"; content: "\f035";
} }
.fa-align-left:before { .fa-align-left:before {
content: "\f036"; content: "\f036";
} }
.fa-align-center:before { .fa-align-center:before {
content: "\f037"; content: "\f037";
} }
.fa-align-right:before { .fa-align-right:before {
content: "\f038"; content: "\f038";
} }
.fa-align-justify:before { .fa-align-justify:before {
content: "\f039"; content: "\f039";
} }
.fa-list:before { .fa-list:before {
content: "\f03a"; content: "\f03a";
} }
.fa-dedent:before { .fa-dedent:before {
content: "\f03b"; content: "\f03b";
} }
.fa-outdent:before { .fa-outdent:before {
content: "\f03b"; content: "\f03b";
} }
.fa-indent:before { .fa-indent:before {
content: "\f03c"; content: "\f03c";
} }
.fa-pencil:before { .fa-pencil:before {
content: "\f040"; content: "\f040";
} }
.fa-adjust:before { .fa-adjust:before {
content: "\f042"; content: "\f042";
} }
.fa-edit:before { .fa-edit:before {
content: "\f044"; content: "\f044";
} }
.fa-pencil-square-o:before { .fa-pencil-square-o:before {
content: "\f044"; content: "\f044";
} }
.fa-share-square-o:before { .fa-share-square-o:before {
content: "\f045"; content: "\f045";
} }
.fa-check-square-o:before { .fa-check-square-o:before {
content: "\f046"; content: "\f046";
} }
.fa-arrows:before { .fa-arrows:before {
content: "\f047"; content: "\f047";
} }
.fa-step-backward:before { .fa-step-backward:before {
content: "\f048"; content: "\f048";
} }
.fa-fast-backward:before { .fa-fast-backward:before {
content: "\f049"; content: "\f049";
} }
.fa-backward:before { .fa-backward:before {
content: "\f04a"; content: "\f04a";
} }
.fa-play:before { .fa-play:before {
content: "\f04b"; content: "\f04b";
} }
.fa-pause:before { .fa-pause:before {
content: "\f04c"; content: "\f04c";
} }
.fa-stop:before { .fa-stop:before {
content: "\f04d"; content: "\f04d";
} }
.fa-forward:before { .fa-forward:before {
content: "\f04e"; content: "\f04e";
} }
.fa-fast-forward:before { .fa-fast-forward:before {
content: "\f050"; content: "\f050";
} }
.fa-step-forward:before { .fa-step-forward:before {
content: "\f051"; content: "\f051";
} }
.fa-eject:before { .fa-eject:before {
content: "\f052"; content: "\f052";
} }
.fa-chevron-left:before { .fa-chevron-left:before {
content: "\f053"; content: "\f053";
} }
.fa-chevron-right:before { .fa-chevron-right:before {
content: "\f054"; content: "\f054";
} }
.fa-plus-circle:before { .fa-plus-circle:before {
content: "\f055"; content: "\f055";
} }
.fa-minus-circle:before { .fa-minus-circle:before {
content: "\f056"; content: "\f056";
} }
.fa-times-circle:before { .fa-times-circle:before {
content: "\f057"; content: "\f057";
} }
.fa-check-circle:before { .fa-check-circle:before {
content: "\f058"; content: "\f058";
} }
.fa-question-circle:before { .fa-question-circle:before {
content: "\f059"; content: "\f059";
} }
.fa-info-circle:before { .fa-info-circle:before {
content: "\f05a"; content: "\f05a";
} }
.fa-crosshairs:before { .fa-crosshairs:before {
content: "\f05b"; content: "\f05b";
} }
.fa-times-circle-o:before { .fa-times-circle-o:before {
content: "\f05c"; content: "\f05c";
} }
.fa-check-circle-o:before { .fa-check-circle-o:before {
content: "\f05d"; content: "\f05d";
} }
.fa-ban:before { .fa-ban:before {
content: "\f05e"; content: "\f05e";
} }
.fa-arrow-left:before { .fa-arrow-left:before {
content: "\f060"; content: "\f060";
} }
.fa-arrow-right:before { .fa-arrow-right:before {
content: "\f061"; content: "\f061";
} }
.fa-arrow-up:before { .fa-arrow-up:before {
content: "\f062"; content: "\f062";
} }
.fa-arrow-down:before { .fa-arrow-down:before {
content: "\f063"; content: "\f063";
} }
.fa-mail-forward:before { .fa-mail-forward:before {
content: "\f064"; content: "\f064";
} }
.fa-share:before { .fa-share:before {
content: "\f064"; content: "\f064";
} }
.fa-expand:before { .fa-expand:before {
content: "\f065"; content: "\f065";
} }
.fa-compress:before { .fa-compress:before {
content: "\f066"; content: "\f066";
} }
.fa-exclamation-circle:before { .fa-exclamation-circle:before {
content: "\f06a"; content: "\f06a";
} }
.fa-eye:before { .fa-eye:before {
content: "\f06e"; content: "\f06e";
} }
.fa-eye-slash:before { .fa-eye-slash:before {
content: "\f070"; content: "\f070";
} }
.fa-exclamation-triangle:before { .fa-exclamation-triangle:before {
content: "\f071"; content: "\f071";
} }
.fa-warning:before { .fa-warning:before {
content: "\f071"; content: "\f071";
} }
.fa-calendar:before { .fa-calendar:before {
content: "\f073"; content: "\f073";
} }
.fa-random:before { .fa-random:before {
content: "\f074"; content: "\f074";
} }
.fa-comment:before { .fa-comment:before {
content: "\f075"; content: "\f075";
} }
.fa-chevron-up:before { .fa-chevron-up:before {
content: "\f077"; content: "\f077";
} }
.fa-chevron-down:before { .fa-chevron-down:before {
content: "\f078"; content: "\f078";
} }
.fa-retweet:before { .fa-retweet:before {
content: "\f079"; content: "\f079";
} }
.fa-folder:before { .fa-folder:before {
content: "\f07b"; content: "\f07b";
} }
.fa-folder-open:before { .fa-folder-open:before {
content: "\f07c"; content: "\f07c";
} }
.fa-arrows-v:before { .fa-arrows-v:before {
content: "\f07d"; content: "\f07d";
} }
.fa-arrows-h:before { .fa-arrows-h:before {
content: "\f07e"; content: "\f07e";
} }
.fa-cogs:before { .fa-cogs:before {
content: "\f085"; content: "\f085";
} }
.fa-gears:before { .fa-gears:before {
content: "\f085"; content: "\f085";
} }
.fa-star-half:before { .fa-star-half:before {
content: "\f089"; content: "\f089";
} }
.fa-heart-o:before { .fa-heart-o:before {
content: "\f08a"; content: "\f08a";
} }
.fa-sign-out:before { .fa-sign-out:before {
content: "\f08b"; content: "\f08b";
} }
.fa-thumb-tack:before { .fa-thumb-tack:before {
content: "\f08d"; content: "\f08d";
} }
.fa-external-link:before { .fa-external-link:before {
content: "\f08e"; content: "\f08e";
} }
.fa-sign-in:before { .fa-sign-in:before {
content: "\f090"; content: "\f090";
} }
.fa-upload:before { .fa-upload:before {
content: "\f093"; content: "\f093";
} }
.fa-square-o:before { .fa-square-o:before {
content: "\f096"; content: "\f096";
} }
.fa-bookmark-o:before { .fa-bookmark-o:before {
content: "\f097"; content: "\f097";
} }
.fa-hdd-o:before { .fa-hdd-o:before {
content: "\f0a0"; content: "\f0a0";
} }
.fa-bell-o:before { .fa-bell-o:before {
content: "\f0a2"; content: "\f0a2";
} }
.fa-certificate:before { .fa-certificate:before {
content: "\f0a3"; content: "\f0a3";
} }
.fa-arrow-circle-left:before { .fa-arrow-circle-left:before {
content: "\f0a8"; content: "\f0a8";
} }
.fa-arrow-circle-right:before { .fa-arrow-circle-right:before {
content: "\f0a9"; content: "\f0a9";
} }
.fa-arrow-circle-up:before { .fa-arrow-circle-up:before {
content: "\f0aa"; content: "\f0aa";
} }
.fa-arrow-circle-down:before { .fa-arrow-circle-down:before {
content: "\f0ab"; content: "\f0ab";
} }
.fa-wrench:before { .fa-wrench:before {
content: "\f0ad"; content: "\f0ad";
} }
.fa-tasks:before { .fa-tasks:before {
content: "\f0ae"; content: "\f0ae";
} }
.fa-filter:before { .fa-filter:before {
content: "\f0b0"; content: "\f0b0";
} }
.fa-briefcase:before { .fa-briefcase:before {
content: "\f0b1"; content: "\f0b1";
} }
.fa-arrows-alt:before { .fa-arrows-alt:before {
content: "\f0b2"; content: "\f0b2";
} }
.fa-cloud:before { .fa-cloud:before {
content: "\f0c2"; content: "\f0c2";
} }
.fa-copy:before { .fa-copy:before {
content: "\f0c5"; content: "\f0c5";
} }
.fa-files-o:before { .fa-files-o:before {
content: "\f0c5"; content: "\f0c5";
} }
.fa-floppy-o:before { .fa-floppy-o:before {
content: "\f0c7"; content: "\f0c7";
} }
.fa-save:before { .fa-save:before {
content: "\f0c7"; content: "\f0c7";
} }
.fa-square:before { .fa-square:before {
content: "\f0c8"; content: "\f0c8";
} }
.fa-bars:before { .fa-bars:before {
content: "\f0c9"; content: "\f0c9";
} }
.fa-navicon:before { .fa-navicon:before {
content: "\f0c9"; content: "\f0c9";
} }
.fa-reorder:before { .fa-reorder:before {
content: "\f0c9"; content: "\f0c9";
} }
.fa-list-ul:before { .fa-list-ul:before {
content: "\f0ca"; content: "\f0ca";
} }
.fa-list-ol:before { .fa-list-ol:before {
content: "\f0cb"; content: "\f0cb";
} }
.fa-table:before { .fa-table:before {
content: "\f0ce"; content: "\f0ce";
} }
.fa-caret-down:before { .fa-caret-down:before {
content: "\f0d7"; content: "\f0d7";
} }
.fa-caret-up:before { .fa-caret-up:before {
content: "\f0d8"; content: "\f0d8";
} }
.fa-caret-left:before { .fa-caret-left:before {
content: "\f0d9"; content: "\f0d9";
} }
.fa-caret-right:before { .fa-caret-right:before {
content: "\f0da"; content: "\f0da";
} }
.fa-columns:before { .fa-columns:before {
content: "\f0db"; content: "\f0db";
} }
.fa-sort:before { .fa-sort:before {
content: "\f0dc"; content: "\f0dc";
} }
.fa-unsorted:before { .fa-unsorted:before {
content: "\f0dc"; content: "\f0dc";
} }
.fa-sort-desc:before { .fa-sort-desc:before {
content: "\f0dd"; content: "\f0dd";
} }
.fa-sort-down:before { .fa-sort-down:before {
content: "\f0dd"; content: "\f0dd";
} }
.fa-sort-asc:before { .fa-sort-asc:before {
content: "\f0de"; content: "\f0de";
} }
.fa-sort-up:before { .fa-sort-up:before {
content: "\f0de"; content: "\f0de";
} }
.fa-rotate-left:before { .fa-rotate-left:before {
content: "\f0e2"; content: "\f0e2";
} }
.fa-undo:before { .fa-undo:before {
content: "\f0e2"; content: "\f0e2";
} }
.fa-file-text-o:before { .fa-file-text-o:before {
content: "\f0f6"; content: "\f0f6";
} }
.fa-plus-square:before { .fa-plus-square:before {
content: "\f0fe"; content: "\f0fe";
} }
.fa-angle-double-left:before { .fa-angle-double-left:before {
content: "\f100"; content: "\f100";
} }
.fa-angle-double-right:before { .fa-angle-double-right:before {
content: "\f101"; content: "\f101";
} }
.fa-angle-double-up:before { .fa-angle-double-up:before {
content: "\f102"; content: "\f102";
} }
.fa-angle-double-down:before { .fa-angle-double-down:before {
content: "\f103"; content: "\f103";
} }
.fa-angle-left:before { .fa-angle-left:before {
content: "\f104"; content: "\f104";
} }
.fa-angle-right:before { .fa-angle-right:before {
content: "\f105"; content: "\f105";
} }
.fa-angle-up:before { .fa-angle-up:before {
content: "\f106"; content: "\f106";
} }
.fa-angle-down:before { .fa-angle-down:before {
content: "\f107"; content: "\f107";
} }
.fa-circle-o:before { .fa-circle-o:before {
content: "\f10c"; content: "\f10c";
} }
.fa-spinner:before { .fa-spinner:before {
content: "\f110"; content: "\f110";
} }
.fa-circle:before { .fa-circle:before {
content: "\f111"; content: "\f111";
} }
.fa-mail-reply:before { .fa-mail-reply:before {
content: "\f112"; content: "\f112";
} }
.fa-reply:before { .fa-reply:before {
content: "\f112"; content: "\f112";
} }
.fa-folder-o:before { .fa-folder-o:before {
content: "\f114"; content: "\f114";
} }
.fa-folder-open-o:before { .fa-folder-open-o:before {
content: "\f115"; content: "\f115";
} }
.fa-keyboard-o:before { .fa-keyboard-o:before {
content: "\f11c"; content: "\f11c";
} }
.fa-terminal:before { .fa-terminal:before {
content: "\f120"; content: "\f120";
} }
.fa-code:before { .fa-code:before {
content: "\f121"; content: "\f121";
} }
.fa-mail-reply-all:before { .fa-mail-reply-all:before {
content: "\f122"; content: "\f122";
} }
.fa-reply-all:before { .fa-reply-all:before {
content: "\f122"; content: "\f122";
} }
.fa-star-half-empty:before { .fa-star-half-empty:before {
content: "\f123"; content: "\f123";
} }
.fa-star-half-full:before { .fa-star-half-full:before {
content: "\f123"; content: "\f123";
} }
.fa-star-half-o:before { .fa-star-half-o:before {
content: "\f123"; content: "\f123";
} }
.fa-crop:before { .fa-crop:before {
content: "\f125"; content: "\f125";
} }
.fa-code-fork:before { .fa-code-fork:before {
content: "\f126"; content: "\f126";
} }
.fa-chain-broken:before { .fa-chain-broken:before {
content: "\f127"; content: "\f127";
} }
.fa-unlink:before { .fa-unlink:before {
content: "\f127"; content: "\f127";
} }
.fa-info:before { .fa-info:before {
content: "\f129"; content: "\f129";
} }
.fa-exclamation:before { .fa-exclamation:before {
content: "\f12a"; content: "\f12a";
} }
.fa-rocket:before { .fa-rocket:before {
content: "\f135"; content: "\f135";
} }
.fa-maxcdn:before { .fa-maxcdn:before {
content: "\f136"; content: "\f136";
} }
.fa-chevron-circle-left:before { .fa-chevron-circle-left:before {
content: "\f137"; content: "\f137";
} }
.fa-chevron-circle-right:before { .fa-chevron-circle-right:before {
content: "\f138"; content: "\f138";
} }
.fa-chevron-circle-up:before { .fa-chevron-circle-up:before {
content: "\f139"; content: "\f139";
} }
.fa-chevron-circle-down:before { .fa-chevron-circle-down:before {
content: "\f13a"; content: "\f13a";
} }
.fa-ellipsis-h:before { .fa-ellipsis-h:before {
content: "\f141"; content: "\f141";
} }
.fa-long-arrow-down:before { .fa-long-arrow-down:before {
content: "\f175"; content: "\f175";
} }
.fa-long-arrow-up:before { .fa-long-arrow-up:before {
content: "\f176"; content: "\f176";
} }
.fa-long-arrow-left:before { .fa-long-arrow-left:before {
content: "\f177"; content: "\f177";
} }
.fa-long-arrow-right:before { .fa-long-arrow-right:before {
content: "\f178"; content: "\f178";
} }
.fa-microchip:before { .fa-microchip:before {
content: "\f2db"; content: "\f2db";
} }
.fa-arduino-cloud-download:before { .fa-arduino-cloud-download:before {
content: "\e910"; content: "\e910";
} }
.fa-arduino-cloud-upload:before { .fa-arduino-cloud-upload:before {
content: "\e914"; content: "\e914";
} }
.fa-arduino-cloud:before { .fa-arduino-cloud:before {
content: "\e915"; content: "\e915";
} }
.fa-arduino-cloud-filled:before { .fa-arduino-cloud-filled:before {
content: "\e912"; content: "\e912";
} }
.fa-arduino-cloud-offline:before { .fa-arduino-cloud-offline:before {
content: "\e913"; content: "\e913";
} }
.fa-arduino-cloud-filled-offline:before { .fa-arduino-cloud-filled-offline:before {
content: "\e911"; content: "\e911";
} }

View File

@@ -1,124 +1,124 @@
#ide-updater-dialog-container > .dialogBlock { #ide-updater-dialog-container > .dialogBlock {
width: 546px; width: 546px;
} }
.ide-updater-dialog .bold { .ide-updater-dialog .bold {
font-weight: bold; font-weight: bold;
} }
.ide-updater-dialog--pre-download { .ide-updater-dialog--pre-download {
display: flex; display: flex;
} }
.ide-updater-dialog--downloading { .ide-updater-dialog--downloading {
flex: 1; flex: 1;
} }
.ide-updater-dialog--logo-container { .ide-updater-dialog--logo-container {
margin-right: var(--arduino-button-height); margin-right: var(--arduino-button-height);
} }
.ide-updater-dialog--logo { .ide-updater-dialog--logo {
background: url('./ide-logo.png') round; background: url("./ide-logo.png") round;
width: 52px; width: 52px;
height: 52px; height: 52px;
} }
.dialogContent.ide-updater-dialog .dialogContent.ide-updater-dialog
.ide-updater-dialog--content .ide-updater-dialog--content
.ide-updater-dialog--new-version-text.dialogSection { .ide-updater-dialog--new-version-text.dialogSection {
display: flex; display: flex;
flex: 1; flex: 1;
flex-direction: column; flex-direction: column;
margin-top: 0; margin-top: 0;
min-width: 0; min-width: 0;
} }
.ide-updater-dialog .changelog { .ide-updater-dialog .changelog {
color: var(--theia-editor-foreground); color: var(--theia-editor-foreground);
background-color: var(--theia-editor-background); background-color: var(--theia-editor-background);
font-size: 12px; font-size: 12px;
overflow: auto; overflow: auto;
padding: 0 12px; padding: 0 12px;
cursor: text; cursor: text;
width: 100%; width: 100%;
} }
.ide-updater-dialog .changelog .fallback { .ide-updater-dialog .changelog .fallback {
min-height: 180px; min-height: 180px;
width: 100%; width: 100%;
display: flex; display: flex;
} }
.ide-updater-dialog .changelog .fallback .spinner { .ide-updater-dialog .changelog .fallback .spinner {
align-self: center; align-self: center;
} }
.dialogContent.ide-updater-dialog .dialogContent.ide-updater-dialog
.ide-updater-dialog--content .ide-updater-dialog--content
.ide-updater-dialog--new-version-text .ide-updater-dialog--new-version-text
.dialogRow.changelog-container { .dialogRow.changelog-container {
align-items: flex-start; align-items: flex-start;
border: 1px solid var(--theia-editorWidget-border); border: 1px solid var(--theia-editorWidget-border);
border-radius: 2px; border-radius: 2px;
overflow: auto; overflow: auto;
max-height: 180px; max-height: 180px;
} }
.ide-updater-dialog .changelog a { .ide-updater-dialog .changelog a {
color: var(--theia-textLink-foreground); color: var(--theia-textLink-foreground);
} }
.ide-updater-dialog .changelog a:hover { .ide-updater-dialog .changelog a:hover {
text-decoration: underline; text-decoration: underline;
cursor: pointer; cursor: pointer;
} }
.ide-updater-dialog .changelog code { .ide-updater-dialog .changelog code {
background: var(--theia-textBlockQuote-background); background: var(--theia-textBlockQuote-background);
border-radius: 2px; border-radius: 2px;
padding: 0 2px; padding: 0 2px;
} }
.ide-updater-dialog .changelog a code { .ide-updater-dialog .changelog a code {
color: var(--theia-textLink-foreground); color: var(--theia-textLink-foreground);
} }
.ide-updater-dialog .buttons-container { .ide-updater-dialog .buttons-container {
display: flex; display: flex;
justify-content: flex-end; justify-content: flex-end;
margin-top: var(--arduino-button-height); margin-top: var(--arduino-button-height);
} }
.ide-updater-dialog .buttons-container a.theia-button { .ide-updater-dialog .buttons-container a.theia-button {
text-decoration: none; text-decoration: none;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
} }
.ide-updater-dialog .buttons-container a.theia-button:hover { .ide-updater-dialog .buttons-container a.theia-button:hover {
color: var(--theia-button-foreground); color: var(--theia-button-foreground);
} }
.ide-updater-dialog .buttons-container .push { .ide-updater-dialog .buttons-container .push {
margin-right: auto; margin-right: auto;
} }
.ide-updater-dialog--content { .ide-updater-dialog--content {
max-height: 100%; max-height: 100%;
overflow: hidden; overflow: hidden;
display: flex; display: flex;
} }
#ide-updater-dialog-container .skip-version-button { #ide-updater-dialog-container .skip-version-button {
margin-left: 79px; margin-left: 79px;
margin-right: auto; margin-right: auto;
} }
/* https://github.com/arduino/arduino-ide/pull/2027#issuecomment-1533174638 */ /* https://github.com/arduino/arduino-ide/pull/2027#issuecomment-1533174638 */
/* https://github.com/eclipse-theia/theia/commit/1b5ff9ee459df14cedc0e8266dd02dae93fcd1bf#diff-d8d45a890507a01141c010ad4a6891edf2ae727cfa6dfe604cebbd667812cccbR68 */ /* https://github.com/eclipse-theia/theia/commit/1b5ff9ee459df14cedc0e8266dd02dae93fcd1bf#diff-d8d45a890507a01141c010ad4a6891edf2ae727cfa6dfe604cebbd667812cccbR68 */
/* Use normal whitespace handling for the changelog content in the update dialog. */ /* Use normal whitespace handling for the changelog content in the update dialog. */
.p-Widget.dialogOverlay .dialogContent.ide-updater-dialog { .p-Widget.dialogOverlay .dialogContent.ide-updater-dialog {
white-space: normal; white-space: normal;
} }

View File

@@ -1,175 +1,177 @@
@import './list-widget.css'; @import "./list-widget.css";
@import './boards-config-dialog.css'; @import "./boards-config-dialog.css";
@import './main.css'; @import "./main.css";
@import './dialogs.css'; @import "./dialogs.css";
@import './monitor.css'; @import "./monitor.css";
@import './arduino-select.css'; @import "./arduino-select.css";
@import './status-bar.css'; @import "./status-bar.css";
@import './terminal.css'; @import "./terminal.css";
@import './editor.css'; @import "./editor.css";
@import './settings-dialog.css'; @import "./settings-dialog.css";
@import './firmware-uploader-dialog.css'; @import "./firmware-uploader-dialog.css";
@import './ide-updater-dialog.css'; @import "./ide-updater-dialog.css";
@import './certificate-uploader-dialog.css'; @import "./certificate-uploader-dialog.css";
@import './user-fields-dialog.css'; @import "./user-fields-dialog.css";
@import './debug.css'; @import "./debug.css";
@import './sketchbook.css'; @import "./sketchbook.css";
@import './cloud-sketchbook.css'; @import "./cloud-sketchbook.css";
@import './fonts.css'; @import "./fonts.css";
@import './custom-codicon.css'; @import "./custom-codicon.css";
@import './progress-bar.css'; @import "./progress-bar.css";
@import './settings-step-input.css'; @import "./settings-step-input.css";
:root { :root {
--arduino-button-height: 28px; --arduino-button-height: 28px;
} }
/* Revive of the `--theia-icon-loading`. The variable has been removed from Theia while IDE2 still uses is. */ /* Revive of the `--theia-icon-loading`. The variable has been removed from Theia while IDE2 still uses is. */
/* The SVG icons are still part of Theia (1.31.1) */ /* The SVG icons are still part of Theia (1.31.1) */
/* https://github.com/arduino/arduino-ide/pull/1662#issuecomment-1324997134 */ /* https://github.com/arduino/arduino-ide/pull/1662#issuecomment-1324997134 */
body { body {
--theia-icon-loading: url(../icons/loading-light.svg); --theia-icon-loading: url(../icons/loading-light.svg);
--theia-icon-loading-warning: url(../icons/loading-dark.svg); --theia-icon-loading-warning: url(../icons/loading-dark.svg);
} }
body.theia-dark { body.theia-dark {
--theia-icon-loading: url(../icons/loading-dark.svg); --theia-icon-loading: url(../icons/loading-dark.svg);
--theia-icon-loading-warning: url(../icons/loading-light.svg); --theia-icon-loading-warning: url(../icons/loading-light.svg);
} }
.theia-input.warning:focus { .theia-input.warning:focus {
outline-width: 1px; outline-width: 1px;
outline-style: solid; outline-style: solid;
outline-offset: -1px; outline-offset: -1px;
opacity: 1 !important; opacity: 1 !important;
color: var(--theia-warningForeground); color: var(--theia-warningForeground);
background-color: var(--theia-warningBackground); background-color: var(--theia-warningBackground);
} }
.theia-input.warning { .theia-input.warning {
background-color: var(--theia-warningBackground); background-color: var(--theia-warningBackground);
} }
.theia-input.warning::placeholder { .theia-input.warning::placeholder {
color: var(--theia-warningForeground); color: var(--theia-warningForeground);
background-color: var(--theia-warningBackground); background-color: var(--theia-warningBackground);
} }
.theia-input.error:focus { .theia-input.error:focus {
outline-width: 1px; outline-width: 1px;
outline-style: solid; outline-style: solid;
outline-offset: -1px; outline-offset: -1px;
opacity: 1 !important; opacity: 1 !important;
color: var(--theia-errorForeground); color: var(--theia-errorForeground);
background-color: var(--theia-errorBackground); background-color: var(--theia-errorBackground);
} }
.theia-input.error { .theia-input.error {
background-color: var(--theia-errorBackground); background-color: var(--theia-errorBackground);
} }
.theia-input.error::placeholder { .theia-input.error::placeholder {
color: var(--theia-errorForeground); color: var(--theia-errorForeground);
background-color: var(--theia-errorBackground); background-color: var(--theia-errorBackground);
} }
/* Makes the sidepanel a bit wider when opening the widget */ /* Makes the sidepanel a bit wider when opening the widget */
.p-DockPanel-widget { .p-DockPanel-widget {
min-width: 220px; min-width: 220px;
min-height: 20px; min-height: 20px;
height: 220px; height: 220px;
} }
/* Overrule the default Theia CSS button styles. */ /* Overrule the default Theia CSS button styles. */
button.theia-button, button.theia-button,
.theia-button { .theia-button {
align-items: center; align-items: center;
display: flex; display: flex;
font-family: 'Open Sans Bold',sans-serif; font-family: "Open Sans Bold", sans-serif;
font-style: normal; font-style: normal;
font-weight: 700; font-weight: 700;
font-size: 14px; font-size: 14px;
justify-content: center; justify-content: center;
cursor: pointer; cursor: pointer;
letter-spacing: .01em; letter-spacing: 0.01em;
line-height: 24px; line-height: 24px;
outline: none; outline: none;
padding: 0 16px; padding: 0 16px;
position: relative; position: relative;
text-align: center; text-align: center;
text-decoration: none; text-decoration: none;
border-width: 2px; border-width: 2px;
border-radius: 32px; border-radius: 32px;
text-transform: uppercase; text-transform: uppercase;
transition: none; transition: none;
box-shadow: none; box-shadow: none;
} }
button.theia-button { button.theia-button {
height: var(--arduino-button-height); height: var(--arduino-button-height);
max-width: none; max-width: none;
} }
.theia-button:active, .theia-button:active,
.theia-button:focus { .theia-button:focus {
box-shadow: 0 0 0 2px var(--theia-focusBorder); box-shadow: 0 0 0 2px var(--theia-focusBorder);
} }
button.theia-button.secondary { button.theia-button.secondary {
border: 2px solid var(--theia-secondaryButton-foreground); border: 2px solid var(--theia-secondaryButton-foreground);
} }
button.theia-button[disabled], .theia-button[disabled] { button.theia-button[disabled],
opacity: 0.5; .theia-button[disabled] {
color: var(--theia-button-foreground); opacity: 0.5;
background-color: var(--theia-button-background); color: var(--theia-button-foreground);
background-color: var(--theia-button-background);
} }
button.secondary[disabled], .theia-button.secondary[disabled] { button.secondary[disabled],
color: var(--theia-secondaryButton-foreground); .theia-button.secondary[disabled] {
background-color: var(--theia-secondaryButton-background); color: var(--theia-secondaryButton-foreground);
background-color: var(--theia-secondaryButton-background);
} }
button.theia-button.message-box-dialog-button { button.theia-button.message-box-dialog-button {
white-space: nowrap; white-space: nowrap;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
display: inline-block; display: inline-block;
} }
/* To make the progress-bar slightly thicker, and use the color from the status bar */ /* To make the progress-bar slightly thicker, and use the color from the status bar */
.theia-progress-bar-container { .theia-progress-bar-container {
width: 100%; width: 100%;
height: 4px; height: 4px;
} }
.theia-progress-bar { .theia-progress-bar {
height: 4px; height: 4px;
width: 3%; width: 3%;
animation: progress-animation 1.3s 0s infinite animation: progress-animation 1.3s 0s infinite
cubic-bezier(0.645, 0.045, 0.355, 1); cubic-bezier(0.645, 0.045, 0.355, 1);
} }
.theia-notification-item-progressbar { .theia-notification-item-progressbar {
height: 4px; height: 4px;
width: 66%; width: 66%;
} }
.flex-line { .flex-line {
display: flex; display: flex;
align-items: center; align-items: center;
white-space: nowrap; white-space: nowrap;
} }
.fa-reload { .fa-reload {
font-size: 14px; font-size: 14px;
} }
.debug-toolbar .debug-action>div { .debug-toolbar .debug-action > div {
font-family: var(--theia-ui-font-family); font-family: var(--theia-ui-font-family);
font-size: var(--theia-ui-font-size0); font-size: var(--theia-ui-font-size0);
display: flex; display: flex;
align-items: center; align-items: center;
align-self: center; align-self: center;
justify-content: center; justify-content: center;
min-height: inherit; min-height: inherit;
} }

View File

@@ -1,6 +1,6 @@
.library-tab-icon { .library-tab-icon {
-webkit-mask: url('../icons/library-tab-icon.svg'); -webkit-mask: url("../icons/library-tab-icon.svg");
mask: url('../icons/library-tab-icon.svg'); mask: url("../icons/library-tab-icon.svg");
} }
.arduino-list-widget { .arduino-list-widget {
@@ -82,7 +82,7 @@
} }
.component-list-item .header .title .name { .component-list-item .header .title .name {
font-family: 'Open Sans Bold'; font-family: "Open Sans Bold";
font-style: normal; font-style: normal;
font-weight: 700; font-weight: 700;
font-size: 14px; font-size: 14px;
@@ -112,7 +112,9 @@
display: inline-block; display: inline-block;
justify-self: end; justify-self: end;
text-align: center; text-align: center;
background-color: var(--theia-arduino-toolbar-dropdown-option-backgroundHover); background-color: var(
--theia-arduino-toolbar-dropdown-option-backgroundHover
);
padding: 2px 4px 2px 4px; padding: 2px 4px 2px 4px;
font-size: 12px; font-size: 12px;
max-height: calc(1em + 4px); max-height: calc(1em + 4px);
@@ -131,7 +133,7 @@
display: flex; display: flex;
flex-direction: column; flex-direction: column;
padding-top: 4px; padding-top: 4px;
font-family: 'Open Sans'; font-family: "Open Sans";
font-style: normal; font-style: normal;
font-weight: 400; font-weight: 400;
font-size: 12px; font-size: 12px;
@@ -175,8 +177,20 @@
max-width: 8px; max-width: 8px;
} }
div.filterable-list-container > div > div > div > div:nth-child(1) > div.separator :first-child, div.filterable-list-container
div.filterable-list-container > div > div > div > div:nth-child(1) > div.separator :last-child { > div
> div
> div
> div:nth-child(1)
> div.separator
:first-child,
div.filterable-list-container
> div
> div
> div
> div:nth-child(1)
> div.separator
:last-child {
display: none; display: none;
} }
@@ -202,11 +216,11 @@ div.filterable-list-container > div > div > div > div:nth-child(1) > div.separat
} }
.component-list-item .theia-button.secondary.no-border { .component-list-item .theia-button.secondary.no-border {
border: 2px solid var(--theia-button-foreground) border: 2px solid var(--theia-button-foreground);
} }
.component-list-item .theia-button.secondary.no-border:hover { .component-list-item .theia-button.secondary.no-border:hover {
border: 2px solid var(--theia-secondaryButton-foreground) border: 2px solid var(--theia-secondaryButton-foreground);
} }
.component-list-item .theia-button { .component-list-item .theia-button {

View File

@@ -1,11 +1,12 @@
#theia-bottom-content-panel .p-TabBar[data-orientation='horizontal'].theia-app-bottom { #theia-bottom-content-panel
.p-TabBar[data-orientation="horizontal"].theia-app-bottom {
background: var(--theia-editorGroupHeader-tabsBackground); background: var(--theia-editorGroupHeader-tabsBackground);
} }
/* Avoid the Intellisense widget may be cover by the bottom panel partially. /* Avoid the Intellisense widget may be cover by the bottom panel partially.
TODO: This issue may be resolved after monaco-editor upgrade */ TODO: This issue may be resolved after monaco-editor upgrade */
#theia-main-content-panel { #theia-main-content-panel {
z-index: auto z-index: auto;
} }
#theia-main-content-panel div[id^="code-editor-opener"] { #theia-main-content-panel div[id^="code-editor-opener"] {
@@ -23,7 +24,6 @@
border-radius: 1px; border-radius: 1px;
} }
.p-TabBar-toolbar .item.arduino-tool-item > div { .p-TabBar-toolbar .item.arduino-tool-item > div {
display: flex; display: flex;
justify-content: center; justify-content: center;
@@ -42,9 +42,15 @@
background: var(--theia-arduino-toolbar-button-hoverBackground); background: var(--theia-arduino-toolbar-button-hoverBackground);
} }
.p-TabBar-toolbar .item.arduino-tool-item.enabled:hover > div.toggle-serial-monitor, .p-TabBar-toolbar
.p-TabBar-toolbar .item.arduino-tool-item.enabled:hover > div.toggle-serial-plotter { .item.arduino-tool-item.enabled:hover
background-color: var(--theia-arduino-toolbar-button-secondary-hoverBackground); > div.toggle-serial-monitor,
.p-TabBar-toolbar
.item.arduino-tool-item.enabled:hover
> div.toggle-serial-plotter {
background-color: var(
--theia-arduino-toolbar-button-secondary-hoverBackground
);
border-radius: 14px; border-radius: 14px;
} }
@@ -55,14 +61,14 @@
} }
.item.arduino-tool-item.toggled { .item.arduino-tool-item.toggled {
background-color: unset; background-color: unset;
opacity: 1; opacity: 1;
border: none; border: none;
} }
.item.arduino-tool-item.toggled .arduino-verify-sketch--toolbar, .item.arduino-tool-item.toggled .arduino-verify-sketch--toolbar,
.item.arduino-tool-item.toggled .arduino-upload-sketch--toolbar { .item.arduino-tool-item.toggled .arduino-upload-sketch--toolbar {
background-color: var(--theia-arduino-toolbar-toggleBackground) !important; background-color: var(--theia-arduino-toolbar-toggleBackground) !important;
} }
.arduino-tool-icon { .arduino-tool-icon {
@@ -91,7 +97,7 @@
} }
.arduino-start-debug-icon { .arduino-start-debug-icon {
-webkit-mask: url('../icons/debug-dark.svg') 50% 60%; -webkit-mask: url("../icons/debug-dark.svg") 50% 60%;
-webkit-mask-size: 70%; -webkit-mask-size: 70%;
-webkit-mask-repeat: no-repeat; -webkit-mask-repeat: no-repeat;
display: flex; display: flex;
@@ -175,7 +181,12 @@
background-color: var(--theia-terminal-background); background-color: var(--theia-terminal-background);
} }
.theia-output .monaco-editor .lines-content.monaco-editor-background .view-lines .view-line .mtk1:not(.theia-output-error):not(.theia-output-warning) { .theia-output
.monaco-editor
.lines-content.monaco-editor-background
.view-lines
.view-line
.mtk1:not(.theia-output-error):not(.theia-output-warning) {
color: var(--theia-terminal-foreground); color: var(--theia-terminal-foreground);
} }

View File

@@ -1,6 +1,6 @@
.monitor-tab-icon { .monitor-tab-icon {
-webkit-mask: url('../icons/monitor-tab-icon.svg'); -webkit-mask: url("../icons/monitor-tab-icon.svg");
mask: url('../icons/monitor-tab-icon.svg'); mask: url("../icons/monitor-tab-icon.svg");
} }
.serial-monitor { .serial-monitor {
@@ -10,8 +10,8 @@
} }
.serial-monitor-messages { .serial-monitor-messages {
white-space: 'pre'; white-space: "pre";
font-family: monospace font-family: monospace;
} }
.serial-monitor-messages pre { .serial-monitor-messages pre {

View File

@@ -1,32 +1,32 @@
.progress-bar { .progress-bar {
margin-top: 20px; margin-top: 20px;
} }
.progress-bar--outer { .progress-bar--outer {
background: var(--theia-editorWidget-background); background: var(--theia-editorWidget-background);
border-radius: 11px; border-radius: 11px;
height: 6px; height: 6px;
position: relative; position: relative;
overflow: hidden; overflow: hidden;
} }
.progress-bar--inner { .progress-bar--inner {
transition: width 1s; transition: width 1s;
height: 100%; height: 100%;
background: var(--theia-progressBar-background); background: var(--theia-progressBar-background);
border-radius: 11px; border-radius: 11px;
} }
.progress-bar--percentage { .progress-bar--percentage {
align-items: flex-end; align-items: flex-end;
display: flex; display: flex;
height: 40px; height: 40px;
justify-content: center; justify-content: center;
margin-top: 10px; margin-top: 10px;
width: 100%; width: 100%;
} }
.progress-bar--percentage-text { .progress-bar--percentage-text {
font-size: 14px; font-size: 14px;
line-height: 24px; line-height: 24px;
} }

View File

@@ -1,5 +1,5 @@
#arduino-settings-dialog-container > .dialogBlock { #arduino-settings-dialog-container > .dialogBlock {
height: 531px;; height: 531px;
max-width: 740px !important; max-width: 740px !important;
width: calc(100% - 96px); width: calc(100% - 96px);
} }

View File

@@ -1,5 +1,5 @@
.settings-step-input-container { .settings-step-input-container {
position: relative position: relative;
} }
.settings-step-input-element::-webkit-inner-spin-button, .settings-step-input-element::-webkit-inner-spin-button,
@@ -25,7 +25,7 @@
right: 14px; right: 14px;
} }
.settings-step-input-container:hover>.settings-step-input-buttons-container { .settings-step-input-container:hover > .settings-step-input-buttons-container {
display: flex; display: flex;
} }

View File

@@ -1,37 +1,36 @@
.sketchbook-tab-icon { .sketchbook-tab-icon {
-webkit-mask: url('./sketchbook.svg'); -webkit-mask: url("./sketchbook.svg");
mask: url('./sketchbook.svg'); mask: url("./sketchbook.svg");
} }
.p-TabBar-tabIcon.sketchbook-tree-icon { .p-TabBar-tabIcon.sketchbook-tree-icon {
background-color: var(--theia-foreground); background-color: var(--theia-foreground);
-webkit-mask: url(./sketchbook-tree-icon.svg); -webkit-mask: url(./sketchbook-tree-icon.svg);
-webkit-mask-position: center; -webkit-mask-position: center;
-webkit-mask-repeat: no-repeat; -webkit-mask-repeat: no-repeat;
width: 19px !important; width: 19px !important;
height: var(--theia-icon-size); height: var(--theia-icon-size);
-webkit-mask-size: 100%; -webkit-mask-size: 100%;
} }
.p-mod-current .p-mod-current .sketchbook-tree-icon {
.sketchbook-tree-icon { background-color: var(--theia-foreground);
background-color: var(--theia-foreground); -webkit-mask: url(./sketchbook-tree-icon-filled.svg);
-webkit-mask: url(./sketchbook-tree-icon-filled.svg); -webkit-mask-position: center;
-webkit-mask-position: center; -webkit-mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat; -webkit-mask-size: 100%;
-webkit-mask-size: 100%;
} }
.sketchbook-trees-container { .sketchbook-trees-container {
height: 100%; height: 100%;
} }
.sketchbook-trees-container .create-new { .sketchbook-trees-container .create-new {
min-height: 58px; min-height: 58px;
height: 58px; height: 58px;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
} }
/* /*
By default, theia-button has a left-margin. IDE2 does not need the left margin By default, theia-button has a left-margin. IDE2 does not need the left margin
@@ -39,44 +38,43 @@ for the _New Remote? Sketch_. Otherwise, the button does not fit the default
widget width. widget width.
*/ */
.sketchbook-trees-container .create-new .theia-button { .sketchbook-trees-container .create-new .theia-button {
margin-left: unset; margin-left: unset;
} }
.sketchbook-tree__opts { .sketchbook-tree__opts {
background-color: var(--theia-foreground); background-color: var(--theia-foreground);
-webkit-mask: url(./sketchbook-opts-icon.svg); -webkit-mask: url(./sketchbook-opts-icon.svg);
-webkit-mask-position: center; -webkit-mask-position: center;
-webkit-mask-repeat: no-repeat; -webkit-mask-repeat: no-repeat;
width: var(--theia-icon-size); width: var(--theia-icon-size);
height: var(--theia-icon-size); height: var(--theia-icon-size);
} }
.active-sketch { .active-sketch {
font-weight: 500; font-weight: 500;
background-color: var(--theia-list-activeSelectionBackground) !important; background-color: var(--theia-list-activeSelectionBackground) !important;
color: var(--theia-list-activeSelectionForeground) !important; color: var(--theia-list-activeSelectionForeground) !important;
} }
#arduino-sketchbook-tree-widget .theia-TreeNode { #arduino-sketchbook-tree-widget .theia-TreeNode {
line-height: 30px; line-height: 30px;
} }
#arduino-sketchbook-tree-widget .theia-TreeNodeSegmentGrow { #arduino-sketchbook-tree-widget .theia-TreeNodeSegmentGrow {
flex: 1; flex: 1;
} }
.theia-TreeNode .sketchbook-commands-icons { .theia-TreeNode .sketchbook-commands-icons {
display: none; display: none;
} }
.theia-TreeNode:hover .sketchbook-commands-icons, .theia-TreeNode:hover .sketchbook-commands-icons,
.theia-TreeNode.theia-mod-selected .sketchbook-commands-icons { .theia-TreeNode.theia-mod-selected .sketchbook-commands-icons {
display: block; display: block;
} }
.theia-Tree:focus .theia-TreeNode.theia-mod-selected, .theia-Tree:focus .theia-TreeNode.theia-mod-selected,
.theia-Tree .ReactVirtualized__List:focus .theia-TreeNode.theia-mod-selected { .theia-Tree .ReactVirtualized__List:focus .theia-TreeNode.theia-mod-selected {
background: var(--theia-list-inactiveSelectionBackground); background: var(--theia-list-inactiveSelectionBackground);
color: var(--theia-list-inactiveSelectionForeground) !important; color: var(--theia-list-inactiveSelectionForeground) !important;
} }

View File

@@ -1,32 +1,32 @@
.user-fields-container { .user-fields-container {
max-height: 332px; max-height: 332px;
overflow: auto; overflow: auto;
padding: 2px; padding: 2px;
} }
.user-fields-list { .user-fields-list {
margin: 16px 0; margin: 16px 0;
} }
.user-fields-dialog-content { .user-fields-dialog-content {
width: 408px; width: 408px;
max-height: 491px; max-height: 491px;
} }
.user-fields-dialog-content .field-label { .user-fields-dialog-content .field-label {
color: var(--theia-editorWidget-foreground); color: var(--theia-editorWidget-foreground);
font-size: 14px; font-size: 14px;
font-style: normal; font-style: normal;
font-weight: 400; font-weight: 400;
line-height: 21px; line-height: 21px;
letter-spacing: 0.01em; letter-spacing: 0.01em;
text-align: left; text-align: left;
} }
.user-fields-dialog-content .theia-input { .user-fields-dialog-content .theia-input {
flex-grow: 1; flex-grow: 1;
} }
.user-fields-dialog-content .button-container { .user-fields-dialog-content .button-container {
justify-content: flex-end; justify-content: flex-end;
} }

View File

@@ -1,13 +1,13 @@
import { injectable } from '@theia/core/shared/inversify';
import { MenuModelRegistry } from '@theia/core/lib/common/menu';
import { import {
CommonFrontendContribution as TheiaCommonFrontendContribution,
CommonCommands, CommonCommands,
CommonFrontendContribution as TheiaCommonFrontendContribution,
} from '@theia/core/lib/browser/common-frontend-contribution'; } from '@theia/core/lib/browser/common-frontend-contribution';
import { CommandRegistry } from '@theia/core/lib/common/command';
import type { OnWillStopAction } from '@theia/core/lib/browser/frontend-application'; import type { OnWillStopAction } from '@theia/core/lib/browser/frontend-application';
import { KeybindingRegistry } from '@theia/core/lib/browser'; import type { KeybindingRegistry } from '@theia/core/lib/browser/keybinding';
import { isOSX } from '@theia/core'; import type { CommandRegistry } from '@theia/core/lib/common/command';
import type { MenuModelRegistry } from '@theia/core/lib/common/menu';
import { isOSX } from '@theia/core/lib/common/os';
import { injectable } from '@theia/core/shared/inversify';
@injectable() @injectable()
export class CommonFrontendContribution extends TheiaCommonFrontendContribution { export class CommonFrontendContribution extends TheiaCommonFrontendContribution {
@@ -25,6 +25,7 @@ export class CommonFrontendContribution extends TheiaCommonFrontendContribution
CommonCommands.PIN_TAB, CommonCommands.PIN_TAB,
CommonCommands.UNPIN_TAB, CommonCommands.UNPIN_TAB,
CommonCommands.NEW_UNTITLED_FILE, CommonCommands.NEW_UNTITLED_FILE,
CommonCommands.NEW_UNTITLED_TEXT_FILE,
]) { ]) {
commandRegistry.unregisterCommand(command); commandRegistry.unregisterCommand(command);
} }
@@ -48,6 +49,7 @@ export class CommonFrontendContribution extends TheiaCommonFrontendContribution
CommonCommands.ABOUT_COMMAND, CommonCommands.ABOUT_COMMAND,
CommonCommands.SAVE_WITHOUT_FORMATTING, // Patched for https://github.com/eclipse-theia/theia/pull/8877, CommonCommands.SAVE_WITHOUT_FORMATTING, // Patched for https://github.com/eclipse-theia/theia/pull/8877,
CommonCommands.NEW_UNTITLED_FILE, CommonCommands.NEW_UNTITLED_FILE,
CommonCommands.NEW_UNTITLED_TEXT_FILE,
]) { ]) {
registry.unregisterMenuAction(command); registry.unregisterMenuAction(command);
} }

View File

@@ -74,8 +74,8 @@ export class DaemonPort implements FrontendApplicationContribution {
@inject(NotificationCenter) @inject(NotificationCenter)
private readonly notificationCenter: NotificationCenter; private readonly notificationCenter: NotificationCenter;
private readonly onPortDidChangeEmitter = new Emitter<string | undefined>(); private readonly onPortDidChangeEmitter = new Emitter<number | undefined>();
private _port: string | undefined; private _port: number | undefined;
onStart(): void { onStart(): void {
this.daemon.tryGetPort().then( this.daemon.tryGetPort().then(
@@ -91,15 +91,15 @@ export class DaemonPort implements FrontendApplicationContribution {
this.onPortDidChangeEmitter.dispose(); this.onPortDidChangeEmitter.dispose();
} }
get port(): string | undefined { get port(): number | undefined {
return this._port; return this._port;
} }
get onDidChangePort(): Event<string | undefined> { get onDidChangePort(): Event<number | undefined> {
return this.onPortDidChangeEmitter.event; return this.onPortDidChangeEmitter.event;
} }
private setPort(port: string | undefined): void { private setPort(port: number | undefined): void {
const oldPort = this._port; const oldPort = this._port;
this._port = port; this._port = port;
if (this._port !== oldPort) { if (this._port !== oldPort) {

View File

@@ -1,4 +1,8 @@
import { inject, injectable, postConstruct } from '@theia/core/shared/inversify'; import {
inject,
injectable,
postConstruct,
} from '@theia/core/shared/inversify';
import URI from '@theia/core/lib/common/uri'; import URI from '@theia/core/lib/common/uri';
import { Title, Widget } from '@theia/core/shared/@phosphor/widgets'; import { Title, Widget } from '@theia/core/shared/@phosphor/widgets';
import { EditorWidget } from '@theia/editor/lib/browser'; import { EditorWidget } from '@theia/editor/lib/browser';

View File

@@ -1,44 +1,44 @@
import debounce from 'p-debounce';
import { inject, injectable } from '@theia/core/shared/inversify';
import URI from '@theia/core/lib/common/uri';
import { Event, Emitter } from '@theia/core/lib/common/event';
import { FrontendApplicationStateService } from '@theia/core/lib/browser/frontend-application-state'; import { FrontendApplicationStateService } from '@theia/core/lib/browser/frontend-application-state';
import { DebugConfiguration } from '@theia/debug/lib/common/debug-common'; import { Disposable } from '@theia/core/lib/common/disposable';
import { DebugConfigurationModel as TheiaDebugConfigurationModel } from '@theia/debug/lib/browser/debug-configuration-model'; import { Emitter, Event } from '@theia/core/lib/common/event';
import URI from '@theia/core/lib/common/uri';
import { inject, injectable } from '@theia/core/shared/inversify';
import { DebugConfigurationManager as TheiaDebugConfigurationManager } from '@theia/debug/lib/browser/debug-configuration-manager'; import { DebugConfigurationManager as TheiaDebugConfigurationManager } from '@theia/debug/lib/browser/debug-configuration-manager';
import { DebugConfigurationModel as TheiaDebugConfigurationModel } from '@theia/debug/lib/browser/debug-configuration-model';
import { DebugConfiguration } from '@theia/debug/lib/common/debug-common';
import { FileService } from '@theia/filesystem/lib/browser/file-service';
import {
FileOperationError,
FileOperationResult,
} from '@theia/filesystem/lib/common/files';
import debounce from 'p-debounce';
import { SketchesService } from '../../../common/protocol'; import { SketchesService } from '../../../common/protocol';
import { import {
CurrentSketch, CurrentSketch,
SketchesServiceClientImpl, SketchesServiceClientImpl,
} from '../../sketches-service-client-impl'; } from '../../sketches-service-client-impl';
import { maybeUpdateReadOnlyState } from '../monaco/monaco-editor-provider';
import { DebugConfigurationModel } from './debug-configuration-model'; import { DebugConfigurationModel } from './debug-configuration-model';
import {
FileOperationError,
FileOperationResult,
} from '@theia/filesystem/lib/common/files';
import { FileService } from '@theia/filesystem/lib/browser/file-service';
@injectable() @injectable()
export class DebugConfigurationManager extends TheiaDebugConfigurationManager { export class DebugConfigurationManager extends TheiaDebugConfigurationManager {
@inject(SketchesService) @inject(SketchesService)
protected readonly sketchesService: SketchesService; private readonly sketchesService: SketchesService;
@inject(SketchesServiceClientImpl) @inject(SketchesServiceClientImpl)
protected readonly sketchesServiceClient: SketchesServiceClientImpl; private readonly sketchesServiceClient: SketchesServiceClientImpl;
@inject(FrontendApplicationStateService) @inject(FrontendApplicationStateService)
protected readonly appStateService: FrontendApplicationStateService; private readonly appStateService: FrontendApplicationStateService;
@inject(FileService) @inject(FileService)
protected readonly fileService: FileService; private readonly fileService: FileService;
protected onTempContentDidChangeEmitter = private onTempContentDidChangeEmitter =
new Emitter<TheiaDebugConfigurationModel.JsonContent>(); new Emitter<TheiaDebugConfigurationModel.JsonContent>();
get onTempContentDidChange(): Event<TheiaDebugConfigurationModel.JsonContent> { get onTempContentDidChange(): Event<TheiaDebugConfigurationModel.JsonContent> {
return this.onTempContentDidChangeEmitter.event; return this.onTempContentDidChangeEmitter.event;
} }
protected override async doInit(): Promise<void> { protected override async doInit(): Promise<void> {
this.watchLaunchConfigEditor();
this.appStateService.reachedState('ready').then(async () => { this.appStateService.reachedState('ready').then(async () => {
const tempContent = await this.getTempLaunchJsonContent(); const tempContent = await this.getTempLaunchJsonContent();
if (!tempContent) { if (!tempContent) {
@@ -75,6 +75,19 @@ export class DebugConfigurationManager extends TheiaDebugConfigurationManager {
return super.doInit(); return super.doInit();
} }
/**
* Sets a listener on current sketch change, and maybe updates the readonly state of the editor showing the debug configuration. aka the `launch.json`.
*/
private watchLaunchConfigEditor(): Disposable {
return this.sketchesServiceClient.onCurrentSketchDidChange(() => {
for (const widget of this.editorManager.all) {
maybeUpdateReadOnlyState(widget, (uri) =>
this.sketchesServiceClient.isReadOnly(uri)
);
}
});
}
protected override updateModels = debounce(async () => { protected override updateModels = debounce(async () => {
await this.appStateService.reachedState('ready'); await this.appStateService.reachedState('ready');
const roots = await this.workspaceService.roots; const roots = await this.workspaceService.roots;
@@ -111,7 +124,7 @@ export class DebugConfigurationManager extends TheiaDebugConfigurationManager {
this.updateCurrent(); this.updateCurrent();
}, 500); }, 500);
protected async getTempLaunchJsonContent(): Promise< private async getTempLaunchJsonContent(): Promise<
(TheiaDebugConfigurationModel.JsonContent & { uri: URI }) | URI | undefined (TheiaDebugConfigurationModel.JsonContent & { uri: URI }) | URI | undefined
> { > {
const sketch = await this.sketchesServiceClient.currentSketch(); const sketch = await this.sketchesServiceClient.currentSketch();

View File

@@ -0,0 +1,76 @@
import { SelectOption } from '@theia/core/lib/browser/widgets/select-component';
import { DisposableCollection } from '@theia/core/lib/common/disposable';
import { nls } from '@theia/core/lib/common/nls';
import { injectable } from '@theia/core/shared/inversify';
import React from '@theia/core/shared/react';
import { DebugAction } from '@theia/debug/lib/browser/view/debug-action';
import { DebugConfigurationSelect as TheiaDebugConfigurationSelect } from '@theia/debug/lib/browser/view/debug-configuration-select';
import { DebugConfigurationWidget as TheiaDebugConfigurationWidget } from '@theia/debug/lib/browser/view/debug-configuration-widget';
/**
* Patched to programmatically update the debug config <select> in the widget.
*/
@injectable()
export class DebugConfigurationWidget extends TheiaDebugConfigurationWidget {
override render(): React.ReactNode {
return (
<React.Fragment>
<DebugAction
run={this.start}
label={nls.localizeByDefault('Start Debugging')}
iconClass="debug-start"
ref={this.setStepRef}
/>
{/* The customized select component that will refresh when the config manager did change */}
<DebugConfigurationSelect
manager={this.manager}
quickInputService={this.quickInputService}
isMultiRoot={this.workspaceService.isMultiRootWorkspaceOpened}
/>
<DebugAction
run={this.openConfiguration}
label={nls.localizeByDefault('Open {0}', '"launch.json"')}
iconClass="settings-gear"
/>
<DebugAction
run={this.openConsole}
label={nls.localizeByDefault('Debug Console')}
iconClass="terminal"
/>
</React.Fragment>
);
}
}
class DebugConfigurationSelect extends TheiaDebugConfigurationSelect {
private readonly toDisposeOnUnmount = new DisposableCollection();
override componentDidMount(): void {
super.componentDidMount();
this.toDisposeOnUnmount.push(
this['manager'].onDidChange(() => this.refreshDebugConfigurations())
);
}
protected override renderOptions(): SelectOption[] {
const options = super.renderOptions();
const addConfiguration = options[options.length - 1];
const separator = options[options.length - 2];
// Remove "Add configuration..." and the preceding separator options.
// They're expected to be the last two items.
if (
addConfiguration.value ===
TheiaDebugConfigurationSelect.ADD_CONFIGURATION &&
separator.separator
) {
options.splice(options.length - 2, 2);
return options;
}
// Something is unexpected with the select options.
return options;
}
override componentWillUnmount(): void {
this.toDisposeOnUnmount.dispose();
}
}

View File

@@ -0,0 +1,39 @@
import { inject, injectable } from '@theia/core/shared/inversify';
import { DebugSession } from '@theia/debug/lib/browser/debug-session';
import { DebugSessionManager as TheiaDebugSessionManager } from '@theia/debug/lib/browser/debug-session-manager';
import { DebugConfigurationSessionOptions } from '@theia/debug/lib/browser/debug-session-options';
import { WorkspaceService } from '@theia/workspace/lib/browser/workspace-service';
import deepEqual from 'fast-deep-equal';
@injectable()
export class DebugSessionManager extends TheiaDebugSessionManager {
@inject(WorkspaceService)
private readonly workspaceService: WorkspaceService;
protected override doStart(
sessionId: string,
options: DebugConfigurationSessionOptions
): Promise<DebugSession> {
this.syncCurrentOptions(options);
return super.doStart(sessionId, options);
}
/**
* If the debug config manager knows about the currently started options, and it's not the currently selected one, select it.
*/
private syncCurrentOptions(options: DebugConfigurationSessionOptions): void {
const knownConfigOptions = this.debugConfigurationManager.find(
options.configuration,
options.workspaceFolderUri ??
this.workspaceService
.tryGetRoots()
.map((stat) => stat.resource.toString())[0]
);
if (
knownConfigOptions &&
!deepEqual(knownConfigOptions, this.debugConfigurationManager.current)
) {
this.debugConfigurationManager.current = knownConfigOptions;
}
}
}

View File

@@ -0,0 +1,56 @@
import { codicon } from '@theia/core/lib/browser/widgets/widget';
import { Widget } from '@theia/core/shared/@phosphor/widgets';
import {
inject,
injectable,
postConstruct,
} from '@theia/core/shared/inversify';
import { DebugWidget as TheiaDebugWidget } from '@theia/debug/lib/browser/view/debug-widget';
import { DebugDisabledStatusMessageSource } from '../../contributions/debug';
import {
removeWidgetIfPresent,
unshiftWidgetIfNotPresent,
} from '../dialogs/widgets';
@injectable()
export class DebugWidget extends TheiaDebugWidget {
@inject(DebugDisabledStatusMessageSource)
private readonly debugStatusMessageSource: DebugDisabledStatusMessageSource;
private readonly statusMessageWidget = new Widget();
private readonly messageNode = document.createElement('div');
@postConstruct()
protected override init(): void {
super.init();
this.messageNode.classList.add('status-message', 'noselect');
this.statusMessageWidget.node.appendChild(this.messageNode);
this.updateState();
this.toDisposeOnDetach.pushAll([
this.debugStatusMessageSource.onDidChangeMessage((message) =>
this.updateState(message)
),
this.statusMessageWidget,
]);
}
private updateState(message = this.debugStatusMessageSource.message): void {
requestAnimationFrame(() => {
this.messageNode.textContent = message ?? '';
const enabled = !message;
updateVisibility(enabled, this.toolbar, this.sessionWidget);
if (enabled) {
removeWidgetIfPresent(this.layout, this.statusMessageWidget);
} else {
unshiftWidgetIfNotPresent(this.layout, this.statusMessageWidget);
}
this.title.iconClass = enabled ? codicon('debug-alt') : 'fa fa-ban'; // TODO: find a better icon?
});
}
}
function updateVisibility(visible: boolean, ...widgets: Widget[]): void {
widgets.forEach((widget) =>
visible ? widget.removeClass('hidden') : widget.addClass('hidden')
);
}

View File

@@ -0,0 +1,51 @@
import {
Layout,
PanelLayout,
Widget,
} from '@theia/core/shared/@phosphor/widgets';
/**
*
* Removes the widget from the layout if the `layout` is a `PanelLayout` and the widget is present in the layout.
* Otherwise, it's NOOP
* @param layout the layout to remove the widget from. Must be a `PanelLayout`.
* @param toRemove the widget to remove from the layout
*/
export function removeWidgetIfPresent(
layout: Layout | null,
toRemove: Widget
): void {
if (layout instanceof PanelLayout) {
const index = layout.widgets.indexOf(toRemove);
if (index < 0) {
// Unlike the default `PanelLayout#removeWidget` behavior, (https://github.com/phosphorjs/phosphor/blob/9f5e11025b62d2c4a6fb59e2681ae1ed323dcde4/packages/widgets/src/panellayout.ts#L154-L156)
// do not try to remove widget if it's not present (the index is negative).
// Otherwise, required widgets could be removed based on the default ArrayExt behavior (https://github.com/phosphorjs/phosphor/blob/9f5e11025b62d2c4a6fb59e2681ae1ed323dcde4/packages/algorithm/src/array.ts#L1075-L1077)
// See https://github.com/arduino/arduino-ide/issues/2354 for more details.
return;
}
layout.removeWidget(toRemove);
}
}
/**
*
* Inserts the widget to the `0` index of the layout if the `layout` is a `PanelLayout` and the widget is not yet part of the layout.
* Otherwise, it's NOOP
* @param layout the layout to add the widget to. Must be a `PanelLayout`.
* @param toAdd the widget to add to the layout
*/
export function unshiftWidgetIfNotPresent(
layout: Layout | null,
toAdd: Widget
): void {
if (layout instanceof PanelLayout) {
const index = layout.widgets.indexOf(toAdd);
if (index >= 0) {
// Do not try to add the widget to the layout if it's already present.
// This is the counterpart logic of the `removeWidgetIfPresent` function.
return;
}
layout.insertWidget(0, toAdd);
}
}

View File

@@ -1,15 +1,20 @@
import { inject, injectable } from '@theia/core/shared/inversify'; import { LOCKED_CLASS, lock } from '@theia/core/lib/browser/widgets/widget';
import URI from '@theia/core/lib/common/uri';
import { import {
Disposable, Disposable,
DisposableCollection, DisposableCollection,
} from '@theia/core/lib/common/disposable'; } from '@theia/core/lib/common/disposable';
import { EditorServiceOverrides, MonacoEditor } from '@theia/monaco/lib/browser/monaco-editor'; import URI from '@theia/core/lib/common/uri';
import { MonacoEditorProvider as TheiaMonacoEditorProvider } from '@theia/monaco/lib/browser/monaco-editor-provider'; import { Title, Widget } from '@theia/core/shared/@phosphor/widgets';
import { SketchesServiceClientImpl } from '../../sketches-service-client-impl'; import { inject, injectable } from '@theia/core/shared/inversify';
import { EditorWidget } from '@theia/editor/lib/browser/editor-widget';
import * as monaco from '@theia/monaco-editor-core'; import * as monaco from '@theia/monaco-editor-core';
import type { ReferencesModel } from '@theia/monaco-editor-core/esm/vs/editor/contrib/gotoSymbol/browser/referencesModel'; import type { ReferencesModel } from '@theia/monaco-editor-core/esm/vs/editor/contrib/gotoSymbol/browser/referencesModel';
import {
EditorServiceOverrides,
MonacoEditor,
} from '@theia/monaco/lib/browser/monaco-editor';
import { MonacoEditorProvider as TheiaMonacoEditorProvider } from '@theia/monaco/lib/browser/monaco-editor-provider';
import { SketchesServiceClientImpl } from '../../sketches-service-client-impl';
type CancelablePromise = Promise<ReferencesModel> & { type CancelablePromise = Promise<ReferencesModel> & {
cancel: () => void; cancel: () => void;
@@ -39,7 +44,9 @@ export class MonacoEditorProvider extends TheiaMonacoEditorProvider {
private installCustomReferencesController(editor: MonacoEditor): Disposable { private installCustomReferencesController(editor: MonacoEditor): Disposable {
const control = editor.getControl(); const control = editor.getControl();
const referencesController: any = control.getContribution('editor.contrib.referencesController'); const referencesController: any = control.getContribution(
'editor.contrib.referencesController'
);
const originalToggleWidget = referencesController.toggleWidget; const originalToggleWidget = referencesController.toggleWidget;
const toDispose = new DisposableCollection(); const toDispose = new DisposableCollection();
const toDisposeBeforeToggleWidget = new DisposableCollection(); const toDisposeBeforeToggleWidget = new DisposableCollection();
@@ -97,3 +104,30 @@ export class MonacoEditorProvider extends TheiaMonacoEditorProvider {
editor.updateOptions({ readOnly }); editor.updateOptions({ readOnly });
} }
} }
// Theia cannot dynamically set an editor to writable once it was readonly.
export function maybeUpdateReadOnlyState(
widget: EditorWidget,
isReadOnly: (uri: string | URI | monaco.Uri) => boolean
): void {
const editor = widget.editor;
if (!(editor instanceof MonacoEditor)) {
return;
}
const model = editor.document;
const oldReadOnly = model.readOnly;
const resource = model['resource'];
const newReadOnly = Boolean(resource.isReadonly) || isReadOnly(resource.uri);
if (oldReadOnly !== newReadOnly) {
editor.getControl().updateOptions({ readOnly: newReadOnly });
if (newReadOnly) {
lock(widget.title);
} else {
unlock(widget.title);
}
}
}
function unlock(title: Title<Widget>): void {
title.className = title.className.replace(LOCKED_CLASS, '').trim();
}

View File

@@ -19,7 +19,9 @@ export class MonacoTextModelService extends TheiaMonacoTextModelService {
const factory = this.factories const factory = this.factories
.getContributions() .getContributions()
.find(({ scheme }) => resource.uri.scheme === scheme); .find(({ scheme }) => resource.uri.scheme === scheme);
const readOnly = this.sketchesServiceClient.isReadOnly(resource.uri); const readOnly =
Boolean(resource.isReadonly) ||
this.sketchesServiceClient.isReadOnly(resource.uri);
return factory return factory
? factory.createModel(resource) ? factory.createModel(resource)
: new MaybeReadonlyMonacoEditorModel( : new MaybeReadonlyMonacoEditorModel(
@@ -75,7 +77,7 @@ class MaybeReadonlyMonacoEditorModel extends SilentMonacoEditorModel {
} }
this._dirty = dirty; this._dirty = dirty;
if (dirty === false) { if (dirty === false) {
(this as any).updateSavedVersionId(); this['updateSavedVersionId']();
} }
this.onDirtyChangedEmitter.fire(undefined); this.onDirtyChangedEmitter.fire(undefined);
} }

View File

@@ -20,7 +20,7 @@ import {
} from '@theia/monaco/lib/browser/monaco-theming-service'; } from '@theia/monaco/lib/browser/monaco-theming-service';
import { MonacoThemeRegistry as TheiaMonacoThemeRegistry } from '@theia/monaco/lib/browser/textmate/monaco-theme-registry'; import { MonacoThemeRegistry as TheiaMonacoThemeRegistry } from '@theia/monaco/lib/browser/textmate/monaco-theme-registry';
import type { ThemeMix } from '@theia/monaco/lib/browser/textmate/monaco-theme-types'; import type { ThemeMix } from '@theia/monaco/lib/browser/textmate/monaco-theme-types';
import { HostedPluginSupport } from '@theia/plugin-ext/lib/hosted/browser/hosted-plugin'; import { HostedPluginSupport } from '../../hosted/hosted-plugin-support';
import { ArduinoThemes, compatibleBuiltInTheme } from '../core/theming'; import { ArduinoThemes, compatibleBuiltInTheme } from '../core/theming';
import { WindowServiceExt } from '../core/window-service-ext'; import { WindowServiceExt } from '../core/window-service-ext';

View File

@@ -14,7 +14,8 @@ export class FileNavigatorContribution extends TheiaFileNavigatorContribution {
constructor( constructor(
@inject(FileNavigatorPreferences) @inject(FileNavigatorPreferences)
protected override readonly fileNavigatorPreferences: FileNavigatorPreferences, protected override readonly fileNavigatorPreferences: FileNavigatorPreferences,
@inject(OpenerService) protected override readonly openerService: OpenerService, @inject(OpenerService)
protected override readonly openerService: OpenerService,
@inject(FileNavigatorFilter) @inject(FileNavigatorFilter)
protected override readonly fileNavigatorFilter: FileNavigatorFilter, protected override readonly fileNavigatorFilter: FileNavigatorFilter,
@inject(WorkspaceService) @inject(WorkspaceService)

View File

@@ -8,7 +8,9 @@ import { OutputToolbarContribution as TheiaOutputToolbarContribution } from '@th
@injectable() @injectable()
export class OutputToolbarContribution extends TheiaOutputToolbarContribution { export class OutputToolbarContribution extends TheiaOutputToolbarContribution {
override async registerToolbarItems(registry: TabBarToolbarRegistry): Promise<void> { override async registerToolbarItems(
registry: TabBarToolbarRegistry
): Promise<void> {
await super.registerToolbarItems(registry); // Why is it async? await super.registerToolbarItems(registry); // Why is it async?
// It's a hack. Currently, it's not possible to unregister a toolbar contribution via API. // It's a hack. Currently, it's not possible to unregister a toolbar contribution via API.
( (

View File

@@ -5,9 +5,13 @@ import {
PluginContributions, PluginContributions,
HostedPluginSupport as TheiaHostedPluginSupport, HostedPluginSupport as TheiaHostedPluginSupport,
} from '@theia/plugin-ext/lib/hosted/browser/hosted-plugin'; } from '@theia/plugin-ext/lib/hosted/browser/hosted-plugin';
import { HostedPluginSupport } from '../../hosted/hosted-plugin-support';
@injectable() @injectable()
export class HostedPluginSupport extends TheiaHostedPluginSupport { export class HostedPluginSupportImpl
extends TheiaHostedPluginSupport
implements HostedPluginSupport
{
private readonly onDidLoadEmitter = new Emitter<void>(); private readonly onDidLoadEmitter = new Emitter<void>();
private readonly onDidCloseConnectionEmitter = new Emitter<void>(); private readonly onDidCloseConnectionEmitter = new Emitter<void>();

View File

@@ -8,7 +8,10 @@ export class ScmContribution extends TheiaScmContribution {
// NOOP // NOOP
} }
protected override setStatusBarEntry(id: string, entry: StatusBarEntry): void { protected override setStatusBarEntry(
id: string,
entry: StatusBarEntry
): void {
// NOOP // NOOP
} }
} }

View File

@@ -1,4 +1,3 @@
import { MaybePromise } from '@theia/core';
import { Dialog, DialogError } from '@theia/core/lib/browser/dialogs'; import { Dialog, DialogError } from '@theia/core/lib/browser/dialogs';
import { LabelProvider } from '@theia/core/lib/browser/label-provider'; import { LabelProvider } from '@theia/core/lib/browser/label-provider';
import { CancellationTokenSource } from '@theia/core/lib/common/cancellation'; import { CancellationTokenSource } from '@theia/core/lib/common/cancellation';
@@ -10,13 +9,14 @@ import type {
Progress, Progress,
ProgressUpdate, ProgressUpdate,
} from '@theia/core/lib/common/message-service-protocol'; } from '@theia/core/lib/common/message-service-protocol';
import type { MaybePromise } from '@theia/core/lib/common/types';
import { UUID } from '@theia/core/shared/@phosphor/coreutils';
import { Widget } from '@theia/core/shared/@phosphor/widgets'; import { Widget } from '@theia/core/shared/@phosphor/widgets';
import { inject } from '@theia/core/shared/inversify'; import { inject } from '@theia/core/shared/inversify';
import { import {
WorkspaceInputDialog as TheiaWorkspaceInputDialog, WorkspaceInputDialog as TheiaWorkspaceInputDialog,
WorkspaceInputDialogProps, WorkspaceInputDialogProps,
} from '@theia/workspace/lib/browser/workspace-input-dialog'; } from '@theia/workspace/lib/browser/workspace-input-dialog';
import { v4 } from 'uuid';
export class WorkspaceInputDialog extends TheiaWorkspaceInputDialog { export class WorkspaceInputDialog extends TheiaWorkspaceInputDialog {
private skipShowErrorMessageOnOpen: boolean; private skipShowErrorMessageOnOpen: boolean;
@@ -161,7 +161,7 @@ export class WorkspaceInputDialogWithProgress<
const cancellationSource = new CancellationTokenSource(); const cancellationSource = new CancellationTokenSource();
const progress: Progress = { const progress: Progress = {
id: v4(), id: UUID.uuid4(),
cancel: () => cancellationSource.cancel(), cancel: () => cancellationSource.cancel(),
report: (update: ProgressUpdate) => { report: (update: ProgressUpdate) => {
this.setProgressMessage(update); this.setProgressMessage(update);

Some files were not shown because too many files have changed in this diff Show More